Codebase list virt-viewer / 07c89c3
Initial commit Daniel P. Berrange 16 years ago
13 changed file(s) with 840 addition(s) and 0 deletion(s). Raw diff Collapse all Expand all
0 .*~$
1 ^build/
2 ^aclocal\.m4$
3 ^autom4te\.cache/
4 Makefile\.in$
5 ^config\.guess$
6 ^config\.sub$
7 ^configure$
(New empty file)
(New empty file)
0
1 SUBDIRS = src
2
3 EXTRA_DIST = @PACKAGE@.spec
4
5 DISTCLEAN_FILES = @PACKAGE@.spec
(New empty file)
(New empty file)
0 dnl
1 dnl Taken from gnome-common/macros2/gnome-compiler-flags.m4
2 dnl
3 dnl We've added:
4 dnl -Wextra -Wshadow -Wcast-align -Wwrite-strings -Waggregate-return -Wstrict-prototypes -Winline -Wredundant-decls
5 dnl We've removed
6 dnl CFLAGS="$realsave_CFLAGS"
7 dnl to avoid clobbering user-specified CFLAGS
8 dnl
9 AC_DEFUN([VIRT_VIEWER_COMPILE_WARNINGS],[
10 dnl ******************************
11 dnl More compiler warnings
12 dnl ******************************
13
14 AC_ARG_ENABLE(compile-warnings,
15 AC_HELP_STRING([--enable-compile-warnings=@<:@no/minimum/yes/maximum/error@:>@],
16 [Turn on compiler warnings]),,
17 [enable_compile_warnings="m4_default([$1],[maximum])"])
18
19 warnCFLAGS=
20
21 try_compiler_flags="-Wp,-D_FORTIFY_SOURCE=2 -fexceptions -fstack-protector --param=ssp-buffer-size=4 -fasynchronous-unwind-tables"
22
23 case "$enable_compile_warnings" in
24 no)
25 ;;
26 minimum)
27 try_compiler_flags="$try_compiler_flags -Wall"
28 ;;
29 yes)
30 try_compiler_flags="$try_compiler_flags -Wall -Wmissing-prototypes -std=c99"
31 ;;
32 maximum|error)
33 try_compiler_flags="$try_compiler_flags -Wall -Wmissing-prototypes -std=c99 -Wnested-externs -Wpointer-arith"
34 try_compiler_flags="$try_compiler_flags -Wextra -Wshadow -Wcast-align -Wwrite-strings -Waggregate-return"
35 try_compiler_flags="$try_compiler_flags -Wstrict-prototypes -Winline -Wredundant-decls -Wno-sign-compare"
36 if test "$enable_compile_warnings" = "error" ; then
37 try_compiler_flags="$try_compiler_flags -Werror"
38 fi
39 ;;
40 *)
41 AC_MSG_ERROR(Unknown argument '$enable_compile_warnings' to --enable-compile-warnings)
42 ;;
43 esac
44
45 compiler_flags=
46 for option in $try_compiler_flags; do
47 SAVE_CFLAGS="$CFLAGS"
48 CFLAGS="$CFLAGS $option"
49 AC_MSG_CHECKING([whether gcc understands $option])
50 AC_TRY_COMPILE([], [],
51 has_option=yes,
52 has_option=no,)
53 CFLAGS="$SAVE_CFLAGS"
54 AC_MSG_RESULT($has_option)
55 if test $has_option = yes; then
56 compiler_flags="$compiler_flags $option"
57 fi
58 unset has_option
59 unset SAVE_CFLAGS
60 done
61 unset option
62 unset try_compiler_flags
63
64 AC_ARG_ENABLE(iso-c,
65 AC_HELP_STRING([--enable-iso-c],
66 [Try to warn if code is not ISO C ]),,
67 [enable_iso_c=no])
68
69 AC_MSG_CHECKING(what language compliance flags to pass to the C compiler)
70 complCFLAGS=
71 if test "x$enable_iso_c" != "xno"; then
72 if test "x$GCC" = "xyes"; then
73 case " $CFLAGS " in
74 *[\ \ ]-ansi[\ \ ]*) ;;
75 *) complCFLAGS="$complCFLAGS -ansi" ;;
76 esac
77 case " $CFLAGS " in
78 *[\ \ ]-pedantic[\ \ ]*) ;;
79 *) complCFLAGS="$complCFLAGS -pedantic" ;;
80 esac
81 fi
82 fi
83 AC_MSG_RESULT($complCFLAGS)
84
85 WARN_CFLAGS="$compiler_flags $complCFLAGS"
86 AC_SUBST(WARN_CFLAGS)
87 ])
0 #!/bin/sh
1
2 set -e
3 set -v
4
5 # Make things clean.
6 test -f Makefile && make -k distclean || :
7
8 rm -rf build
9 mkdir build
10 cd build
11
12 ../autogen.sh --prefix=$AUTOBUILD_INSTALL_ROOT --enable-fatal-warnings
13
14 make
15 make install
16
17 rm -f *.tar.gz
18 make dist
19
20 if [ -f /usr/bin/rpmbuild ]; then
21 if [ -n "$AUTOBUILD_COUNTER" ]; then
22 EXTRA_RELEASE=".auto$AUTOBUILD_COUNTER"
23 else
24 NOW=`date +"%s"`
25 EXTRA_RELEASE=".$USER$NOW"
26 fi
27 rpmbuild --nodeps --define "extra_release $EXTRA_RELEASE" -ta --clean *.tar.gz
28 fi
0 #!/bin/sh
1 # Run this to generate all the initial makefiles, etc.
2
3 set -e
4 srcdir=`dirname $0`
5 test -z "$srcdir" && srcdir=.
6
7 THEDIR=`pwd`
8 cd $srcdir
9
10 DIE=0
11
12 (autoconf --version) < /dev/null > /dev/null 2>&1 || {
13 echo
14 echo "You must have autoconf installed to compile virt-viewer."
15 echo "Download the appropriate package for your distribution,"
16 echo "or see http://www.gnu.org/software/autoconf"
17 DIE=1
18 }
19
20 (automake --version) < /dev/null > /dev/null 2>&1 || {
21 echo
22 DIE=1
23 echo "You must have automake installed to compile virt-viewer."
24 echo "Download the appropriate package for your distribution,"
25 echo "or see http://www.gnu.org/software/automake"
26 }
27
28 if test "$DIE" -eq 1; then
29 exit 1
30 fi
31
32 if test -z "$*"; then
33 echo "I am going to run ./configure with --enable-warnings - if you "
34 echo "wish to pass any extra arguments to it, please specify them on "
35 echo "the $0 command line."
36 fi
37
38 aclocal
39 automake --add-missing
40 autoconf
41
42 cd $THEDIR
43
44 $srcdir/configure --enable-warnings "$@" && {
45 echo
46 echo "Now type 'make' to compile gtk-vnc."
47 }
0 AC_INIT(virt-viewer.spec.in)
1 AM_INIT_AUTOMAKE(virt-viewer, 0.0.1)
2
3 AC_PROG_CC
4
5 VIRT_VIEWER_COMPILE_WARNINGS(maximum)
6
7 PKG_CHECK_MODULES(LIBXML2, libxml-2.0 >= 2.6.0)
8 PKG_CHECK_MODULES(LIBVIRT, libvirt >= 0.2.0)
9 PKG_CHECK_MODULES(GTK2, gtk+-2.0 >= 2.2.0)
10 PKG_CHECK_MODULES(GTKVNC, gtk-vnc-1.0 >= 0.0.1)
11
12 AC_OUTPUT(Makefile
13 src/Makefile
14 virt-viewer.spec)
0
1 bin_PROGRAMS = virt-viewer
2
3 virt_viewer_SOURCES = main.c
4 virt_viewer_LDADD = @GTKVNC_LIBS@ @GTK2_LIBS@ @LIBXML2_LIBS@ @LIBVIRT_LIBS@
5 virt_viewer_CFLAGS = @GTKVNC_CFLAGS@ @GTK2_CFLAGS@ @LIBXML2_CFLAGS@ @LIBVIRT_CFLAGS@ @WARN_CFLAGS@
0 /*
1 * Virt Viewer: A virtual machine console viewer
2 *
3 * Copyright (C) 2007 Red Hat,
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 *
19 * Author: Daniel P. Berrange <berrange@redhat.com>
20 */
21
22 #define _GNU_SOURCE
23
24 #include <vncdisplay.h>
25 #include <gtk/gtk.h>
26 #include <gdk/gdkkeysyms.h>
27 #include <stdlib.h>
28 #include <string.h>
29 #include <unistd.h>
30 #include <getopt.h>
31 #include <libvirt/libvirt.h>
32 #include <libxml/xpath.h>
33
34
35 #define DEBUG 1
36 #ifdef DEBUG
37 #define DEBUG_LOG(s, ...) fprintf(stderr, (s), ## __VA_ARGS__)
38 #else
39 #define DEBUG_LOG(s, ...) do {} while (0)
40 #endif
41
42 static int verbose = 0;
43 #define MAX_KEY_COMBO 3
44 struct keyComboDef {
45 guint keys[MAX_KEY_COMBO];
46 guint nkeys;
47 const char *label;
48 };
49
50 static const struct keyComboDef keyCombos[] = {
51 { { GDK_Control_L, GDK_Alt_L, GDK_Delete }, 3, "Ctrl+Alt+_Del"},
52 { { GDK_Control_L, GDK_Alt_L, GDK_BackSpace }, 3, "Ctrl+Alt+_Backspace"},
53 { { GDK_Control_L, GDK_Alt_L, GDK_F1 }, 3, "Ctrl+Alt+F_1"},
54 { { GDK_Control_L, GDK_Alt_L, GDK_F2 }, 3, "Ctrl+Alt+F_2"},
55 { { GDK_Control_L, GDK_Alt_L, GDK_F3 }, 3, "Ctrl+Alt+F_3"},
56 { { GDK_Control_L, GDK_Alt_L, GDK_F4 }, 3, "Ctrl+Alt+F_4"},
57 { { GDK_Control_L, GDK_Alt_L, GDK_F5 }, 3, "Ctrl+Alt+F_5"},
58 { { GDK_Control_L, GDK_Alt_L, GDK_F6 }, 3, "Ctrl+Alt+F_6"},
59 { { GDK_Control_L, GDK_Alt_L, GDK_F7 }, 3, "Ctrl+Alt+F_7"},
60 { { GDK_Control_L, GDK_Alt_L, GDK_F8 }, 3, "Ctrl+Alt+F_8"},
61 { { GDK_Print }, 1, "_PrintScreen"},
62 };
63
64
65 static void viewer_set_title(VncDisplay *vnc, GtkWidget *window, gboolean grabbed)
66 {
67 const char *name;
68 char title[1024];
69 const char *subtitle;
70
71 if (grabbed)
72 subtitle = "(Press Ctrl+Alt to release pointer) ";
73 else
74 subtitle = "";
75
76 name = vnc_display_get_name(VNC_DISPLAY(vnc));
77 snprintf(title, sizeof(title), "%s%s - Virt Viewer",
78 subtitle, name);
79
80 gtk_window_set_title(GTK_WINDOW(window), title);
81 }
82
83 static void viewer_grab(GtkWidget *vnc, GtkWidget *window)
84 {
85 viewer_set_title(VNC_DISPLAY(vnc), window, TRUE);
86 }
87
88 static void viewer_ungrab(GtkWidget *vnc, GtkWidget *window)
89 {
90 viewer_set_title(VNC_DISPLAY(vnc), window, FALSE);
91 }
92
93 static void viewer_shutdown(GtkWidget *src G_GNUC_UNUSED, GtkWidget *vnc)
94 {
95 vnc_display_close(VNC_DISPLAY(vnc));
96 gtk_main_quit();
97 }
98
99 static void viewer_connected(GtkWidget *vnc G_GNUC_UNUSED)
100 {
101 DEBUG_LOG("Connected to server\n");
102 }
103
104 static void viewer_initialized(GtkWidget *vnc, GtkWidget *window)
105 {
106 DEBUG_LOG("Connection initialized\n");
107 gtk_widget_show_all(window);
108 viewer_set_title(VNC_DISPLAY(vnc), window, FALSE);
109 }
110
111 static void viewer_disconnected(GtkWidget *vnc G_GNUC_UNUSED)
112 {
113 DEBUG_LOG("Disconnected from server\n");
114 gtk_main_quit();
115 }
116
117 static void viewer_send_key(GtkWidget *menu, GtkWidget *vnc)
118 {
119 int i;
120 GtkWidget *label = gtk_bin_get_child(GTK_BIN(menu));
121 const char *text = gtk_label_get_label(GTK_LABEL(label));
122
123 for (i = 0 ; i < (sizeof(keyCombos)/sizeof(keyCombos[0])) ; i++) {
124 if (!strcmp(text, keyCombos[i].label)) {
125 DEBUG_LOG("Sending key combo %s\n", gtk_label_get_text(GTK_LABEL(label)));
126 vnc_display_send_keys(VNC_DISPLAY(vnc),
127 keyCombos[i].keys,
128 keyCombos[i].nkeys);
129 return;
130 }
131 }
132 DEBUG_LOG("Failed to find key combo %s\n", gtk_label_get_text(GTK_LABEL(label)));
133 }
134
135 #if 0
136 static void viewer_save_screenshot(GtkWidget *vnc, const char *file)
137 {
138 GdkPixbuf *pix = vnc_display_get_pixbuf(VNC_DISPLAY(vnc));
139 gdk_pixbuf_save(pix, file, "png", NULL,
140 "tEXt::Generator App", PACKAGE, NULL);
141 gdk_pixbuf_unref(pix);
142 }
143
144 static void viewer_screenshot(GtkWidget *menu, GtkWidget *vnc)
145 {
146 GtkWidget *dialog;
147
148 dialog = gtk_file_chooser_dialog_new ("Save screenshot",
149 NULL,
150 GTK_FILE_CHOOSER_ACTION_SAVE,
151 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
152 GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
153 NULL);
154 gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog), TRUE);
155
156 //gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (dialog), default_folder_for_saving);
157 //gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), "Screenshot");
158
159 if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT) {
160 char *filename;
161
162 filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
163 viewer_save_screenshot(vnc, filename);
164 g_free (filename);
165 }
166
167 gtk_widget_destroy (dialog);
168 }
169 #endif
170
171 static void viewer_credential(GtkWidget *vnc, GValueArray *credList)
172 {
173 GtkWidget *dialog, **label, **entry, *box, *vbox;
174 int response;
175 unsigned int i;
176
177 DEBUG_LOG("Got credential request for %d credential(s)\n", credList->n_values);
178
179 dialog = gtk_dialog_new_with_buttons("Authentication required",
180 NULL,
181 0,
182 GTK_STOCK_CANCEL,
183 GTK_RESPONSE_CANCEL,
184 GTK_STOCK_OK,
185 GTK_RESPONSE_OK,
186 NULL);
187 gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_OK);
188
189 box = gtk_table_new(credList->n_values, 2, FALSE);
190 label = g_new(GtkWidget *, credList->n_values);
191 entry = g_new(GtkWidget *, credList->n_values);
192
193 for (i = 0 ; i < credList->n_values ; i++) {
194 GValue *cred = g_value_array_get_nth(credList, i);
195 int credType = g_value_get_enum(cred);
196 switch (credType) {
197 case VNC_DISPLAY_CREDENTIAL_USERNAME:
198 label[i] = gtk_label_new("Username:");
199 break;
200 default:
201 label[i] = gtk_label_new("Password:");
202 break;
203 }
204 entry[i] = gtk_entry_new();
205
206 gtk_table_attach(GTK_TABLE(box), label[i], 0, 1, i, i+1, GTK_SHRINK, GTK_SHRINK, 3, 3);
207 gtk_table_attach(GTK_TABLE(box), entry[i], 1, 2, i, i+1, GTK_SHRINK, GTK_SHRINK, 3, 3);
208 }
209
210
211 vbox = gtk_bin_get_child(GTK_BIN(dialog));
212
213 gtk_container_add(GTK_CONTAINER(vbox), box);
214
215 gtk_widget_show_all(dialog);
216 response = gtk_dialog_run(GTK_DIALOG(dialog));
217 gtk_widget_hide(GTK_WIDGET(dialog));
218
219 if (response == GTK_RESPONSE_OK) {
220 for (i = 0 ; i < credList->n_values ; i++) {
221 GValue *cred = g_value_array_get_nth(credList, i);
222 int credType = g_value_get_enum(cred);
223 const char *data = gtk_entry_get_text(GTK_ENTRY(entry[i]));
224 vnc_display_set_credential(VNC_DISPLAY(vnc), credType, data);
225 }
226 } else {
227 DEBUG_LOG("Aborting connection\n");
228 vnc_display_close(VNC_DISPLAY(vnc));
229 }
230
231 gtk_widget_destroy(GTK_WIDGET(dialog));
232 }
233
234 static GtkWidget *viewer_build_file_menu(VncDisplay *vnc)
235 {
236 GtkWidget *file;
237 GtkWidget *filemenu;
238 GtkWidget *quit;
239
240 file = gtk_menu_item_new_with_mnemonic("_File");
241
242 filemenu = gtk_menu_new();
243 gtk_menu_item_set_submenu(GTK_MENU_ITEM(file), filemenu);
244
245 quit = gtk_image_menu_item_new_from_stock(GTK_STOCK_QUIT, NULL);
246 gtk_menu_append(GTK_MENU(filemenu), quit);
247 g_signal_connect(quit, "activate", GTK_SIGNAL_FUNC(viewer_shutdown), vnc);
248
249 return file;
250 }
251
252 static GtkWidget *viewer_build_sendkey_menu(VncDisplay *vnc)
253 {
254 GtkWidget *sendkey;
255 GtkWidget *sendkeymenu;
256 int i;
257
258 sendkey = gtk_menu_item_new_with_mnemonic("_Send Key");
259
260 sendkeymenu = gtk_menu_new();
261 gtk_menu_item_set_submenu(GTK_MENU_ITEM(sendkey), sendkeymenu);
262
263 for (i = 0 ; i < (sizeof(keyCombos)/sizeof(keyCombos[0])) ; i++) {
264 GtkWidget *key;
265
266 key = gtk_menu_item_new_with_mnemonic(keyCombos[i].label);
267 gtk_menu_append(GTK_MENU(sendkeymenu), key);
268 g_signal_connect(key, "activate", GTK_SIGNAL_FUNC(viewer_send_key), vnc);
269 }
270
271 return sendkey;
272 }
273
274 static GtkWidget *viewer_build_help_menu(void)
275 {
276 GtkWidget *help;
277 GtkWidget *helpmenu;
278 GtkWidget *about;
279
280 help = gtk_menu_item_new_with_mnemonic("_Help");
281
282 helpmenu = gtk_menu_new();
283 gtk_menu_item_set_submenu(GTK_MENU_ITEM(help), helpmenu);
284
285 about = gtk_image_menu_item_new_from_stock(GTK_STOCK_ABOUT, NULL);
286 gtk_menu_append(GTK_MENU(helpmenu), about);
287 //g_signal_connect(about, "activate", GTK_SIGNAL_FUNC(viewer_shutdown), vnc);
288
289 return help;
290 }
291
292 static GtkWidget *viewer_build_menu(VncDisplay *vnc)
293 {
294 GtkWidget *menubar;
295 GtkWidget *file;
296 GtkWidget *sendkey;
297 GtkWidget *help;
298
299 menubar = gtk_menu_bar_new();
300
301 file = viewer_build_file_menu(vnc);
302 sendkey = viewer_build_sendkey_menu(vnc);
303 help = viewer_build_help_menu();
304
305 gtk_menu_bar_append(GTK_MENU_BAR(menubar), file);
306 gtk_menu_bar_append(GTK_MENU_BAR(menubar), sendkey);
307 gtk_menu_bar_append(GTK_MENU_BAR(menubar), help);
308
309
310 return menubar;
311 }
312
313 static GtkWidget *viewer_build_window(VncDisplay *vnc)
314 {
315 GtkWidget *window;
316 GtkWidget *menubar;
317 GtkWidget *layout;
318
319 window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
320 layout = gtk_vbox_new(FALSE, 3);
321 menubar = viewer_build_menu(vnc);
322
323 gtk_container_add(GTK_CONTAINER(window), layout);
324 gtk_container_add(GTK_CONTAINER(layout), menubar);
325 gtk_container_add(GTK_CONTAINER(layout), GTK_WIDGET(vnc));
326 gtk_window_set_resizable(GTK_WINDOW(window), FALSE);
327
328 gtk_signal_connect(GTK_OBJECT(vnc), "vnc-pointer-grab",
329 GTK_SIGNAL_FUNC(viewer_grab), window);
330 gtk_signal_connect(GTK_OBJECT(vnc), "vnc-pointer-ungrab",
331 GTK_SIGNAL_FUNC(viewer_ungrab), window);
332
333 gtk_signal_connect(GTK_OBJECT(window), "delete-event",
334 GTK_SIGNAL_FUNC(viewer_shutdown), vnc);
335
336 gtk_signal_connect(GTK_OBJECT(vnc), "vnc-connected",
337 GTK_SIGNAL_FUNC(viewer_connected), NULL);
338 gtk_signal_connect(GTK_OBJECT(vnc), "vnc-initialized",
339 GTK_SIGNAL_FUNC(viewer_initialized), window);
340 gtk_signal_connect(GTK_OBJECT(vnc), "vnc-disconnected",
341 GTK_SIGNAL_FUNC(viewer_disconnected), NULL);
342
343 g_signal_connect(GTK_OBJECT(vnc), "vnc-auth-credential",
344 GTK_SIGNAL_FUNC(viewer_credential), NULL);
345
346 return window;
347 }
348
349
350 static void viewer_version(FILE *out)
351 {
352 fprintf(out, "%s version %s\n", PACKAGE, VERSION);
353 }
354
355 static void viewer_help(FILE *out, const char *app)
356 {
357 fprintf(out, "\n");
358 fprintf(out, "syntax: %s [OPTIONS] DOMAIN-NAME|ID|UUID\n", app);
359 fprintf(out, "\n");
360 viewer_version(out);
361 fprintf(out, "\n");
362 fprintf(out, "Options:");
363 fprintf(out, " -h, --help display command line help\n");
364 fprintf(out, " -v, --verbose display verbose information\n");
365 fprintf(out, " -V, --version display verion informaton\n");
366 fprintf(out, " -c URI, --connect URI connect to hypervisor URI\n");
367 fprintf(out, " -w, --wait wait for domain to start\n");
368 fprintf(out, "\n");
369 }
370
371 static int viewer_parse_uuid(const char *name, unsigned char *uuid)
372 {
373 int i;
374
375 const char *cur = name;
376 for (i = 0;i < 16;) {
377 uuid[i] = 0;
378 if (*cur == 0)
379 return -1;
380 if ((*cur == '-') || (*cur == ' ')) {
381 cur++;
382 continue;
383 }
384 if ((*cur >= '0') && (*cur <= '9'))
385 uuid[i] = *cur - '0';
386 else if ((*cur >= 'a') && (*cur <= 'f'))
387 uuid[i] = *cur - 'a' + 10;
388 else if ((*cur >= 'A') && (*cur <= 'F'))
389 uuid[i] = *cur - 'A' + 10;
390 else
391 return -1;
392 uuid[i] *= 16;
393 cur++;
394 if (*cur == 0)
395 return -1;
396 if ((*cur >= '0') && (*cur <= '9'))
397 uuid[i] += *cur - '0';
398 else if ((*cur >= 'a') && (*cur <= 'f'))
399 uuid[i] += *cur - 'a' + 10;
400 else if ((*cur >= 'A') && (*cur <= 'F'))
401 uuid[i] += *cur - 'A' + 10;
402 else
403 return -1;
404 i++;
405 cur++;
406 }
407
408 return 0;
409 }
410
411
412 static virDomainPtr viewer_lookup_domain(virConnectPtr conn, const char *name)
413 {
414 char *end;
415 int id = strtol(name, &end, 10);
416 virDomainPtr dom = NULL;
417 unsigned char uuid[16];
418
419 if (id >= 0 && end && !*end) {
420 dom = virDomainLookupByID(conn, id);
421 }
422 if (!dom && viewer_parse_uuid(name, uuid) == 0) {
423 dom = virDomainLookupByUUID(conn, uuid);
424 }
425 if (!dom) {
426 dom = virDomainLookupByName(conn, name);
427 }
428 return dom;
429 }
430
431 static int viewer_extract_vnc_graphics(virDomainPtr dom, char **host, char **port)
432 {
433 char *xmldesc = virDomainGetXMLDesc(dom, 0);
434 xmlDocPtr xml = NULL;
435 xmlParserCtxtPtr pctxt = NULL;
436 xmlXPathContextPtr ctxt = NULL;
437 xmlXPathObjectPtr obj = NULL;
438 int ret = -1;
439
440 pctxt = xmlNewParserCtxt();
441 if (!pctxt || !pctxt->sax)
442 goto error;
443
444 xml = xmlCtxtReadDoc(pctxt, (const xmlChar *)xmldesc, "domain.xml", NULL,
445 XML_PARSE_NOENT | XML_PARSE_NONET |
446 XML_PARSE_NOWARNING);
447 free(xmldesc);
448 if (!xml)
449 goto error;
450
451 ctxt = xmlXPathNewContext(xml);
452 if (!ctxt)
453 goto error;
454
455 obj = xmlXPathEval((const xmlChar *)"string(/domain/devices/graphics[@type='vnc']/@port)", ctxt);
456 if (!obj || obj->type != XPATH_STRING || !obj->stringval || !obj->stringval[0])
457 goto error;
458 if (!strcmp((const char*)obj->stringval, "-1"))
459 goto missing;
460
461 *port = strdup((const char*)obj->stringval);
462 xmlXPathFreeObject(obj);
463 obj = NULL;
464 /*
465 obj = xmlXPathEval((const xmlChar *)"string(/domain/devices/graphics[@type='vnc']/@listen)", ctxt);
466 if (!obj || obj->type != XPATH_STRING || !obj->stringval || !obj->stringval[0])
467 */
468 *host = NULL;
469
470 missing:
471 ret = 0;
472
473 error:
474 if (obj)
475 xmlXPathFreeObject(obj);
476 if (ctxt)
477 xmlXPathFreeContext(ctxt);
478 if (xml)
479 xmlFreeDoc(xml);
480 if (pctxt)
481 xmlFreeParserCtxt(pctxt);
482 return ret;
483 }
484
485 int main(int argc, char **argv)
486 {
487 GtkWidget *window;
488 GtkWidget *vnc;
489 char *uri = NULL;
490 int opt_ind;
491 const char *sopts = "hVc:";
492 static const struct option lopts[] = {
493 { "help", 0, 0, 'h' },
494 { "version", 0, 0, 'V' },
495 { "verbose", 0, 0, 'v' },
496 { "connect", 1, 0, 'c' },
497 { "wait", 0, 0, 'w' },
498 { 0, 0, 0, 0 }
499 };
500 int ch;
501 int waitvnc = 0;
502 virConnectPtr conn = NULL;
503 virDomainPtr dom = NULL;
504 char *host = NULL;
505 char *port = NULL;
506
507 while ((ch = getopt_long(argc, argv, sopts, lopts, &opt_ind)) != -1) {
508 switch (ch) {
509 case 'h':
510 viewer_help(stdout, argv[0]);
511 return 0;
512 case 'V':
513 viewer_version(stdout);
514 return 0;
515 case 'v':
516 verbose = 1;
517 break;
518 case 'c':
519 uri = strdup(optarg);
520 break;
521 case 'w':
522 waitvnc = 1;
523 break;
524 case '?':
525 viewer_help(stderr, argv[0]);
526 return 1;
527 }
528 }
529
530
531 if (argc != (optind+1)) {
532 viewer_help(stderr, argv[0]);
533 return 1;
534 }
535
536 gtk_init(&argc, &argv);
537
538 conn = virConnectOpenReadOnly(uri);
539 if (!conn) {
540 fprintf(stderr, "unable to connect to libvirt %s\n",
541 uri ? uri : "xen");
542 return 2;
543 }
544
545 do {
546 dom = viewer_lookup_domain(conn, argv[optind]);
547 if (!dom && !waitvnc) {
548 fprintf(stderr, "unable to lookup domain %s\n", argv[optind]);
549 return 3;
550 }
551 usleep(500*1000);
552 } while (!dom);
553
554 do {
555 viewer_extract_vnc_graphics(dom, &host, &port);
556 if (!port && !waitvnc) {
557 fprintf(stderr, "unable to find vnc graphics for %s\n", argv[optind]);
558 return 4;
559 }
560 usleep(300*1000);
561 } while (!port);
562
563 if (!host)
564 host = strdup("localhost");
565
566 vnc = vnc_display_new();
567 window = viewer_build_window(VNC_DISPLAY(vnc));
568 gtk_widget_realize(vnc);
569
570 vnc_display_set_keyboard_grab(VNC_DISPLAY(vnc), TRUE);
571 vnc_display_set_pointer_grab(VNC_DISPLAY(vnc), TRUE);
572
573 vnc_display_open_host(VNC_DISPLAY(vnc), host, port);
574
575 gtk_main();
576
577 return 0;
578 }
579
580 /*
581 * Local variables:
582 * c-indent-level: 8
583 * c-basic-offset: 8
584 * tab-width: 8
585 * End:
586 */
0 # -*- rpm-spec -*-
1
2 # This macro is used for the continuous automated builds. It just
3 # allows an extra fragment based on the timestamp to be appended
4 # to the release. This distinguishes automated builds, from formal
5 # Fedora RPM builds
6 %define _extra_release %{?dist:%{dist}}%{!?dist:%{?extra_release:%{extra_release}}}
7
8 Name: @PACKAGE@
9 Version: @VERSION@
10 Release: 1%{_extra_release}
11 Summary: Virtual Machine Viewer
12
13 Group: Applications/Emulators
14 License: GPL
15 URL: http://virt-manager.org/
16 Source0: http://virt-manager.org/download/sources/%{name}/%{name}-%{version}.tar.gz
17 BuildRoot: %{_tmppath}/%{name}-%{version}-%{release}-root-%(%{__id_u} -n)
18
19 ExclusiveArch: %{ix86} x86_64 ia64
20
21 BuildRequires: gtk2-devel
22 BuildRequires: gtk-vnc-devel
23
24 %description
25 Virtual Machine Viewer provides a graphical console client for connecting
26 to virtual machines. It uses the GTK-VNC widget to provide the display,
27 and libvirt for looking up VNC server details.
28
29 %prep
30 %setup -q
31
32 %build
33 %configure
34 make %{?_smp_mflags}
35
36
37 %install
38 rm -rf $RPM_BUILD_ROOT
39 make install DESTDIR=$RPM_BUILD_ROOT
40
41 %clean
42 rm -rf $RPM_BUILD_ROOT
43
44 %files
45 %defattr(-,root,root,-)
46 %doc README COPYING AUTHORS ChangeLog NEWS
47 %{_bindir}/%{name}
48
49 %changelog
50 * Fri Jul 20 2007 Daniel P. Berrange <berrange@redhat.com> - 0.0.1-1
51 - First release
52