In Wireshark and TShark, run dumpcap to get interface lists and lists of
[obnox/wireshark/wip.git] / gtk / capture_if_dlg.c
1 /* capture_if_dlg.c
2  * Routines for the capture interface dialog
3  *
4  * $Id$
5  *
6  * Wireshark - Network traffic analyzer
7  * By Gerald Combs <gerald@wireshark.org>
8  * Copyright 1998 Gerald Combs
9  *
10  * This program is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU General Public License
12  * as published by the Free Software Foundation; either version 2
13  * of the License, or (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program; if not, write to the Free Software
22  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
23  */
24
25 #ifdef HAVE_CONFIG_H
26 # include "config.h"
27 #endif
28
29 #include <gtk/gtk.h>
30
31 #ifdef HAVE_LIBPCAP
32
33 #include <string.h>
34
35 #ifdef __linux__
36 #include <sys/types.h>
37 #include <sys/stat.h>
38 #endif
39
40 #include <epan/prefs.h>
41
42 #include "../globals.h"
43 #include "../capture_errs.h"
44 #include "../capture_ifinfo.h"
45 #include "../simple_dialog.h"
46 #include "../capture.h"
47 #include "../capture-pcap-util.h"
48 #include "../capture_ui_utils.h"
49 #include "wsutil/file_util.h"
50 #include <wiretap/wtap.h>
51
52 #ifdef _WIN32
53 #include "../capture-wpcap.h"
54 #include "gtk/capture_if_details_dlg_win32.h"
55 #endif
56
57 #include "gtk/stock_icons.h"
58 #include "gtk/capture_dlg.h"
59 #include "gtk/capture_if_dlg.h"
60 #include "gtk/recent.h"
61 #include "gtk/gui_utils.h"
62 #include "gtk/dlg_utils.h"
63 #include "gtk/main.h"
64 #include "gtk/main_toolbar.h"
65 #include "gtk/help_dlg.h"
66 #include "gtk/keys.h"
67 #include "gtk/webbrowser.h"
68 #include "gtk/capture_globals.h"
69 #include "gtk/network_icons.h"
70
71 #ifdef HAVE_AIRPCAP
72 #include "../image/toolbar/capture_airpcap_16.xpm"
73 #endif
74
75 #ifdef _WIN32
76 #include "../image/toolbar/capture_ethernet_16.xpm"
77 #include "../image/toolbar/modem_16.xpm"
78 #endif
79
80 #include "../image/toolbar/network_virtual_16.xpm"
81
82 /* new buttons to be used instead of labels for 'Capture','Prepare',' */
83 /*#include "../image/toolbar/capture_capture_16.xpm"*/
84 /*#include "../image/toolbar/capture_prepare_16.xpm"*/
85 /*#include "../image/toolbar/capture_details_16.xpm"*/
86
87
88 #ifdef HAVE_AIRPCAP
89 #include "airpcap.h"
90 #include "airpcap_loader.h"
91 #include "airpcap_gui_utils.h"
92 #include "airpcap_dlg.h"
93 #endif
94
95 #define CAPTURE_IF_IP_ADDR_LABEL      "capture_if_ip_addr_label"
96 #define CAPTURE_IF_SELECTED_IP_ADDR   "capture_if_selected_ip_addr"
97
98 /*
99  * Keep a static pointer to the current "Capture Interfaces" window, if
100  * any, so that if somebody tries to do "Capture:Start" while there's
101  * already a "Capture Interfaces" window up, we just pop up the existing
102  * one, rather than creating a new one.
103  */
104 static GtkWidget *cap_if_w;
105
106 static GList     *if_data_list = NULL;
107
108 static guint      timer_id;
109
110 static GtkWidget *stop_bt;
111
112 static GList     *if_list;
113
114 /*
115  * Timeout, in milliseconds, for reads from the stream of captured packets.
116  */
117 #define CAP_READ_TIMEOUT        250
118
119
120 /* the "runtime" data of one interface */
121 typedef struct if_dlg_data_s {
122     GtkWidget   *device_lb;
123     GtkWidget   *descr_lb;
124     GtkWidget   *ip_lb;
125     GtkWidget   *curr_lb;
126     GtkWidget   *last_lb;
127     GtkWidget   *capture_bt;
128     GtkWidget   *prepare_bt;
129 #ifdef _WIN32
130     GtkWidget   *details_bt;
131 #endif
132     guint32     last_packets;
133     gchar       *device;
134     if_info_t   if_info;
135 } if_dlg_data_t;
136
137
138 /* start capture button was pressed */
139 static void
140 capture_do_cb(GtkWidget *capture_bt _U_, gpointer if_data)
141 {
142   if_dlg_data_t *if_dlg_data = if_data;
143
144 #ifdef HAVE_AIRPCAP
145   airpcap_if_active = get_airpcap_if_from_name(airpcap_if_list, if_dlg_data->if_info.name);
146   airpcap_if_selected = airpcap_if_active;
147 #endif
148
149   g_free(global_capture_opts.iface);
150   g_free(global_capture_opts.iface_descr);
151
152   global_capture_opts.iface = g_strdup(if_dlg_data->device);
153   global_capture_opts.iface_descr = get_interface_descriptive_name(global_capture_opts.iface);
154
155   /* XXX - remove this? */
156   if (global_capture_opts.save_file) {
157     g_free(global_capture_opts.save_file);
158     global_capture_opts.save_file = NULL;
159   }
160
161   /* stop capturing from all interfaces, we are going to do real work now ... */
162   window_destroy(cap_if_w);
163
164   capture_start_cb(NULL, NULL);
165 }
166
167
168 /* prepare capture button was pressed */
169 static void
170 capture_prepare_cb(GtkWidget *prepare_bt _U_, gpointer if_data)
171 {
172   if_dlg_data_t *if_dlg_data = if_data;
173
174   g_free(global_capture_opts.iface);
175   g_free(global_capture_opts.iface_descr);
176
177   global_capture_opts.iface = g_strdup(if_dlg_data->device);
178   global_capture_opts.iface_descr = get_interface_descriptive_name(global_capture_opts.iface);
179
180   /* stop capturing from all interfaces, we are going to do real work now ... */
181   window_destroy(cap_if_w);
182
183   capture_prep_cb(NULL, NULL);
184 }
185
186
187 #ifdef _WIN32
188 /* capture details button was pressed */
189 static void
190 capture_details_cb(GtkWidget *details_bt _U_, gpointer if_data)
191 {
192   if_dlg_data_t *if_dlg_data = if_data;
193
194
195   capture_if_details_open(if_dlg_data->device);
196 }
197 #endif
198
199 /* update a single interface */
200 static void
201 update_if(if_dlg_data_t *if_dlg_data, if_stat_cache_t *sc)
202 {
203   struct pcap_stat stats;
204   gchar *str;
205   guint diff;
206
207
208   /*
209    * Note that some versions of libpcap, on some versions of UN*X,
210    * pcap_stats() returns the number of packets since the last
211    * pcap_stats call.
212    *
213    * That's a bug, and should be fixed; "pcap_stats()" is supposed
214    * to work the same way on all platforms.
215    */
216   if (sc) {
217     if(capture_stats(sc, if_dlg_data->device, &stats)) {
218       diff = stats.ps_recv - if_dlg_data->last_packets;
219       if_dlg_data->last_packets = stats.ps_recv;
220
221       str = g_strdup_printf("%u", if_dlg_data->last_packets);
222       gtk_label_set_text(GTK_LABEL(if_dlg_data->curr_lb), str);
223       g_free(str);
224       str = g_strdup_printf("%u", diff);
225       gtk_label_set_text(GTK_LABEL(if_dlg_data->last_lb), str);
226       g_free(str);
227
228       gtk_widget_set_sensitive(if_dlg_data->curr_lb, diff);
229       gtk_widget_set_sensitive(if_dlg_data->last_lb, diff);
230     } else {
231       gtk_label_set_text(GTK_LABEL(if_dlg_data->curr_lb), "error");
232       gtk_label_set_text(GTK_LABEL(if_dlg_data->last_lb), "error");
233     }
234   }
235 }
236
237 /* update all interfaces */
238 static gboolean
239 update_all(gpointer data)
240 {
241     GList *curr;
242     int ifs;
243     if_stat_cache_t *sc = data;
244
245     if(!cap_if_w) {
246         return FALSE;
247     }
248
249     for(ifs = 0; (curr = g_list_nth(if_data_list, ifs)); ifs++) {
250         update_if(curr->data, sc);
251     }
252
253     return TRUE;
254 }
255
256 gboolean g_capture_in_progress = FALSE;
257
258 /* a live capture has started or stopped */
259 void
260 set_capture_if_dialog_for_capture_in_progress(gboolean capture_in_progress)
261 {
262     GList *curr;
263     int ifs;
264
265     g_capture_in_progress = capture_in_progress;
266
267     if(cap_if_w) {
268         gtk_widget_set_sensitive(stop_bt, capture_in_progress);
269
270         for(ifs = 0; (curr = g_list_nth(if_data_list, ifs)); ifs++) {
271             if_dlg_data_t *if_dlg_data = curr->data;
272
273             gtk_widget_set_sensitive(if_dlg_data->capture_bt, !capture_in_progress);
274             gtk_widget_set_sensitive(if_dlg_data->prepare_bt, !capture_in_progress);
275         }
276     }
277 }
278
279
280 /* the window was closed, cleanup things */
281 static void
282 capture_if_destroy_cb(GtkWidget *win _U_, gpointer user_data)
283 {
284     GList *curr;
285     int ifs;
286     if_stat_cache_t *sc = user_data;
287
288     g_source_remove(timer_id);
289
290     for(ifs = 0; (curr = g_list_nth(if_data_list, ifs)); ifs++) {
291         g_free(curr->data);
292     }
293
294     if_data_list = NULL;
295
296     free_interface_list(if_list);
297
298     /* Note that we no longer have a "Capture Options" dialog box. */
299     cap_if_w = NULL;
300
301     capture_stat_stop(sc);
302
303 #ifdef HAVE_AIRPCAP
304     airpcap_set_toolbar_stop_capture(airpcap_if_active);
305 #endif
306 }
307
308
309 /*
310  * Sorts the Interface List in alphabetical order
311  */
312 gint if_list_comparator_alph (const void *first_arg, const void *second_arg){
313   const if_info_t *first = first_arg, *second = second_arg;
314
315   if (first != NULL && first->description != NULL &&
316       second != NULL && second->description != NULL) {
317     return g_ascii_strcasecmp(first->description, second->description);
318   } else {
319     return 0;
320   }
321 }
322
323
324 /*
325  * Used to retrieve the interface icon.
326  * This is hideously platform-dependent.
327  */
328 GtkWidget * capture_get_if_icon(const if_info_t* if_info _U_)
329 {
330 #if defined(_WIN32)
331   /*
332    * Much digging failed to reveal any obvious way to get something such
333    * as the SNMP MIB-II ifType value for an interface:
334    *
335    *    http://www.iana.org/assignments/ianaiftype-mib
336    *
337    * by making some NDIS request.
338    */
339   if ( if_info->description && ( strstr(if_info->description,"generic dialup") != NULL ||
340        strstr(if_info->description,"PPP/SLIP") != NULL ) ) {
341     return xpm_to_widget(modem_16_xpm);
342   }
343
344   if ( if_info->description && ( strstr(if_info->description,"Wireless") != NULL ||
345        strstr(if_info->description,"802.11") != NULL || strstr(if_info->description,"AirPcap") != NULL ) ) {
346     return pixbuf_to_widget(network_wireless_pb_data);
347   }
348
349   if ( strstr(if_info->name,"airpcap") != NULL ) {
350     return pixbuf_to_widget(network_wireless_pb_data);
351   }
352
353   if ( if_info->description && strstr(if_info->description, "Bluetooth") != NULL ) {
354     return pixbuf_to_widget(network_bluetooth_pb_data);
355   }
356 #elif defined(__APPLE__)
357   /*
358    * XXX - yes, fetching all the network addresses for an interface
359    * gets you an AF_LINK address, of type "struct sockaddr_dl", and,
360    * yes, that includes an SNMP MIB-II ifType value.
361    *
362    * However, it's IFT_ETHER, i.e. Ethernet, for AirPort interfaces,
363    * not IFT_IEEE80211 (which isn't defined in OS X in any case).
364    *
365    * Perhaps some other BSD-flavored OSes won't make this mistake;
366    * however, FreeBSD 7.0 and OpenBSD 4.2, at least, appear to have
367    * made the same mistake, at least for my Belkin ZyDAS stick.
368    *
369    * On Mac OS X, one might be able to get the information one wants from
370    * IOKit.
371    */
372   if ( strcmp(if_info->name, "en1") == 0) {
373     return pixbuf_to_widget(network_wireless_pb_data);
374   }
375
376   /*
377    * XXX - PPP devices have names beginning with "ppp" and an IFT_ of
378    * IFT_PPP, but they could be dial-up, or PPPoE, or mobile phone modem,
379    * or VPN, or... devices.  One might have to dive into the bowels of
380    * IOKit to find out.
381    */
382
383   /*
384    * XXX - there's currently no support for raw Bluetooth capture,
385    * and IP-over-Bluetooth devices just look like fake Ethernet
386    * devices.  There's also Bluetooth modem support, but that'll
387    * probably just give you a device that looks like a PPP device.
388    */
389 #elif defined(__linux__)
390   /*
391    * Look for /sys/class/net/{device}/wireless.
392    */
393   struct stat statb;
394   char *wireless_path;
395
396   wireless_path = g_strdup_printf("/sys/class/net/%s/wireless", if_info->name);
397   if (wireless_path != NULL) {
398     if (ws_stat(wireless_path, &statb) == 0) {
399       g_free(wireless_path);
400       return pixbuf_to_widget(network_wireless_pb_data);
401     }
402     g_free(wireless_path);
403   }
404
405   /*
406    * Bluetooth devices.
407    *
408    * XXX - this is for raw Bluetooth capture; what about IP-over-Bluetooth
409    * devices?
410    */
411   if ( strstr(if_info->name,"bluetooth") != NULL) {
412     return pixbuf_to_widget(network_bluetooth_pb_data);
413   }
414
415   /*
416    * USB devices.
417    */
418   if ( strstr(if_info->name,"usbmon") != NULL ) {
419     return pixbuf_to_widget(network_usb_pb_data);
420   }
421 #endif
422
423   /*
424    * TODO: find a better icon!
425    * Bridge, NAT, or host-only interfaces on VMWare hosts have the name
426    * vmnet[0-9]+ or VMnet[0-9+ on Windows. Guests might use a native
427    * (LANCE or E1000) driver or the vmxnet driver. These devices have an
428    * IFT_ of IFT_ETHER, so we have to check the name.
429    */
430   if ( g_ascii_strncasecmp(if_info->name, "vmnet", 5) == 0) {
431     return xpm_to_widget(network_virtual_16_xpm);
432   }
433
434   if ( g_ascii_strncasecmp(if_info->name, "vmxnet", 6) == 0) {
435     return xpm_to_widget(network_virtual_16_xpm);
436   }
437
438   if ( if_info->description && strstr(if_info->description, "VMware") != NULL ) {
439     return xpm_to_widget(network_virtual_16_xpm);
440   }
441
442   return pixbuf_to_widget(network_wired_pb_data);
443 }
444
445
446 static int
447 get_ip_addr_count(GSList *addr_list)
448 {
449   GSList *curr_addr;
450   if_addr_t *addr;
451   int count;
452
453   count = 0;
454   for (curr_addr = addr_list; curr_addr != NULL;
455        curr_addr = g_slist_next(curr_addr)) {
456     addr = (if_addr_t *)curr_addr->data;
457     switch (addr->ifat_type) {
458
459     case IF_AT_IPv4:
460     case IF_AT_IPv6:
461       count++;
462       break;
463
464     default:
465       /* In case we add non-IP addresses */
466       break;
467     }
468   }
469   return count;
470 }
471
472 static const gchar *
473 set_ip_addr_label(GSList *addr_list, GtkWidget *ip_lb, guint selected_ip_addr)
474 {
475   GSList *curr_addr;
476   if_addr_t *addr;
477   const gchar *addr_str = NULL;
478
479   curr_addr = g_slist_nth(addr_list, selected_ip_addr);
480   if (curr_addr) {
481     addr = (if_addr_t *)curr_addr->data;
482     switch (addr->ifat_type) {
483
484     case IF_AT_IPv4:
485       addr_str = ip_to_str((guint8 *)&addr->addr.ip4_addr);
486       break;
487
488     case IF_AT_IPv6:
489       addr_str = ip6_to_str((struct e_in6_addr *)&addr->addr.ip6_addr);
490       break;
491
492     default:
493       /* Ignore non-IP addresses, in case we ever support them */
494       break;
495     }
496   }
497
498   if (addr_str) {
499     gtk_label_set_text(GTK_LABEL(ip_lb), addr_str);
500   } else {
501     gtk_label_set_text(GTK_LABEL(ip_lb), "unknown");
502   }
503   g_object_set_data(G_OBJECT(ip_lb), CAPTURE_IF_SELECTED_IP_ADDR, GINT_TO_POINTER(selected_ip_addr));
504
505   return addr_str;
506 }
507
508
509 static gboolean
510 ip_label_enter_cb(GtkWidget *eb, GdkEventCrossing *event _U_, gpointer user_data _U_)
511 {
512     gtk_widget_set_state(eb, GTK_STATE_PRELIGHT);
513
514     return FALSE;
515 }
516
517
518 static gboolean
519 ip_label_leave_cb(GtkWidget *eb, GdkEvent *event _U_, gpointer user_data _U_)
520 {
521     gtk_widget_set_state(eb, GTK_STATE_NORMAL);
522
523     return FALSE;
524 }
525
526
527 static gboolean
528 ip_label_press_cb(GtkWidget *widget, GdkEvent *event _U_, gpointer data)
529 {
530   GtkWidget *ip_lb = g_object_get_data(G_OBJECT(widget), CAPTURE_IF_IP_ADDR_LABEL);
531   GSList *addr_list = data;
532   GSList *curr_addr, *start_addr;
533   if_addr_t *addr;
534   guint selected_ip_addr = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(ip_lb), CAPTURE_IF_SELECTED_IP_ADDR));
535
536   /* Select next IP address */
537   start_addr = g_slist_nth(addr_list, selected_ip_addr);
538   for (;;) {
539     selected_ip_addr++;
540     if (g_slist_length(addr_list) <= selected_ip_addr) {
541       /* Wrap around */
542       selected_ip_addr = 0;
543     }
544     curr_addr = g_slist_nth(addr_list, selected_ip_addr);
545     if (curr_addr == start_addr) {
546       /* We wrapped all the way around */
547       break;
548     }
549
550     addr = (if_addr_t *)curr_addr->data;
551     switch (addr->ifat_type) {
552
553     case IF_AT_IPv4:
554     case IF_AT_IPv6:
555       goto found;
556
557     default:
558       /* In case we add non-IP addresses */
559       break;
560     }
561   }
562
563 found:
564   set_ip_addr_label(addr_list, ip_lb, selected_ip_addr);
565
566   return FALSE;
567 }
568
569
570
571 /* start getting capture stats from all interfaces */
572 void
573 capture_if_cb(GtkWidget *w _U_, gpointer d _U_)
574 {
575   GtkWidget     *main_vb,
576                                 *main_sw,
577                                 *bbox,
578                                 *close_bt,
579                                 *help_bt,
580                                 *icon;
581
582 #ifdef HAVE_AIRPCAP
583   GtkWidget             *decryption_cb;
584 #endif
585
586   GtkWidget     *if_tb;
587   GtkWidget     *if_lb;
588   GtkWidget     *eb;
589   GtkTooltips   *tooltips;
590   int           err;
591   gchar         *err_str;
592   GtkRequisition requisition;
593   int           row, height;
594   if_dlg_data_t *if_dlg_data;
595   int           ifs;
596   GList         *curr;
597   if_info_t     *if_info;
598   GString       *if_tool_str = g_string_new("");
599   const gchar   *addr_str;
600   gchar         *tmp_str;
601   gchar         *user_descr;
602   if_stat_cache_t *sc;
603
604   if (cap_if_w != NULL) {
605     /* There's already a "Capture Interfaces" dialog box; reactivate it. */
606     reactivate_window(cap_if_w);
607     return;
608   }
609
610 #ifdef _WIN32
611   /* Is WPcap loaded? */
612   if (!has_wpcap) {
613     char *detailed_err;
614
615     detailed_err = cant_load_winpcap_err("Wireshark");
616     simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", detailed_err);
617     g_free(detailed_err);
618     return;
619   }
620 #endif
621
622   /* LOAD THE INTERFACES */
623   if_list = capture_interface_list(&err, &err_str);
624   if_list = g_list_sort (if_list, if_list_comparator_alph);
625   if (if_list == NULL && err == CANT_GET_INTERFACE_LIST) {
626     simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", err_str);
627     g_free(err_str);
628     return;
629   }
630
631 #ifdef HAVE_AIRPCAP
632   /* LOAD AIRPCAP INTERFACES */
633   airpcap_if_list = get_airpcap_interface_list(&err, &err_str);
634   if (airpcap_if_list == NULL)
635     airpcap_if_active = airpcap_if_selected = NULL;
636
637   decryption_cb = g_object_get_data(G_OBJECT(airpcap_tb),AIRPCAP_TOOLBAR_DECRYPTION_KEY);
638   update_decryption_mode_list(decryption_cb);
639
640   if (airpcap_if_list == NULL && err == CANT_GET_AIRPCAP_INTERFACE_LIST) {
641 #if 0
642     /* XXX - Do we need to show an error here? */
643     simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", err_str);
644 #endif
645     g_free(err_str);
646   }
647
648   /* If no airpcap interface is present, gray everything */
649   if (airpcap_if_active == NULL) {
650     if (airpcap_if_list == NULL) {
651       /*No airpcap device found */
652       airpcap_enable_toolbar_widgets(airpcap_tb,FALSE);
653     } else {
654       /* default adapter is not airpcap... or is airpcap but is not found*/
655       airpcap_set_toolbar_stop_capture(airpcap_if_active);
656       airpcap_enable_toolbar_widgets(airpcap_tb,FALSE);
657     }
658   }
659
660   airpcap_set_toolbar_start_capture(airpcap_if_active);
661 #endif
662
663   cap_if_w = dlg_window_new("Wireshark: Capture Interfaces");  /* transient_for top_level */
664   gtk_window_set_destroy_with_parent (GTK_WINDOW(cap_if_w), TRUE);
665
666   tooltips = gtk_tooltips_new();
667
668   main_sw = gtk_scrolled_window_new(NULL, NULL);
669   gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(main_sw), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
670   gtk_container_add(GTK_CONTAINER(cap_if_w), main_sw);
671
672   main_vb = gtk_vbox_new(FALSE, 0);
673   gtk_container_set_border_width(GTK_CONTAINER(main_vb), 5);
674   gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(main_sw), main_vb);
675
676
677   if_tb = gtk_table_new(1,9, FALSE);
678   gtk_table_set_row_spacings(GTK_TABLE(if_tb), 3);
679   gtk_table_set_col_spacings(GTK_TABLE(if_tb), 3);
680   gtk_box_pack_start(GTK_BOX(main_vb), if_tb, FALSE, FALSE, 0);
681
682   row = 0;
683   height = 0;
684
685   /* This is the icon column, used to display which kind of interface we have */
686   if_lb = gtk_label_new("");
687   gtk_table_attach_defaults(GTK_TABLE(if_tb), if_lb, 0, 1, row, row+1);
688
689 #ifndef _WIN32
690   /*
691    * On Windows, device names are generally not meaningful - NT 5
692    * uses long blobs with GUIDs in them, for example - so we don't
693    * bother showing them.
694    */
695   if_lb = gtk_label_new("Device");
696   gtk_table_attach_defaults(GTK_TABLE(if_tb), if_lb, 1, 2, row, row+1);
697 #endif
698
699   if_lb = gtk_label_new("Description");
700   gtk_table_attach_defaults(GTK_TABLE(if_tb), if_lb, 2, 3, row, row+1);
701
702   if_lb = gtk_label_new(" IP ");
703   gtk_table_attach_defaults(GTK_TABLE(if_tb), if_lb, 3, 4, row, row+1);
704
705   if_lb = gtk_label_new("Packets");
706   gtk_table_attach_defaults(GTK_TABLE(if_tb), if_lb, 4, 5, row, row+1);
707
708   if_lb = gtk_label_new(" Packets/s ");
709   gtk_table_attach_defaults(GTK_TABLE(if_tb), if_lb, 5, 6, row, row+1);
710
711   stop_bt = gtk_button_new_from_stock(WIRESHARK_STOCK_CAPTURE_STOP);
712   gtk_tooltips_set_tip(tooltips, stop_bt,
713           "Stop a running capture.", NULL);
714 #ifdef _WIN32
715   gtk_table_attach_defaults(GTK_TABLE(if_tb), stop_bt, 6, 9, row, row+1);
716 #else
717   gtk_table_attach_defaults(GTK_TABLE(if_tb), stop_bt, 6, 8, row, row+1);
718 #endif
719   g_signal_connect(stop_bt, "clicked", G_CALLBACK(capture_stop_cb), NULL);
720
721   row++;
722   gtk_widget_size_request(stop_bt, &requisition);
723   height += requisition.height + 15;
724
725   /* Start gathering statistics (using dumpcap) */
726   sc = capture_stat_start(if_list);
727
728   /* List the interfaces */
729   for(ifs = 0; (curr = g_list_nth(if_list, ifs)); ifs++) {
730       g_string_assign(if_tool_str, "");
731       if_info = curr->data;
732
733       /* Continue if capture device is hidden */
734       if (prefs_is_capture_device_hidden(if_info->name)) {
735           continue;
736       }
737
738       if_dlg_data = g_malloc0(sizeof(if_dlg_data_t));
739       if_dlg_data->if_info = *if_info;
740
741       /* Kind of adaptor (icon) */
742 #ifdef HAVE_AIRPCAP
743       if(get_airpcap_if_from_name(airpcap_if_list,if_info->name) != NULL)
744         icon = xpm_to_widget(capture_airpcap_16_xpm);
745       else
746         icon = capture_get_if_icon(if_info);
747 #else
748       icon = capture_get_if_icon(if_info);
749 #endif
750
751       gtk_table_attach_defaults(GTK_TABLE(if_tb), icon, 0, 1, row, row+1);
752
753       /* device name */
754       if_dlg_data->device_lb = gtk_label_new(if_info->name);
755       if_dlg_data->device = if_info->name;
756 #ifndef _WIN32
757       gtk_misc_set_alignment(GTK_MISC(if_dlg_data->device_lb), 0.0f, 0.5f);
758       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->device_lb, 1, 2, row, row+1);
759 #endif
760       g_string_append(if_tool_str, "Device: ");
761       g_string_append(if_tool_str, if_info->name);
762       g_string_append(if_tool_str, "\n");
763
764       /* description */
765       user_descr = capture_dev_user_descr_find(if_info->name);
766       if (user_descr) {
767         if_dlg_data->descr_lb = gtk_label_new(user_descr);
768         g_free (user_descr);
769       } else {
770         if (if_info->description)
771           if_dlg_data->descr_lb = gtk_label_new(if_info->description);
772         else
773           if_dlg_data->descr_lb = gtk_label_new("");
774       }
775       gtk_misc_set_alignment(GTK_MISC(if_dlg_data->descr_lb), 0.0f, 0.5f);
776       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->descr_lb, 2, 3, row, row+1);
777
778       if (if_info->description) {
779         g_string_append(if_tool_str, "Description: ");
780         g_string_append(if_tool_str, if_info->description);
781         g_string_append(if_tool_str, "\n");
782       }
783
784       /* IP address */
785       /* Only one IP address will be shown, start with the first */
786       g_string_append(if_tool_str, "IP: ");
787       if_dlg_data->ip_lb = gtk_label_new("");
788       addr_str = set_ip_addr_label (if_info->addrs, if_dlg_data->ip_lb, 0);
789       if (addr_str) {
790         gtk_widget_set_sensitive(if_dlg_data->ip_lb, TRUE);
791         g_string_append(if_tool_str, addr_str);
792       } else {
793         gtk_widget_set_sensitive(if_dlg_data->ip_lb, FALSE);
794         g_string_append(if_tool_str, "unknown");
795       }
796       eb = gtk_event_box_new ();
797       gtk_container_add(GTK_CONTAINER(eb), if_dlg_data->ip_lb);
798       gtk_table_attach_defaults(GTK_TABLE(if_tb), eb, 3, 4, row, row+1);
799       if (get_ip_addr_count(if_info->addrs) > 1) {
800         /* More than one IP address, make it possible to toggle */
801         g_object_set_data(G_OBJECT(eb), CAPTURE_IF_IP_ADDR_LABEL, if_dlg_data->ip_lb);
802         g_signal_connect(eb, "enter-notify-event", G_CALLBACK(ip_label_enter_cb), NULL);
803         g_signal_connect(eb, "leave-notify-event", G_CALLBACK(ip_label_leave_cb), NULL);
804         g_signal_connect(eb, "button-press-event", G_CALLBACK(ip_label_press_cb), if_info->addrs);
805       }
806       g_string_append(if_tool_str, "\n");
807
808       /* packets */
809       if_dlg_data->curr_lb = gtk_label_new("-");
810       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->curr_lb, 4, 5, row, row+1);
811
812       /* packets/s */
813       if_dlg_data->last_lb = gtk_label_new("-");
814       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->last_lb, 5, 6, row, row+1);
815
816       /* capture button */
817       if_dlg_data->capture_bt = gtk_button_new_from_stock(WIRESHARK_STOCK_CAPTURE_START);
818           g_signal_connect(if_dlg_data->capture_bt, "clicked", G_CALLBACK(capture_do_cb), if_dlg_data);
819       tmp_str = g_strdup_printf("Immediately start a capture from this interface:\n\n%s", if_tool_str->str);
820       gtk_tooltips_set_tip(tooltips, if_dlg_data->capture_bt,
821           tmp_str, NULL);
822       g_free(tmp_str);
823       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->capture_bt, 6, 7, row, row+1);
824
825       /* prepare button */
826       if_dlg_data->prepare_bt = gtk_button_new_from_stock(WIRESHARK_STOCK_CAPTURE_OPTIONS);
827       g_signal_connect(if_dlg_data->prepare_bt, "clicked", G_CALLBACK(capture_prepare_cb), if_dlg_data);
828       gtk_tooltips_set_tip(tooltips, if_dlg_data->prepare_bt,
829           "Open the capture options dialog with this interface selected.", NULL);
830       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->prepare_bt, 7, 8, row, row+1);
831
832       /* details button */
833 #ifdef _WIN32
834       if_dlg_data->details_bt = gtk_button_new_from_stock(WIRESHARK_STOCK_CAPTURE_DETAILS);
835       gtk_tooltips_set_tip(tooltips, if_dlg_data->details_bt,
836           "Open the capture details dialog of this interface.", NULL);
837       gtk_table_attach_defaults(GTK_TABLE(if_tb), if_dlg_data->details_bt, 8, 9, row, row+1);
838       if (capture_if_has_details(if_dlg_data->device)) {
839         g_signal_connect(if_dlg_data->details_bt, "clicked", G_CALLBACK(capture_details_cb), if_dlg_data);
840       } else {
841         gtk_widget_set_sensitive(if_dlg_data->details_bt, FALSE);
842       }
843 #endif
844
845       if_data_list = g_list_append(if_data_list, if_dlg_data);
846
847       row++;
848       if (row <= 10) {
849           /* Lets add up 10 rows of interfaces, otherwise the window may become too high */
850           gtk_widget_size_request(GTK_WIDGET(if_dlg_data->prepare_bt), &requisition);
851           height += requisition.height;
852       }
853   }
854
855   g_string_free(if_tool_str, TRUE);
856
857   /* Button row: close and help button */
858   bbox = dlg_button_row_new(GTK_STOCK_CLOSE, GTK_STOCK_HELP, NULL);
859   gtk_box_pack_start(GTK_BOX(main_vb), bbox, FALSE, FALSE, 5);
860
861   close_bt = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_CLOSE);
862   window_set_cancel_button(cap_if_w, close_bt, window_cancel_button_cb);
863   gtk_tooltips_set_tip(tooltips, close_bt, "Close this window.", NULL);
864
865   help_bt = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_HELP);
866   g_signal_connect(help_bt, "clicked", G_CALLBACK(topic_cb), (gpointer)(HELP_CAPTURE_INTERFACES_DIALOG));
867
868   gtk_widget_size_request(GTK_WIDGET(close_bt), &requisition);
869   /* height + static offset + what the GTK MS Windows Engine needs in addition per interface */
870   height += requisition.height + 20 + ifs;
871   gtk_window_set_default_size(GTK_WINDOW(cap_if_w), -1, height);
872
873   gtk_widget_grab_default(close_bt);
874
875   g_signal_connect(cap_if_w, "delete_event", G_CALLBACK(window_delete_event_cb), NULL);
876   g_signal_connect(cap_if_w, "destroy", G_CALLBACK(capture_if_destroy_cb), sc);
877
878   gtk_widget_show_all(cap_if_w);
879   window_present(cap_if_w);
880
881   set_capture_if_dialog_for_capture_in_progress(g_capture_in_progress);
882
883     /* update the interface list every 1000ms */
884   timer_id = g_timeout_add(1000, update_all, sc);
885 }
886
887 #else /* HAVE_LIBPCAP */
888
889 void
890 set_capture_if_dialog_for_capture_in_progress(gboolean capture_in_progress _U_)
891 {
892 }
893
894 #endif /* HAVE_LIBPCAP */