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