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