summaryrefslogtreecommitdiffstats
path: root/capture
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-10 20:34:10 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-10 20:34:10 +0000
commite4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc (patch)
tree68cb5ef9081156392f1dd62a00c6ccc1451b93df /capture
parentInitial commit. (diff)
downloadwireshark-e4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc.tar.xz
wireshark-e4ba6dbc3f1e76890b22773807ea37fe8fa2b1bc.zip
Adding upstream version 4.2.2.upstream/4.2.2
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'capture')
-rw-r--r--capture/.editorconfig21
-rw-r--r--capture/CMakeLists.txt137
-rw-r--r--capture/airpcap.h906
-rw-r--r--capture/airpcap_loader.c1229
-rw-r--r--capture/airpcap_loader.h409
-rw-r--r--capture/capture-pcap-util-int.h63
-rw-r--r--capture/capture-pcap-util-unix.c209
-rw-r--r--capture/capture-pcap-util.c1826
-rw-r--r--capture/capture-pcap-util.h120
-rw-r--r--capture/capture-wpcap.c967
-rw-r--r--capture/capture-wpcap.h44
-rw-r--r--capture/capture_ifinfo.c359
-rw-r--r--capture/capture_ifinfo.h142
-rw-r--r--capture/capture_session.h145
-rw-r--r--capture/capture_sync.c2206
-rw-r--r--capture/capture_sync.h116
-rw-r--r--capture/capture_win_ifnames.c269
-rw-r--r--capture/capture_win_ifnames.h35
-rw-r--r--capture/iface_monitor.c396
-rw-r--r--capture/iface_monitor.h74
-rw-r--r--capture/ws80211_utils.c1265
-rw-r--r--capture/ws80211_utils.h135
22 files changed, 11073 insertions, 0 deletions
diff --git a/capture/.editorconfig b/capture/.editorconfig
new file mode 100644
index 00000000..9411ef01
--- /dev/null
+++ b/capture/.editorconfig
@@ -0,0 +1,21 @@
+#
+# Editor configuration
+#
+# https://editorconfig.org
+#
+
+[capture-pcap-util-unix.[ch]]
+indent_style = tab
+indent_size = tab
+
+[capture-pcap-util.[ch]]
+indent_style = tab
+indent_size = tab
+
+[capture-wpcap.[ch]]
+indent_style = tab
+indent_size = tab
+
+[ws80211_utils.[ch]]
+indent_style = tab
+indent_size = tab
diff --git a/capture/CMakeLists.txt b/capture/CMakeLists.txt
new file mode 100644
index 00000000..65b91f8f
--- /dev/null
+++ b/capture/CMakeLists.txt
@@ -0,0 +1,137 @@
+# CMakeLists.txt
+#
+# Wireshark - Network traffic analyzer
+# By Gerald Combs <gerald@wireshark.org>
+# Copyright 1998 Gerald Combs
+#
+# SPDX-License-Identifier: GPL-2.0-or-later
+#
+
+
+if(UNIX)
+ set(PLATFORM_CAPUTILS_SRC
+ capture-pcap-util-unix.c
+ )
+endif()
+
+if(WIN32)
+ set(PLATFORM_CAPUTILS_SRC
+ capture_win_ifnames.c
+ capture-wpcap.c
+ )
+endif()
+
+set(CAPUTILS_SRC
+ ${PLATFORM_CAPUTILS_SRC}
+ capture-pcap-util.c
+)
+
+if (AIRPCAP_FOUND)
+ set(CAPUTILS_SRC
+ ${CAPUTILS_SRC}
+ airpcap_loader.c
+ )
+endif()
+
+set_source_files_properties(
+ ${CAPUTILS_SRC}
+ PROPERTIES
+ COMPILE_FLAGS "${WERROR_COMMON_FLAGS}"
+)
+
+
+add_library(caputils STATIC
+ ${CAPUTILS_SRC}
+)
+
+target_link_libraries(caputils
+ PUBLIC
+ $<$<BOOL:${PCAP_FOUND}>:pcap::pcap>
+ PRIVATE
+ wsutil
+ ${WIN_IPHLPAPI_LIBRARY}
+)
+if(WIN32)
+ target_link_libraries(caputils PRIVATE ${GMODULE2_LIBRARIES})
+endif()
+
+set_target_properties(caputils PROPERTIES
+ LINK_FLAGS "${WS_LINK_FLAGS}"
+ FOLDER "Libs")
+if(MSVC)
+ set_target_properties(caputils PROPERTIES LINK_FLAGS_DEBUG "${WS_MSVC_DEBUG_LINK_FLAGS}")
+endif()
+
+set(CAPCHILD_SRC
+ capture_ifinfo.c
+ capture_sync.c
+)
+
+set_source_files_properties(
+ ${CAPCHILD_SRC}
+ PROPERTIES
+ COMPILE_FLAGS "${WERROR_COMMON_FLAGS}"
+)
+
+file(GLOB CAPCHILD_HEADERS capture_session.h)
+
+add_library(capchild STATIC
+ ${CAPCHILD_SRC}
+)
+
+target_link_libraries(capchild
+ PRIVATE
+ wsutil
+ $<$<BOOL:${PCAP_FOUND}>:pcap::pcap>
+)
+
+set_target_properties(capchild PROPERTIES
+ LINK_FLAGS "${WS_LINK_FLAGS}"
+ FOLDER "Libs")
+if(MSVC)
+ set_target_properties(capchild PROPERTIES LINK_FLAGS_DEBUG "${WS_MSVC_DEBUG_LINK_FLAGS}")
+endif()
+
+add_library(iface_monitor STATIC
+ iface_monitor.c
+ ws80211_utils.c
+)
+
+target_link_libraries(iface_monitor
+ PRIVATE
+ ${NL_LIBRARIES}
+)
+
+target_include_directories(iface_monitor SYSTEM
+ PRIVATE
+ ${NL_INCLUDE_DIRS}
+)
+
+CHECKAPI(
+ NAME
+ caputils-base
+ SWITCHES
+ SOURCES
+ ${CAPUTILS_SRC}
+)
+CHECKAPI(
+ NAME
+ caputils-todo
+ SWITCHES
+ -M
+ SOURCES
+ ${CAPUTILS_SRC}
+)
+
+#
+# Editor modelines - https://www.wireshark.org/tools/modelines.html
+#
+# Local variables:
+# c-basic-offset: 8
+# tab-width: 8
+# indent-tabs-mode: t
+# End:
+#
+# vi: set shiftwidth=8 tabstop=8 noexpandtab:
+# :indentSize=8:tabSize=8:noTabs=false:
+#
diff --git a/capture/airpcap.h b/capture/airpcap.h
new file mode 100644
index 00000000..f7549d8c
--- /dev/null
+++ b/capture/airpcap.h
@@ -0,0 +1,906 @@
+/** @file
+ *
+ * Copyright (c) 2006-2007 CACE Technologies, Davis (California)
+ * All rights reserved.
+ *
+ * SPDX-License-Identifier: BSD-3-Clause
+ */
+
+#if !defined(AIRPCAP_H__EAE405F5_0171_9592_B3C2_C19EC426AD34__INCLUDED_)
+#define AIRPCAP_H__EAE405F5_0171_9592_B3C2_C19EC426AD34__INCLUDED_
+
+#ifdef _MSC_VER
+/* This stops VS2005 ranting against stdio. */
+#pragma warning( disable : 4996)
+#endif
+
+#ifdef _WIN32
+#include <winsock2.h>
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*
+ \mainpage AirPcap interface documentation
+
+ \section Introduction
+
+ This document describes the data structures and the functions exported by the CACE Technologies AirPcap library.
+ The AirPcap library provides low-level access to the AirPcap driver including advanced capabilities such as channel setting,
+ link type control and WEP configuration.<br>
+ This manual includes the following sections:
+
+ \note throughout this documentation, \e device refers to a physical USB AirPcap device, while \e adapter is an open API
+ instance. Most of the AirPcap API operations are adapter-specific but some of them, like setting the channel, are
+ per-device and will be reflected on all the open adapters. These functions will have "Device" in their name, e.g.
+ AirpcapSetDeviceChannel().
+
+ \b Sections:
+
+ - \ref airpcapfuncs
+ - \ref airpcapdefs
+ - \ref radiotap
+*/
+
+/** @defgroup airpcapdefs AirPcap definitions and data structures
+ * @{
+ */
+
+/*!
+ \brief This string is the fixed prefix in the airpcap adapter name.
+ It can be used to parse the name field in an AirpcapDeviceDescription structure.
+*/
+#define AIRPCAP_DEVICE_NAME_PREFIX "\\\\.\\airpcap"
+
+/*!
+ \brief This string is the scanf modifier to extract the adapter number from an adapter name.
+ It can be used to parse the name field in an AirpcapDeviceDescription structure with scanf.
+*/
+#define AIRPCAP_DEVICE_NUMBER_EXTRACT_STRING "\\\\.\\airpcap%u"
+
+#define AIRPCAP_DEVICE_ANY_EXTRACT_STRING "\\\\.\\airpcap_any"
+
+/*!
+ \brief Entry in the list returned by \ref AirpcapGetDeviceList();
+*/
+typedef struct _AirpcapDeviceDescription
+{
+ struct _AirpcapDeviceDescription *next; /* < Next element in the list */
+ char * Name; /* < Device name */
+ char * Description; /* < Device description */
+} AirpcapDeviceDescription, *PAirpcapDeviceDescription;
+
+#define MAX_ENCRYPTION_KEYS 64
+
+#define WEP_KEY_MAX_SIZE 32 /* < Maximum size of a WEP key, in bytes. This is the size of an entry in the
+ < AirpcapWepKeysCollection structure. */
+
+#ifdef _WIN32
+#ifndef __MINGW32__
+#pragma pack(push)
+#pragma pack(1)
+#endif
+#endif
+
+#define AIRPCAP_KEYTYPE_WEP 0 /* < Key type: WEP. The key can have an arbitrary length smaller than 32 bytes. */
+#define AIRPCAP_KEYTYPE_TKIP 1 /* < Key type: TKIP (WPA). NOT SUPPORTED YET. */
+#define AIRPCAP_KEYTYPE_CCMP 2 /* < Key type: CCMP (WPA2). NOT SUPPORTED YET. */
+
+/*!
+ \brief WEP key container
+*/
+typedef struct _AirpcapKey
+{
+ unsigned KeyType; /* < Type of key, can be on of: \ref AIRPCAP_KEYTYPE_WEP, \ref AIRPCAP_KEYTYPE_TKIP, \ref AIRPCAP_KEYTYPE_CCMP. Only AIRPCAP_KEYTYPE_WEP is supported by the driver at the moment. */
+ unsigned KeyLen; /* < Length of the key, in bytes */
+ uint8_t KeyData[WEP_KEY_MAX_SIZE]; /* < Key Data */
+}
+#ifdef __MINGW32__
+__attribute__((__packed__))
+#endif
+AirpcapKey, *PAirpcapKey;
+
+/*!
+ \brief frequency Band.
+ 802.11 adapters can support different frequency bands, the most important of which are: 2.4GHz (802.11b/g/n)
+ and 5GHz (802.11a/n).
+*/
+typedef enum _AirpcapChannelBand
+{
+ AIRPCAP_CB_AUTO = 1, /* < Automatically pick the best frequency band */
+ AIRPCAP_CB_2_4_GHZ = 2, /* < 2.4 GHz frequency band */
+ AIRPCAP_CB_4_GHZ = 4, /* < 4 GHz frequency band */
+ AIRPCAP_CB_5_GHZ = 5 /* < 5 GHz frequency band */
+}AirpcapChannelBand, *PAirpcapChannelBand;
+
+/*!
+ \brief Type of frame validation the adapter performs.
+ An adapter can be instructed to accept different kind of frames: correct frames only, frames with wrong Frame Check Sequence (FCS) only, all frames.
+*/
+typedef enum _AirpcapValidationType
+{
+ AIRPCAP_VT_ACCEPT_EVERYTHING = 1, /* < Accept all the frames the device captures */
+ AIRPCAP_VT_ACCEPT_CORRECT_FRAMES = 2, /* < Accept correct frames only, i.e. frames with correct Frame Check Sequence (FCS). */
+ AIRPCAP_VT_ACCEPT_CORRUPT_FRAMES = 3, /* < Accept corrupt frames only, i.e. frames with wrong Frame Check Sequence (FCS). */
+ AIRPCAP_VT_UNKNOWN = 4 /* < Unknown validation type. You should see it only in case of error. */
+}AirpcapValidationType, *PAirpcapValidationType;
+
+/*!
+ \brief Type of decryption the adapter performs.
+ An adapter can be instructed to turn decryption (based on the device-configured keys configured
+ with \ref AirpcapSetDeviceKeys()) on or off.
+*/
+typedef enum _AirpcapDecryptionState
+{
+ AIRPCAP_DECRYPTION_ON = 1, /* < This adapter performs decryption */
+ AIRPCAP_DECRYPTION_OFF = 2 /* < This adapter does not perform decryption */
+}AirpcapDecryptionState, *PAirpcapDecryptionState;
+
+
+/*!
+ \brief Storage for a MAC address
+*/
+typedef struct _AirpcapMacAddress
+{
+ uint8_t Address[6]; /* < MAC address bytes */
+}
+#ifdef __MINGW32__
+__attribute__((__packed__))
+#endif
+AirpcapMacAddress, *PAirpcapMacAddress;
+
+/*!
+ \brief This structure is used to store a collection of WEP keys.
+ Note that the definition of the structure has one key in it
+ (so that this code can be compiled by compilers that don't
+ support zero-length arrays), so be careful to allocate a buffer
+ with the size of the set of keys, as per the following example:
+
+ \code
+ PAirpcapKeysCollection KeysCollection;
+ unsigned KeysCollectionSize;
+
+ KeysCollectionSize = AirpcapKeysCollectionSize(NumKeys);
+
+ KeysCollection = (PAirpcapKeysCollection)malloc(KeysCollectionSize);
+ if(!KeysCollection)
+ {
+ Error
+ }
+ \endcode
+*/
+typedef struct _AirpcapKeysCollection
+{
+ unsigned nKeys; /* < Number of keys in the collection */
+ AirpcapKey Keys[1]; /* < Array of nKeys keys. */
+} AirpcapKeysCollection, *PAirpcapKeysCollection;
+
+#define AirpcapKeysCollectionSize(nKeys) \
+ ((sizeof(AirpcapKeysCollection) - sizeof(AirpcapKey)) + ((nKeys) * sizeof(AirpcapKey)))
+#define AirpcapKeysCollectionSizeToKeyCount(size) \
+ (unsigned)(((size) - AirpcapKeysCollectionSize(0))/sizeof(AirpcapKey))
+
+/*!
+ \brief Packet header.
+
+ This structure defines the BPF that precedes every packet delivered to the application.
+*/
+typedef struct _AirpcapBpfHeader
+{
+ unsigned TsSec; /* < Timestamp associated with the captured packet. SECONDS. */
+ unsigned TsUsec; /* < Timestamp associated with the captured packet. MICROSECONDS. */
+ unsigned Caplen; /* < Length of captured portion. The captured portion <b>can be different</b> from the original packet, because it is possible (with a proper filter) to instruct the driver to capture only a portion of the packets. */
+ unsigned Originallen; /* < Original length of packet */
+ uint16_t Hdrlen; /* < Length of bpf header (this struct plus alignment padding). In some cases, a padding could be added between the end of this structure and the packet data for performance reasons. This field can be used to retrieve the actual data of the packet. */
+}
+#ifdef __MINGW32__
+__attribute__((__packed__))
+#endif
+AirpcapBpfHeader, *PAirpcapBpfHeader;
+
+/* Helper macros to extract packets coming from the driver. Rounds up to the next even multiple of AIRPCAP_ALIGNMENT. */
+#define AIRPCAP_ALIGNMENT sizeof(int)
+#define AIRPCAP_WORDALIGN(x) (((x)+(AIRPCAP_ALIGNMENT-1))&~(AIRPCAP_ALIGNMENT-1))
+
+#ifdef _WIN32
+#ifndef __MINGW32__
+#pragma pack(pop)
+#endif
+#endif
+
+#define AIRPCAP_ERRBUF_SIZE 512 /* < Size of the error buffer, in bytes */
+
+#ifndef __AIRPCAP_DRIVER__
+
+/*!
+ \brief Link type.
+ AirPcap supports two kind of 802.11 linktypes: plain 802.11 and radiotap.
+*/
+#undef _AirpcapLinkType
+typedef enum _AirpcapLinkType
+{
+ AIRPCAP_LT_802_11 = 1, /* < plain 802.11 linktype. Every packet in the buffer contains the raw 802.11 frame, including MAC FCS. */
+ AIRPCAP_LT_802_11_PLUS_RADIO = 2, /* < 802.11 plus radiotap linktype. Every packet in the buffer contains a radiotap header followed by the 802.11 frame. MAC FCS is included. */
+ AIRPCAP_LT_UNKNOWN = 3, /* < Unknown linktype. You should see it only in case of error. */
+ AIRPCAP_LT_802_11_PLUS_PPI = 4 /* < 802.11 plus PPI header linktype. Every packet in the buffer contains a PPI header followed by the 802.11 frame. MAC FCS is included. */
+}AirpcapLinkType, *PAirpcapLinkType;
+
+#if !defined(AIRPCAP_HANDLE__EAE405F5_0171_9592_B3C2_C19EC426AD34__DEFINED_)
+#define AIRPCAP_HANDLE__EAE405F5_0171_9592_B3C2_C19EC426AD34__DEFINED_
+/*!
+ \brief Adapter handle.
+*/
+typedef struct _AirpcapHandle AirpcapHandle, *PAirpcapHandle;
+#endif
+
+/*!
+ \brief Capture statistics.
+ Returned by \ref AirpcapGetStats();
+*/
+typedef struct _AirpcapStats
+{
+ unsigned Recvs; /* < Number of packets that the driver received by the adapter */
+ /* < from the beginning of the current capture. This value includes the packets */
+ /* < dropped because of buffer full. */
+ unsigned Drops; /* < number of packets that the driver dropped from the beginning of a capture. */
+ /* < A packet is lost when the driver's buffer is full. */
+ unsigned IfDrops; /* < Packets dropped by the card before going to the USB bus. */
+ /* < Not supported at the moment. */
+ unsigned Capt; /* < number of packets that pass the BPF filter, find place in the kernel buffer and */
+ /* < therefore reach the application. */
+}AirpcapStats, *PAirpcapStats;
+
+/*!
+ \brief Channel information.
+ Used by \ref AirpcapSetDeviceChannelEx(), \ref AirpcapGetDeviceChannelEx(), \ref AirpcapGetDeviceSupportedChannels()
+*/
+typedef struct _AirpcapChannelInfo
+{
+ unsigned Frequency; /* < Channel frequency, in MHz. */
+ /*!
+ \brief 802.11n specific. Offset of the extension channel in case of 40MHz channels.
+
+ Possible values are -1, 0 +1:
+ - -1 means that the extension channel should be below the control channel (e.g. Control = 5 and Extension = 1)
+ - 0 means that no extension channel should be used (20MHz channels or legacy mode)
+ - +1 means that the extension channel should be above the control channel (e.g. Control = 1 and Extension = 5)
+
+ In case of 802.11a/b/g channels (802.11n legacy mode), this field should be set to 0.
+ */
+ int8_t ExtChannel;
+ uint8_t Reserved[3]; /* < Reserved. It should be set to {0,0,0}. */
+}
+ AirpcapChannelInfo, *PAirpcapChannelInfo;
+
+
+/*@}*/
+
+/** @defgroup airpcapfuncs AirPcap functions
+ * @{
+ */
+
+/*!
+ \brief Return a string with the API version
+ \param VersionMajor Pointer to a variable that will be filled with the major version number.
+ \param VersionMinor Pointer to a variable that will be filled with the minor version number.
+ \param VersionRev Pointer to a variable that will be filled with the revision number.
+ \param VersionBuild Pointer to a variable that will be filled with the build number.
+*/
+void AirpcapGetVersion(unsigned * VersionMajor, unsigned * VersionMinor, unsigned * VersionRev, unsigned * VersionBuild);
+
+/*!
+ \brief Return the last error related to the specified handle
+ \param AdapterHandle Handle to an open adapter.
+ \return The string with the last error.
+*/
+char * AirpcapGetLastError(PAirpcapHandle AdapterHandle);
+
+/*!
+ \brief Return the list of available devices
+ \param PPAllDevs Address to a caller allocated pointer. On success this pointer will receive the head of a list of available devices.
+ \param Ebuf String that will contain error information if false is returned. The size of the string must be AIRPCAP_ERRBUF_SIZE bytes.
+ \return true on success. false is returned on failure, in which case Ebuf is filled in with an appropriate error message.
+
+ Here's a snippet of code that shows how to use AirpcapGetDeviceList():
+
+ \code
+ char Ebuf[AIRPCAP_ERRBUF_SIZE];
+ AirpcapDeviceDescription *Desc, *tDesc;
+
+ if(AirpcapGetDeviceList(&Desc, Ebuf) == -1)
+ {
+ printf("Unable to get the list of devices: %s\n", Ebuf);
+ return -1;
+ }
+
+ for(tDesc = Desc; tDesc; tDesc = tDesc->next)
+ {
+ printf("%u) %s (%s)\n",
+ ++i,
+ tDesc->Name,
+ tDesc->Description);
+ }
+
+ AirpcapFreeDeviceList(Desc);
+ \endcode
+*/
+bool AirpcapGetDeviceList(PAirpcapDeviceDescription *PPAllDevs, char * Ebuf);
+
+/*!
+ \brief Free a list of devices returned by AirpcapGetDeviceList()
+ \param PAllDevs Head of the list of devices returned by \ref AirpcapGetDeviceList().
+*/
+void AirpcapFreeDeviceList(PAirpcapDeviceDescription PAllDevs);
+
+/*!
+ \brief Open an adapter
+ \param DeviceName Name of the device to open. Use \ref AirpcapGetDeviceList() to get the list of devices.
+ \param Ebuf String that will contain error information in case of failure. The size of the string must be AIRPCAP_ERRBUF_SIZE bytes.
+ \return A PAirpcapHandle handle on success. NULL is returned on failure, in which case Ebuf is filled in with an appropriate error message.
+*/
+PAirpcapHandle AirpcapOpen(char * DeviceName, char * Ebuf);
+
+/*!
+ \brief Close an adapter
+ \param AdapterHandle Handle to the adapter to close.
+*/
+void AirpcapClose(PAirpcapHandle AdapterHandle);
+
+/*!
+ \brief Sets the monitor mode for the specified adapter
+ \param AdapterHandle Handle to the adapter.
+ \param MonitorModeEnabled If true, the adapter will be put in monitor mode. If false, the adapter will be configured
+ for normal operation.
+ \return true on success.
+
+ When monitor mode is on, the adapter captures all the packets transmitted on the channel. This includes:
+
+ - unicast packets
+ - multicast packets
+ - broadcast packets
+ - control and management packets
+
+ When monitor mode is off, the adapter has a filter on unicast packets to capture only the packets whose MAC
+ destination address equals to the adapter's address. This means the following frames will be received:
+
+ - unicast packets with the address of the adapter
+ - multicast packets
+ - broadcast packets
+ - beacons and probe requests
+
+ The main reason to turn monitor mode off is that, when not in monitor mode, the adapter will acknowledge the
+ data frames sent to its address. This is useful when the adapter needs to interact with other devices on the
+ 802.11 network, because handling the ACKs in software is too slow.
+
+ \note When an adapter is plugged into the system, it's always configured with monitor mode ON. The monitor mode
+ configuration is not stored persistently, so if you want to turn monitor mode off, you will need to do it
+ every time you open the adapter.
+*/
+bool AirpcapSetMonitorMode(PAirpcapHandle AdapterHandle, bool MonitorModeEnabled);
+
+/*!
+ \brief Returns true if the specified adapter is in monitor mode.
+ \param AdapterHandle Handle to the adapter.
+ \param PMonitorModeEnabled User-provided variable that will be set to true if the adapter is in monitor mode.
+ \return true if the operation is successful. false otherwise.
+
+ \note When an adapter is plugged into the system, it's always configured with monitor mode ON. The monitor mode
+ configuration is not stored persistently, so if you want to turn monitor mode off, you will need to do it
+ every time you open the adapter.
+*/
+bool AirpcapGetMonitorMode(PAirpcapHandle AdapterHandle, bool * PMonitorModeEnabled);
+
+/*!
+ \brief Set the link type of an adapter
+ \param AdapterHandle Handle to the adapter.
+ \param NewLinkType the "link type", i.e. the format of the frames that will be received from the adapter.
+ \return true on success.
+
+ the "link type" determines how the driver will encode the packets captured from the network.
+ Aircap supports two link types:
+ - AIRPCAP_LT_802_11, to capture 802.11 frames (including control frames) without any
+ power information. Look at the Capture_no_radio example application in the developer's pack
+ for a reference on how to decode 802.11 frames with this link type.
+ - AIRPCAP_LT_802_11_PLUS_RADIO, to capture 802.11 frames (including control frames) with a radiotap header
+ that contains power and channel information. More information about the radiotap header can be found in the
+ radiotap section. Moreover, the "Capture_radio" example application in
+ the developer's pack can be used as a reference on how to decode 802.11 frames with radiotap headers.
+ - AIRPCAP_LT_802_11_PLUS_PPI, to capture 802.11 frames (including control frames) with a Per Packet Information (PPI)
+ header that contains per-packet meta information like channel and power information. More details on the PPI header can
+ be found in the PPI online documentation (TODO).
+*/
+bool AirpcapSetLinkType(PAirpcapHandle AdapterHandle, AirpcapLinkType NewLinkType);
+
+/*!
+ \brief Get the link type of the specified adapter
+ \param AdapterHandle Handle to the adapter.
+ \param PLinkType Pointer to a caller allocated AirpcapLinkType variable that will contain the link type of the adapter.
+ \return true on success.
+
+ the "link type" determines how the driver will encode the packets captured from the network.
+ Aircap supports two link types:
+ - AIRPCAP_LT_802_11, to capture 802.11 frames (including control frames) without any
+ power information. Look at the Capture_no_radio example application in the developer's pack
+ for a reference on how to decode 802.11 frames with this link type.
+ - AIRPCAP_LT_802_11_PLUS_RADIO, to capture 802.11 frames (including control frames) with a radiotap header
+ that contains power and channel information. More information about the radiotap header can be found int the
+ radiotap section. Moreover, the "Capture_radio" example application in
+ the developer's pack can be used as a reference on how to decode 802.11 frames with radiotap headers.
+*/
+bool AirpcapGetLinkType(PAirpcapHandle AdapterHandle, PAirpcapLinkType PLinkType);
+
+/*!
+ \brief Configures the adapter on whether to include the MAC Frame Check Sequence in the captured packets.
+ \param AdapterHandle Handle to the adapter.
+ \param IsFcsPresent true if the packets should include the FCS. false otherwise
+ \return true on success.
+
+ In the default configuration, the adapter includes the FCS in the captured packets. The MAC Frame Check Sequence
+ is 4 bytes and is located at the end of the 802.11 packet, with both AIRPCAP_LT_802_11 and AIRPCAP_LT_802_11_PLUS_RADIO
+ link types.
+ When the FCS inclusion is turned on, and if the link type is AIRPCAP_LT_802_11_PLUS_RADIO, the radiotap header
+ that precedes each frame has two additional fields at the end: Padding and FCS. These two fields are not present
+ when FCS inclusion is off.
+*/
+bool AirpcapSetFcsPresence(PAirpcapHandle AdapterHandle, bool IsFcsPresent);
+
+/*!
+ \brief Returns true if the specified adapter includes the MAC Frame Check Sequence in the captured packets
+ \param AdapterHandle Handle to the adapter.
+ \param PIsFcsPresent User-provided variable that will be set to true if the adapter is including the FCS.
+ \return true if the operation is successful. false otherwise.
+
+ In the default configuration, the adapter has FCS inclusion turned on. The MAC Frame Check Sequence is 4 bytes
+ and is located at the end of the 802.11 packet, with both AIRPCAP_LT_802_11 and AIRPCAP_LT_802_11_PLUS_RADIO
+ link types.
+ When the FCS inclusion is turned on, and if the link type is AIRPCAP_LT_802_11_PLUS_RADIO, the radiotap header
+ that precedes each frame has two additional fields at the end: Padding and FCS. These two fields are not present
+ when FCS inclusion is off.
+*/
+bool AirpcapGetFcsPresence(PAirpcapHandle AdapterHandle, bool * PIsFcsPresent);
+
+/*!
+ \brief Configures the adapter to accept or drop frames with an incorrect Frame Check sequence (FCS).
+ \param AdapterHandle Handle to the adapter.
+ \param ValidationType The type of validation the driver will perform. See the documentation of \ref AirpcapValidationType for details.
+ \return true on success.
+
+ \note By default, the driver is configured in AIRPCAP_VT_ACCEPT_EVERYTHING mode.
+*/
+bool AirpcapSetFcsValidation(PAirpcapHandle AdapterHandle, AirpcapValidationType ValidationType);
+
+/*!
+ \brief Checks if the specified adapter is configured to capture frames with incorrect an incorrect Frame Check Sequence (FCS).
+ \param AdapterHandle Handle to the adapter.
+ \param ValidationType Pointer to a user supplied variable that will contain the type of validation the driver will perform. See the documentation of \ref AirpcapValidationType for details.
+ \return true if the operation is successful. false otherwise.
+
+ \note By default, the driver is configured in AIRPCAP_VT_ACCEPT_EVERYTHING mode.
+*/
+bool AirpcapGetFcsValidation(PAirpcapHandle AdapterHandle, PAirpcapValidationType ValidationType);
+
+/*!
+ \brief Set the list of decryption keys that the driver is going to use with the specified device.
+ \param AdapterHandle Handle an open adapter instance.
+ \param KeysCollection Pointer to a PAirpcapKeysCollection structure that contains the keys to be set in the driver.
+ \return true if the operation is successful. false otherwise.
+
+ The AirPcap driver is able to use a set of decryption keys to decrypt the traffic transmitted on a specific SSID. If one of the
+ keys corresponds to the one the frame has been encrypted with, the driver will perform decryption and return the cleartext frames
+ to the application.
+
+ This function allows to set the <b>adapter-specific</b> set of keys. These keys will be used by the specified adapter only,
+ and will not be used by other airpcap devices besides the specified one.
+
+ At this time, the only supported decryption method is WEP.
+
+ The keys are applied to the packets in the same order they appear in the KeysCollection structure until the packet is
+ correctly decrypted, therefore putting frequently used keys at the beginning of the structure improves performance.
+
+ \note: when you change the set of keys from an open capture instance, the change will be
+ immediately reflected on all the other capture instances.
+*/
+bool AirpcapSetDeviceKeys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection);
+
+/*!
+ \brief Returns the list of decryption keys in the driver that are currently associated with the specified device
+ \param AdapterHandle Handle to an open adapter instance.
+ \param KeysCollection User-allocated PAirpcapKeysCollection structure that will be filled with the keys.
+ \param PKeysCollectionSize \b IN: pointer to a user-allocated variable that contains the length of the KeysCollection structure, in bytes.
+ \b OUT: amount of data moved by the driver in the buffer pointed by KeysBuffer, in bytes.
+ \return true if the operation is successful. If an error occurs, the return value is false and KeysCollectionSize is zero.
+ If the provided buffer is too small to contain the keys, the return value is false and KeysCollectionSize contains the
+ needed KeysCollection length, in bytes. If the device doesn't have any decryption key configured, the return value is true, and
+ KeysCollectionSize will be zero.
+
+ This function returns the <b>adapter-specific</b> set of keys. These keys are used by the specified adapter only,
+ and not by other airpcap devices besides the specified one.
+
+ The AirPcap driver is able to use a set of decryption keys to decrypt the traffic transmitted on a specific SSID. If one of the
+ keys corresponds to the one the frame has been encrypted with, the driver will perform decryption and return the cleartext frames
+ to the application.
+ The driver supports, for every device, multiple keys at the same time.
+
+ The configured decryption keys are device-specific, therefore AirpcapGetDeviceKeys() will return a different set of keys
+ when called on different devices.
+
+ At this time, the only supported decryption method is WEP.
+*/
+bool AirpcapGetDeviceKeys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize);
+
+/*!
+ \brief Set the global list of decryption keys that the driver is going to use with all the devices.
+ \param AdapterHandle Handle an open adapter instance.
+ \param KeysCollection Pointer to a PAirpcapKeysCollection structure that contains the keys to be set in the driver.
+ \return true if the operation is successful. false otherwise.
+
+ The AirPcap driver is able to use a set of decryption keys to decrypt the traffic transmitted on a specific SSID. If one of the
+ keys corresponds to the one the frame has been encrypted with, the driver will perform decryption and return the cleartext frames
+ to the application.
+
+ This function allows to set the <b>global driver</b> set of keys. These keys will be used by all the adapters plugged in
+ the machine.
+
+ At this time, the only supported decryption method is WEP.
+
+ The keys are applied to the packets in the same order they appear in the KeysCollection structure until the packet is
+ correctly decrypted, therefore putting frequently used keys at the beginning of the structure improves performance.
+
+ \note: when you change the set of keys from an open capture instance, the change will be
+ immediately reflected on all the other capture instances.
+*/
+bool AirpcapSetDriverKeys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection);
+
+/*!
+ \brief Returns the global list of decryption keys in the driver that are associated with all the devices.
+ \param AdapterHandle Handle to an open adapter instance.
+ \param KeysCollection User-allocated PAirpcapKeysCollection structure that will be filled with the keys.
+ \param PKeysCollectionSize \b IN: pointer to a user-allocated variable that contains the length of the KeysCollection structure, in bytes.
+ \b OUT: amount of data moved by the driver in the buffer pointed by KeysBuffer, in bytes.
+ \return true if the operation is successful. If an error occurs, the return value is false and KeysCollectionSize is zero.
+ If the provided buffer is too small to contain the keys, the return value is false and KeysCollectionSize contains the
+ needed KeysCollection length, in bytes. If the device doesn't have any decryption key configured, the return value is true, and
+ KeysCollectionSize will be zero.
+
+ This function returns the <b>global driver</b> set of keys. These keys will be used by all the adapters plugged in
+ the machine.
+
+ The AirPcap driver is able to use a set of decryption keys to decrypt the traffic transmitted on a specific SSID. If one of the
+ keys corresponds to the one the frame has been encrypted with, the driver will perform decryption and return the cleartext frames
+ to the application.
+
+ At this time, the only supported decryption method is WEP.
+*/
+bool AirpcapGetDriverKeys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize);
+
+/*!
+ \brief Turns on or off the decryption of the incoming frames with the <b>adapter-specific</b> keys.
+ \param AdapterHandle Handle to the adapter.
+ \param Enable Either AIRPCAP_DECRYPTION_ON or AIRPCAP_DECRYPTION_OFF
+ \return true on success.
+
+ The adapter-specific decryption keys can be configured with the \ref AirpcapSetDeviceKeys() function.
+ \note By default, the driver is configured with AIRPCAP_DECRYPTION_ON.
+*/
+bool AirpcapSetDecryptionState(PAirpcapHandle AdapterHandle, AirpcapDecryptionState Enable);
+
+/*!
+ \brief Tells if this open instance is configured to perform the decryption of the incoming frames with the <b>adapter-specific</b> keys.
+ \param AdapterHandle Handle to the adapter.
+ \param PEnable Pointer to a user supplied variable that will contain the decryption configuration. See \ref _AirpcapDecryptionState for details.
+ \return true if the operation is successful. false otherwise.
+
+ The adapter-specific decryption keys can be configured with the \ref AirpcapSetDeviceKeys() function.
+ \note By default, the driver is configured with AIRPCAP_DECRYPTION_ON.
+*/
+bool AirpcapGetDecryptionState(PAirpcapHandle AdapterHandle, PAirpcapDecryptionState PEnable);
+
+/*!
+ \brief Turns on or off the decryption of the incoming frames with the <b>global driver</b> set of keys.
+ \param AdapterHandle Handle to the adapter.
+ \param Enable Either AIRPCAP_DECRYPTION_ON or AIRPCAP_DECRYPTION_OFF
+ \return true on success.
+
+ The global decryption keys can be configured with the \ref AirpcapSetDriverKeys() function.
+ \note By default, the driver is configured with AIRPCAP_DECRYPTION_ON.
+*/
+bool AirpcapSetDriverDecryptionState(PAirpcapHandle AdapterHandle, AirpcapDecryptionState Enable);
+
+/*!
+ \brief Tells if this open instance is configured to perform the decryption of the incoming frames with the <b>global driver</b> set of keys.
+ \param AdapterHandle Handle to the adapter.
+ \param PEnable Pointer to a user supplied variable that will contain the decryption configuration. See \ref _AirpcapDecryptionState for details.
+ \return true if the operation is successful. false otherwise.
+
+ The global decryption keys can be configured with the \ref AirpcapSetDriverKeys() function.
+ \note By default, the driver is configured with AIRPCAP_DECRYPTION_ON.
+*/
+bool AirpcapGetDriverDecryptionState(PAirpcapHandle AdapterHandle, PAirpcapDecryptionState PEnable);
+
+/*!
+ \brief Set the radio channel of a device
+ \param AdapterHandle Handle to the adapter.
+ \param Channel the new channel to set.
+ \return true on success.
+
+ The list of available channels can be retrieved with \ref AirpcapGetDeviceSupportedChannels(). The default channel setting is 6.
+
+ \note this is a device-related function: when you change the channel from an open capture instance, the change will be
+ immediately reflected on all the other capture instances.
+*/
+bool AirpcapSetDeviceChannel(PAirpcapHandle AdapterHandle, unsigned Channel);
+
+/*!
+ \brief Get the radio channel of a device
+ \param AdapterHandle Handle to the adapter.
+ \param PChannel Pointer to a user-supplied variable into which the function will copy the currently configured radio channel.
+ \return true on success.
+
+ The list of available channels can be retrieved with \ref AirpcapGetDeviceSupportedChannels(). The default channel setting is 6.
+
+ \note this is a device-related function: when you change the channel from an open capture instance, the change will be
+ immediately reflected on all the other capture instances.
+*/
+bool AirpcapGetDeviceChannel(PAirpcapHandle AdapterHandle, unsigned * PChannel);
+
+/*!
+ \brief Set the size of the kernel packet buffer for this adapter
+ \param AdapterHandle Handle to the adapter.
+ \param BufferSize New size, in bytes.
+ \return true on success.
+
+ Every AirPcap open instance has an associated kernel buffer, whose default size is 1 Mbyte.
+ This function can be used to change the size of this buffer, and can be called at any time.
+ A bigger kernel buffer size decreases the risk of dropping packets during network bursts or when the
+ application is busy, at the cost of higher kernel memory usage.
+
+ \note don't use this function unless you know what you are doing. Due to caching issues and bigger non-paged
+ memory consumption, bigger buffer sizes can decrease the capture performance instead of improving it.
+*/
+bool AirpcapSetKernelBuffer(PAirpcapHandle AdapterHandle, unsigned BufferSize);
+
+/*!
+ \brief Get the size of the kernel packet buffer for this adapter
+ \param AdapterHandle Handle to the adapter.
+ \param PSizeBytes User-allocated variable that will be filled with the size of the kernel buffer.
+ \return true on success.
+
+ Every AirPcap open instance has an associated kernel buffer, whose default size is 1 Mbyte.
+ This function can be used to get the size of this buffer.
+*/
+bool AirpcapGetKernelBufferSize(PAirpcapHandle AdapterHandle, unsigned * PSizeBytes);
+
+/*!
+ \brief Saves the configuration of the specified adapter in the registry, so that it becomes the default for this adapter.
+ \param AdapterHandle Handle to the adapter.
+ \return true on success. false on failure.
+
+ Almost all the AirPcap calls that modify the configuration (\ref AirpcapSetLinkType(), \ref AirpcapSetFcsPresence(),
+ \ref AirpcapSetFcsValidation(), \ref AirpcapSetKernelBuffer(), \ref AirpcapSetMinToCopy())
+ affect only the referenced AirPcap open instance. This means that if you do another \ref AirpcapOpen() on the same
+ adapter, the configuration changes will not be remembered, and the new adapter handle will have default configuration
+ settings.
+
+ Exceptions to this rule are the \ref AirpcapSetDeviceChannel() and \ref AirpcapSetDeviceKeys() functions: a channel change is
+ reflected on all the open instances, and remembered until the next call to \ref AirpcapSetDeviceChannel(), until the adapter
+ is unplugged, or until the machine is powered off. Same thing for the configuration of the WEP keys.
+
+ AirpcapStoreCurConfigAsAdapterDefault() stores the configuration of the give open instance as the default for the adapter:
+ all the instances opened in the future will have the same configuration that this adapter currently has.
+ The configuration is stored in the registry, therefore it is remembered even when the adapter is unplugged or the
+ machine is turned off. However, an adapter doesn't bring its configuration with it from machine to machine.
+
+ the configuration information saved in the registry includes the following parameters:
+ - channel
+ - kernel buffer size
+ - mintocopy
+ - link type
+ - CRC presence
+ - Encryption keys
+ - Encryption Enabled/Disabled state
+
+ The configuration is adapter-specific. This means that changing the configuration of an adapter
+ doesn't modify the one of the other adapters that are currently used or that will be used in the future.
+
+ \note AirpcapStoreCurConfigAsAdapterDefault() must have exclusive access to the adapter -- it
+ will fail if more than one AirPcap handle is opened at the same time for this adapter.
+ AirpcapStoreCurConfigAsAdapterDefault() needs administrator privileges. It will fail if the calling user
+ is not a local machine administrator.
+*/
+bool AirpcapStoreCurConfigAsAdapterDefault(PAirpcapHandle AdapterHandle);
+
+/*!
+ \brief Set the BPF kernel filter for an adapter
+ \param AdapterHandle Handle to the adapter.
+ \param Instructions pointer to the first BPF instruction in the array. Corresponds to the bf_insns
+ in a bpf_program structure (see the WinPcap documentation at https://www.winpcap.org/devel.htm).
+ \param Len Number of instructions in the array pointed by the previous field. Corresponds to the bf_len in
+ a bpf_program structure (see the WinPcap documentation at https://www.winpcap.org/devel.htm).
+ \return true on success.
+
+ The AirPcap driver is able to perform kernel-level filtering using the standard BPF pseudo-machine format. You can read
+ the WinPcap documentation at https://www.winpcap.org/devel.htm for more details on the BPF filtering mechanism.
+
+ A filter can be automatically created by using the pcap_compile() function of the WinPcap API. This function
+ converts a human readable text expression with the tcpdump/libpcap syntax into a BPF program.
+ If your program doesn't link wpcap, but you need to generate the code for a particular filter, you can run WinDump
+ with the -d or -dd or -ddd flags to obtain the pseudocode.
+
+*/
+bool AirpcapSetFilter(PAirpcapHandle AdapterHandle, void * Instructions, unsigned Len);
+
+/*!
+ \brief Return the MAC address of an adapter.
+ \param AdapterHandle Handle to the adapter.
+ \param PMacAddress Pointer to a user allocated MAC address.
+ The size of this buffer needs to be at least 6 bytes.
+ \return true on success.
+*/
+bool AirpcapGetMacAddress(PAirpcapHandle AdapterHandle, PAirpcapMacAddress PMacAddress);
+
+/*!
+ \brief Set the mintocopy parameter for an open adapter
+ \param AdapterHandle Handle to the adapter.
+ \param MinToCopy is the mintocopy size in bytes.
+ \return true on success.
+
+ When the number of bytes in the kernel buffer changes from less than mintocopy bytes to greater than or equal to mintocopy bytes,
+ the read event is signalled (see \ref AirpcapGetReadEvent()). A high value for mintocopy results in poor responsiveness since the
+ driver may signal the application "long" after the arrival of the packet. And a high value results in low CPU loading
+ by minimizing the number of user/kernel context switches.
+ A low MinToCopy results in good responsiveness since the driver will signal the application close to the arrival time of
+ the packet. This has higher CPU loading over the first approach.
+*/
+bool AirpcapSetMinToCopy(PAirpcapHandle AdapterHandle, unsigned MinToCopy);
+
+/*!
+ \brief Gets an event that is signaled when that is signalled when packets are available in the kernel buffer (see \ref AirpcapSetMinToCopy()).
+ \param AdapterHandle Handle to the adapter.
+ \param PReadEvent Pointer to a user-supplied handle that in which the read event will be copied.
+ \return true on success.
+
+ \note the event is signalled when at least mintocopy bytes are present in the kernel buffer (see \ref AirpcapSetMinToCopy()).
+ This event can be used by WaitForSingleObject() and WaitForMultipleObjects() to create blocking behavior when reading
+ packets from one or more adapters (see \ref AirpcapRead()).
+*/
+bool AirpcapGetReadEvent(PAirpcapHandle AdapterHandle, void *** PReadEvent);
+
+/*!
+ \brief Fills a user-provided buffer with zero or more packets that have been captured on the referenced adapter.
+ \param AdapterHandle Handle to the adapter.
+ \param Buffer pointer to the buffer that will be filled with captured packets.
+ \param BufSize size of the input buffer that will contain the packets, in bytes.
+ \param PReceievedBytes Pointer to a user supplied variable that will receive the number of bytes copied by AirpcapRead.
+ Can be smaller than BufSize.
+ \return true on success.
+
+ 802.11 frames are returned by the driver in buffers. Every 802.11 frame in the buffer is preceded by a \ref AirpcapBpfHeader structure.
+ The suggested way to use an AirPcap adapter is through the pcap API exported by wpcap.dll. If this is not
+ possible, the Capture_radio and Capture_no_radio examples in the AirPcap developer's pack show how to properly decode the
+ packets in the read buffer returned by AirpcapRead().
+
+ \note this function is NOT blocking. Blocking behavior can be obtained using the event returned
+ by \ref AirpcapGetReadEvent(). See also \ref AirpcapSetMinToCopy().
+*/
+bool AirpcapRead(PAirpcapHandle AdapterHandle, uint8_t * Buffer, unsigned BufSize, unsigned * PReceievedBytes);
+
+/*!
+ \brief Transmits a packet.
+ \param AdapterHandle Handle to the adapter.
+ \param TxPacket Pointer to a buffer that contains the packet to be transmitted.
+ \param PacketLen Length of the buffer pointed by the TxPacket argument, in bytes.
+ \return true on success.
+
+ The packet will be transmitted on the channel the device is currently set. To change the device adapter, use the
+ \ref AirpcapSetDeviceChannel() function.
+
+ If the linktype of the adapter is AIRPCAP_LT_802_11, the buffer pointed by TxPacket should contain just the 802.11
+ packet, without additional information. The packet will be transmitted at 1Mbps.
+
+ If the linktype of the adapter is AIRPCAP_LT_802_11_PLUS_RADIO, the buffer pointed by TxPacket should contain a radiotap
+ header followed by the 802.11 packet. AirpcapWrite will use the rate information in the radiotap header when
+ transmitting the packet.
+*/
+bool AirpcapWrite(PAirpcapHandle AdapterHandle, char * TxPacket, uint32_t PacketLen);
+
+/*!
+ \brief Get per-adapter WinPcap-compatible capture statistics.
+ \param AdapterHandle Handle to the adapter.
+ \param PStats pointer to a user-allocated AirpcapStats structure that will be filled with statistical information.
+ \return true on success.
+*/
+bool AirpcapGetStats(PAirpcapHandle AdapterHandle, PAirpcapStats PStats);
+
+/*!
+ \brief Get the number of LEDs the referenced adapter has available.
+ \param AdapterHandle Handle to the adapter.
+ \param NumberOfLeds Number of LEDs available on this adapter.
+ \return true on success.
+*/
+bool AirpcapGetLedsNumber(PAirpcapHandle AdapterHandle, unsigned * NumberOfLeds);
+
+/*!
+ \brief Turn on one of the adapter's LEDs.
+ \param AdapterHandle Handle to the adapter.
+ \param LedNumber zero-based identifier of the LED to turn on.
+ \return true on success.
+*/
+bool AirpcapTurnLedOn(PAirpcapHandle AdapterHandle, unsigned LedNumber);
+
+/*!
+ \brief Turn off one of the adapter's LEDs.
+ \param AdapterHandle Handle to the adapter.
+ \param LedNumber zero-based identifier of the LED to turn off.
+ \return true on success.
+*/
+bool AirpcapTurnLedOff(PAirpcapHandle AdapterHandle, unsigned LedNumber);
+
+/*!
+ \brief Set the channel of a device through its radio frequency. In case of 802.11n enabled devices, it sets the extension channel, if used.
+ \param AdapterHandle Handle to the adapter.
+ \param ChannelInfo The new channel information to set.
+ \return true on success.
+
+ \note this is a device-related function: when you change the channel from an open capture instance, the change will be
+ immediately reflected on all the other capture instances.
+*/
+bool AirpcapSetDeviceChannelEx(PAirpcapHandle AdapterHandle, AirpcapChannelInfo ChannelInfo);
+
+/*!
+ \brief Get the channel of a device through its radiofrequency. In case of 802.11n enabled devices, it gets the extension channel, if in use.
+ \param AdapterHandle Handle to the adapter.
+ \param PChannelInfo Pointer to a user-supplied variable into which the function will copy the currently configured channel information.
+ \return true on success.
+
+ \note this is a device-related function: when you change the channel from an open capture instance, the change will be
+ immediately reflected on all the other capture instances.
+*/
+bool AirpcapGetDeviceChannelEx(PAirpcapHandle AdapterHandle, PAirpcapChannelInfo PChannelInfo);
+
+/*!
+ \brief Get the list of supported channels for a given device. In case of a 802.11n capable device, information related to supported extension channels is also reported.
+
+ Every control channel is listed multiple times, one for each different supported extension channel. For example channel 6 (2437MHz) is usually listed three times:
+ - <b>Frequency 2437 Extension +1</b>. Control channel is 6, extension channel is 10.
+ - <b>Frequency 2437 Extension 0</b>. Control channel is 6, no extension channel is used (20MHz channel and legacy mode).
+ - <b>Frequency 2437 Extension -1</b>. Control channel is 6, extension channel is 2.
+ \param AdapterHandle Handle to the adapter.
+ \param ppChannelInfo Pointer to a user-supplied variable that will point to an array of supported channel. Such list must not be freed by the caller
+ \param pNumChannelInfo Number of channels returned in the array.
+ \return true on success.
+
+ \note The supported channels are not listed in any specific order.
+*/
+bool AirpcapGetDeviceSupportedChannels(PAirpcapHandle AdapterHandle, PAirpcapChannelInfo *ppChannelInfo, unsigned * pNumChannelInfo);
+
+/*!
+ \brief Converts a given frequency to the corresponding channel.
+
+ \param Frequency Frequency of the channel, in MHz.
+ \param PChannel Pointer to a user-supplied variable that will contain the channel number on success.
+ \param PBand Pointer to a user-supplied variable that will contain the band (a or b/g) of the given channel.
+ \return true on success, i.e. the frequency corresponds to a valid a or b/g channel.
+*/
+bool AirpcapConvertFrequencyToChannel(unsigned Frequency, unsigned * PChannel, PAirpcapChannelBand PBand);
+
+/*!
+ \brief Converts a given channel to the corresponding frequency.
+
+ \param Channel Channel number to be converted.
+ \param PFrequency Pointer to a user-supplied variable that will contain the channel frequency in MHz on success.
+ \return true on success, i.e. the given channel number exists.
+*/
+bool AirpcapConvertChannelToFrequency(unsigned Channel, unsigned * PFrequency);
+
+
+/*@}*/
+
+#endif /* __AIRPCAP_DRIVER__ */
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* !defined(AIRPCAP_H__EAE405F5_0171_9592_B3C2_C19EC426AD34__INCLUDED_) */
diff --git a/capture/airpcap_loader.c b/capture/airpcap_loader.c
new file mode 100644
index 00000000..c57f93f4
--- /dev/null
+++ b/capture/airpcap_loader.c
@@ -0,0 +1,1229 @@
+/* airpcap_loader.c
+ *
+ * Giorgio Tino <giorgio.tino@cacetech.com>
+ * Copyright (c) CACE Technologies, LLC 2006
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 2000 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+
+#include <glib.h>
+
+#include <epan/crypt/dot11decrypt_ws.h>
+#include <epan/strutil.h>
+#include <wsutil/file_util.h>
+#include <wsutil/802_11-utils.h>
+
+#include <capture/airpcap.h>
+#include <capture/airpcap_loader.h>
+
+
+/*
+ * Set to true if the DLL was successfully loaded AND all functions
+ * are present.
+ */
+static bool AirpcapLoaded = false;
+
+#ifdef _WIN32
+/*
+ * We load dynamically the dag library in order link it only when
+ * it's present on the system
+ */
+static void * AirpcapLib = NULL;
+
+static AirpcapGetLastErrorHandler g_PAirpcapGetLastError;
+static AirpcapSetKernelBufferHandler g_PAirpcapSetKernelBuffer;
+static AirpcapSetFilterHandler g_PAirpcapSetFilter;
+static AirpcapGetMacAddressHandler g_PAirpcapGetMacAddress;
+static AirpcapSetMinToCopyHandler g_PAirpcapSetMinToCopy;
+static AirpcapGetReadEventHandler g_PAirpcapGetReadEvent;
+static AirpcapReadHandler g_PAirpcapRead;
+static AirpcapGetStatsHandler g_PAirpcapGetStats;
+#endif
+
+static int AirpcapVersion = 3;
+
+static AirpcapGetDeviceListHandler g_PAirpcapGetDeviceList;
+static AirpcapFreeDeviceListHandler g_PAirpcapFreeDeviceList;
+static AirpcapOpenHandler g_PAirpcapOpen;
+static AirpcapCloseHandler g_PAirpcapClose;
+static AirpcapGetLinkTypeHandler g_PAirpcapGetLinkType;
+static AirpcapSetLinkTypeHandler g_PAirpcapSetLinkType;
+static AirpcapTurnLedOnHandler g_PAirpcapTurnLedOn;
+static AirpcapTurnLedOffHandler g_PAirpcapTurnLedOff;
+static AirpcapGetDeviceChannelHandler g_PAirpcapGetDeviceChannel;
+static AirpcapSetDeviceChannelHandler g_PAirpcapSetDeviceChannel;
+static AirpcapGetFcsPresenceHandler g_PAirpcapGetFcsPresence;
+static AirpcapSetFcsPresenceHandler g_PAirpcapSetFcsPresence;
+static AirpcapGetFcsValidationHandler g_PAirpcapGetFcsValidation;
+static AirpcapSetFcsValidationHandler g_PAirpcapSetFcsValidation;
+static AirpcapGetDeviceKeysHandler g_PAirpcapGetDeviceKeys;
+static AirpcapSetDeviceKeysHandler g_PAirpcapSetDeviceKeys;
+static AirpcapGetDriverKeysHandler g_PAirpcapGetDriverKeys;
+static AirpcapSetDriverKeysHandler g_PAirpcapSetDriverKeys;
+static AirpcapGetDecryptionStateHandler g_PAirpcapGetDecryptionState;
+static AirpcapSetDecryptionStateHandler g_PAirpcapSetDecryptionState;
+static AirpcapGetDriverDecryptionStateHandler g_PAirpcapGetDriverDecryptionState;
+static AirpcapSetDriverDecryptionStateHandler g_PAirpcapSetDriverDecryptionState;
+static AirpcapStoreCurConfigAsAdapterDefaultHandler g_PAirpcapStoreCurConfigAsAdapterDefault;
+static AirpcapGetVersionHandler g_PAirpcapGetVersion;
+static AirpcapSetDeviceChannelExHandler g_PAirpcapSetDeviceChannelEx;
+static AirpcapGetDeviceChannelExHandler g_PAirpcapGetDeviceChannelEx;
+static AirpcapGetDeviceSupportedChannelsHandler g_PAirpcapGetDeviceSupportedChannels;
+
+/* Airpcap interface list */
+GList *g_airpcap_if_list = NULL;
+
+/* Airpcap current selected interface */
+airpcap_if_info_t *airpcap_if_selected = NULL;
+
+/* Airpcap current active interface */
+airpcap_if_info_t *airpcap_if_active = NULL;
+
+Dot11Channel *pSupportedChannels;
+unsigned numSupportedChannels;
+
+static AirpcapChannelInfo LegacyChannels[] =
+{
+ {2412, 0, {0,0,0}},
+ {2417, 0, {0,0,0}},
+ {2422, 0, {0,0,0}},
+ {2427, 0, {0,0,0}},
+ {2432, 0, {0,0,0}},
+ {2437, 0, {0,0,0}},
+ {2442, 0, {0,0,0}},
+ {2447, 0, {0,0,0}},
+ {2452, 0, {0,0,0}},
+ {2457, 0, {0,0,0}},
+ {2462, 0, {0,0,0}},
+ {2467, 0, {0,0,0}},
+ {2472, 0, {0,0,0}},
+ {2484, 0, {0,0,0}},
+};
+
+static unsigned num_legacy_channels = 14;
+
+/*
+ * Get an error message string for a CANT_GET_INTERFACE_LIST error from
+ * "get_airpcap_interface_list()".
+ */
+static char *
+cant_get_airpcap_if_list_error_message(const char *err_str)
+{
+ return ws_strdup_printf("Can't get list of Wireless interfaces: %s", err_str);
+}
+
+/*
+ * Airpcap wrapper, used to store the current settings for the selected adapter
+ */
+bool
+airpcap_if_store_cur_config_as_adapter_default(PAirpcapHandle ah)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapStoreCurConfigAsAdapterDefault(ah);
+}
+
+/*
+ * Airpcap wrapper, used to open an airpcap adapter
+ */
+PAirpcapHandle
+airpcap_if_open(char * name, char * err)
+{
+ if (!AirpcapLoaded) return NULL;
+ if (name == NULL) return NULL;
+ return g_PAirpcapOpen(name,err);
+}
+
+/*
+ * Airpcap wrapper, used to close an airpcap adapter
+ */
+void
+airpcap_if_close(PAirpcapHandle handle)
+{
+ if (!AirpcapLoaded) return;
+ g_PAirpcapClose(handle);
+}
+
+/*
+ * Retrieve the state of the Airpcap DLL
+ */
+int
+airpcap_get_dll_state(void)
+{
+ return AirpcapVersion;
+}
+
+/*
+ * Airpcap wrapper, used to turn on the led of an airpcap adapter
+ */
+bool
+airpcap_if_turn_led_on(PAirpcapHandle AdapterHandle, unsigned LedNumber)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapTurnLedOn(AdapterHandle,LedNumber);
+}
+
+/*
+ * Airpcap wrapper, used to turn off the led of an airpcap adapter
+ */
+bool
+airpcap_if_turn_led_off(PAirpcapHandle AdapterHandle, unsigned LedNumber)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapTurnLedOff(AdapterHandle,LedNumber);
+}
+
+/*
+ * Airpcap wrapper, used to get the channel of an airpcap adapter
+ */
+bool
+airpcap_if_get_device_channel(PAirpcapHandle ah, unsigned * ch)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapGetDeviceChannel(ah,ch);
+}
+
+/*
+ * Airpcap wrapper, used to get the supported channels of an airpcap adapter
+ */
+bool
+airpcap_if_get_device_supported_channels(PAirpcapHandle ah, AirpcapChannelInfo **cInfo, unsigned * nInfo)
+{
+ if (!AirpcapLoaded) return false;
+ if (airpcap_get_dll_state() == AIRPCAP_DLL_OLD) {
+ *nInfo = num_legacy_channels;
+ *cInfo = (AirpcapChannelInfo*)&LegacyChannels;
+
+ return true;
+ } else if (airpcap_get_dll_state() == AIRPCAP_DLL_OK) {
+ return g_PAirpcapGetDeviceSupportedChannels(ah, cInfo, nInfo);
+ }
+ return false;
+}
+
+/*
+ * Airpcap wrapper, used to get the supported channels of an airpcap adapter
+ */
+Dot11Channel*
+airpcap_if_get_device_supported_channels_array(PAirpcapHandle ah, unsigned * pNumSupportedChannels)
+{
+ AirpcapChannelInfo *chanInfo;
+ unsigned numInfo = 0;
+
+ if (!AirpcapLoaded)
+ return NULL;
+ if (airpcap_if_get_device_supported_channels(ah, &chanInfo, &numInfo) == false)
+ return NULL;
+ numSupportedChannels = 0;
+
+ /*
+ * allocate a bigger array
+ */
+ if (numInfo == 0)
+ return NULL;
+
+ pSupportedChannels = (Dot11Channel *)g_malloc(numInfo * (sizeof *pSupportedChannels));
+
+ for (unsigned i = 0; i < numInfo; i++)
+ {
+ unsigned supportedChannel = G_MAXUINT;
+
+ /*
+ * search if we have it already
+ */
+ for (unsigned j = 0; j < numSupportedChannels; j++)
+ {
+ if (pSupportedChannels[j].Frequency == chanInfo[i].Frequency)
+ {
+ supportedChannel = j;
+ break;
+ }
+ }
+
+ if (supportedChannel == G_MAXUINT)
+ {
+ /*
+ * not found, create a new item
+ */
+ pSupportedChannels[numSupportedChannels].Frequency = chanInfo[i].Frequency;
+
+ switch(chanInfo[i].ExtChannel)
+ {
+ case -1:
+ pSupportedChannels[numSupportedChannels].Flags = FLAG_CAN_BE_LOW;
+ break;
+ case +1:
+ pSupportedChannels[numSupportedChannels].Flags = FLAG_CAN_BE_HIGH;
+ break;
+ case 0:
+ default:
+ pSupportedChannels[numSupportedChannels].Flags = 0;
+ }
+
+ /*
+ * Gather channel information
+ */
+
+ pSupportedChannels[numSupportedChannels].Flags |=
+ FREQ_IS_BG(pSupportedChannels[numSupportedChannels].Frequency) ?
+ FLAG_IS_BG_CHANNEL : FLAG_IS_A_CHANNEL;
+ pSupportedChannels[numSupportedChannels].Channel =
+ ieee80211_mhz_to_chan(pSupportedChannels[numSupportedChannels].Frequency);
+ numSupportedChannels++;
+ }
+ else
+ {
+ /*
+ * just update the ext channel flags
+ */
+ switch(chanInfo[i].ExtChannel)
+ {
+ case -1:
+ pSupportedChannels[supportedChannel].Flags |= FLAG_CAN_BE_LOW;
+ break;
+ case +1:
+ pSupportedChannels[supportedChannel].Flags |= FLAG_CAN_BE_HIGH;
+ break;
+ case 0:
+ default:
+ break;
+ }
+ }
+ }
+
+ if (numSupportedChannels < 1)
+ return NULL;
+ /*
+ * Now sort the list by frequency
+ */
+ for (unsigned i = 0; i < numSupportedChannels - 1; i++)
+ {
+ for (unsigned j = i + 1; j < numSupportedChannels; j++)
+ {
+ if (pSupportedChannels[i].Frequency > pSupportedChannels[j].Frequency)
+ {
+ Dot11Channel temp = pSupportedChannels[i];
+ pSupportedChannels[i] = pSupportedChannels[j];
+ pSupportedChannels[j] = temp;
+ }
+ }
+ }
+
+ *pNumSupportedChannels = numSupportedChannels;
+ return pSupportedChannels;
+}
+
+/*
+ * Airpcap wrapper, used to set the channel of an airpcap adapter
+ */
+bool
+airpcap_if_set_device_channel(PAirpcapHandle ah, unsigned ch)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapSetDeviceChannel(ah,ch);
+}
+
+/*
+ * Airpcap wrapper, used to set the frequency of an airpcap adapter
+ */
+bool
+airpcap_if_set_device_channel_ex(PAirpcapHandle ah, AirpcapChannelInfo ChannelInfo)
+{
+ if (!AirpcapLoaded) return false;
+ if (airpcap_get_dll_state() == AIRPCAP_DLL_OLD){
+ int channel = 0;
+ channel = ieee80211_mhz_to_chan(ChannelInfo.Frequency);
+
+ if (channel < 0){
+ return false;
+ } else {
+ return airpcap_if_set_device_channel(ah, channel);
+ }
+ } else if (airpcap_get_dll_state() == AIRPCAP_DLL_OK){
+ return g_PAirpcapSetDeviceChannelEx (ah, ChannelInfo);
+ }
+
+ return false;
+}
+
+/*
+ * Airpcap wrapper, used to get the frequency of an airpcap adapter
+ */
+bool
+airpcap_if_get_device_channel_ex(PAirpcapHandle ah, PAirpcapChannelInfo pChannelInfo)
+{
+ if (!AirpcapLoaded) return false;
+
+ pChannelInfo->Frequency = 0;
+ pChannelInfo->ExtChannel = 0;
+ pChannelInfo->Reserved[0] = 0;
+ pChannelInfo->Reserved[1] = 0;
+ pChannelInfo->Reserved[2] = 0;
+
+ if (airpcap_get_dll_state() == AIRPCAP_DLL_OLD){
+ unsigned channel = 0;
+ unsigned chan_freq = 0;
+
+ if (!airpcap_if_get_device_channel(ah, &channel)) return false;
+
+ chan_freq = ieee80211_chan_to_mhz(channel, true);
+ if (chan_freq == 0) return false;
+ pChannelInfo->Frequency = chan_freq;
+
+ return true;
+ } else if (airpcap_get_dll_state() == AIRPCAP_DLL_OK){
+ return g_PAirpcapGetDeviceChannelEx (ah, pChannelInfo);
+ }
+ return false;
+}
+
+/*
+ * Airpcap wrapper, used to get the link type of an airpcap adapter
+ */
+bool
+airpcap_if_get_link_type(PAirpcapHandle ah, PAirpcapLinkType lt)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapGetLinkType(ah,lt);
+}
+
+/*
+ * Airpcap wrapper, used to set the link type of an airpcap adapter
+ */
+bool
+airpcap_if_set_link_type(PAirpcapHandle ah, AirpcapLinkType lt)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapSetLinkType(ah,lt);
+}
+
+/*
+ * Airpcap wrapper, used to get the fcs presence of an airpcap adapter
+ */
+bool
+airpcap_if_get_fcs_presence(PAirpcapHandle ah, bool * fcs)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapGetFcsPresence(ah,fcs);
+}
+
+/*
+ * Airpcap wrapper, used to set the fcs presence of an airpcap adapter
+ */
+bool
+airpcap_if_set_fcs_presence(PAirpcapHandle ah, bool fcs)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapSetFcsPresence(ah,fcs);
+}
+
+/*
+ * Airpcap wrapper, used to get the decryption enabling of an airpcap adapter
+ */
+bool
+airpcap_if_get_decryption_state(PAirpcapHandle ah, PAirpcapDecryptionState PEnable)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapGetDecryptionState(ah,PEnable);
+}
+
+/*
+ * Airpcap wrapper, used to set the decryption enabling of an airpcap adapter
+ */
+bool
+airpcap_if_set_decryption_state(PAirpcapHandle ah, AirpcapDecryptionState Enable)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapSetDecryptionState(ah,Enable);
+}
+
+/*
+ * Airpcap wrapper, used to get the decryption enabling of an airpcap driver
+ */
+bool
+airpcap_if_get_driver_decryption_state(PAirpcapHandle ah, PAirpcapDecryptionState PEnable)
+{
+ if (!AirpcapLoaded || (g_PAirpcapGetDriverDecryptionState==NULL)) return false;
+ return g_PAirpcapGetDriverDecryptionState(ah,PEnable);
+}
+
+/*
+ * Airpcap wrapper, used to set the decryption enabling of an airpcap driver
+ */
+bool
+airpcap_if_set_driver_decryption_state(PAirpcapHandle ah, AirpcapDecryptionState Enable)
+{
+ if (!AirpcapLoaded || (g_PAirpcapSetDriverDecryptionState==NULL)) return false;
+ return g_PAirpcapSetDriverDecryptionState(ah,Enable);
+}
+
+/*
+ * Airpcap wrapper, used to get the fcs validation of an airpcap adapter
+ */
+bool
+airpcap_if_get_fcs_validation(PAirpcapHandle ah, PAirpcapValidationType val)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapGetFcsValidation(ah,val);
+}
+
+/*
+ * Airpcap wrapper, used to set the fcs validation of an airpcap adapter
+ */
+bool
+airpcap_if_set_fcs_validation(PAirpcapHandle ah, AirpcapValidationType val)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapSetFcsValidation(ah,val);
+}
+
+/*
+ * Airpcap wrapper, used to save the settings for the selected_if
+ */
+bool
+airpcap_if_set_device_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapSetDeviceKeys(AdapterHandle,KeysCollection);
+}
+
+/*
+ * Airpcap wrapper, used to save the settings for the selected_if
+ */
+bool
+airpcap_if_get_device_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize)
+{
+ if (!AirpcapLoaded) return false;
+ return g_PAirpcapGetDeviceKeys(AdapterHandle,KeysCollection,PKeysCollectionSize);
+}
+
+/*
+ * Airpcap wrapper, used to save the driver's set of keys
+ */
+bool
+airpcap_if_set_driver_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection)
+{
+ if (!AirpcapLoaded || (g_PAirpcapSetDriverKeys==NULL)) return false;
+ return g_PAirpcapSetDriverKeys(AdapterHandle,KeysCollection);
+}
+
+/*
+ * Airpcap wrapper, used to load the driver's set of keys
+ */
+bool
+airpcap_if_get_driver_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize)
+{
+ if (!AirpcapLoaded || (g_PAirpcapGetDriverKeys==NULL)) return false;
+ return g_PAirpcapGetDriverKeys(AdapterHandle,KeysCollection,PKeysCollectionSize);
+}
+
+/*
+ * This function will create a new airpcap_if_info_t using a name and a description
+ */
+airpcap_if_info_t *
+airpcap_if_info_new(char *name, char *description)
+{
+ PAirpcapHandle ad;
+ char ebuf[AIRPCAP_ERRBUF_SIZE];
+
+ airpcap_if_info_t *if_info = NULL;
+
+ /* Probably I have to switch on the leds!!! */
+ ad = airpcap_if_open(name, ebuf);
+ if (ad)
+ {
+ if_info = g_new0(airpcap_if_info_t, 1);
+ if_info->name = g_strdup(name);
+ if (description == NULL){
+ if_info->description = NULL;
+ }else{
+ if_info->description = g_strdup(description);
+ }
+
+ if_info->ip_addr = NULL;
+ if_info->loopback = false;
+ airpcap_if_get_fcs_validation(ad,&(if_info->CrcValidationOn));
+ airpcap_if_get_fcs_presence(ad,&(if_info->IsFcsPresent));
+ airpcap_if_get_link_type(ad,&(if_info->linkType));
+ airpcap_if_get_device_channel_ex(ad,&(if_info->channelInfo));
+ if_info->pSupportedChannels = airpcap_if_get_device_supported_channels_array(ad, &(if_info->numSupportedChannels));
+ airpcap_if_turn_led_on(ad, 0);
+ airpcap_if_get_decryption_state(ad, &(if_info->DecryptionOn));
+ if_info->led = true;
+ if_info->blinking = false;
+ if_info->saved = true; /* NO NEED TO BE SAVED */
+
+ /* get the keys, if everything is ok, close the adapter */
+ if (airpcap_if_load_keys(ad,if_info))
+ {
+ airpcap_if_close(ad);
+ }
+ }
+ return if_info;
+}
+
+/*
+ * This function will create a new fake drivers' interface, to load global keys...
+ */
+airpcap_if_info_t*
+airpcap_driver_fake_if_info_new(void)
+{
+ PAirpcapHandle ad;
+ char ebuf[AIRPCAP_ERRBUF_SIZE];
+
+ airpcap_if_info_t *if_info = NULL;
+ airpcap_if_info_t *fake_if_info = NULL;
+
+ /* Maybe for some reason no airpcap adapter is found */
+ if (g_airpcap_if_list == NULL)
+ return NULL;
+
+ /*
+ * Retrieve the first AirPcap adapter available. If no interface is found,
+ * it is not possible to retrieve the driver's settings, so return NULL.
+ */
+ if_info = (airpcap_if_info_t *)g_list_nth_data(g_airpcap_if_list,0);
+ if (if_info == NULL)
+ return NULL;
+
+ /* Open the 'fake' adapter */
+ ad = airpcap_if_open(if_info->name, ebuf);
+ if (ad)
+ {
+ fake_if_info = g_new0(airpcap_if_info_t, 1);
+ fake_if_info->name = g_strdup(if_info->name);
+ fake_if_info->description = g_strdup(if_info->description);
+ fake_if_info->loopback = false;
+ fake_if_info->ip_addr = NULL;
+ airpcap_if_get_driver_decryption_state(ad, &(fake_if_info->DecryptionOn));
+ airpcap_if_get_fcs_validation(ad,&(fake_if_info->CrcValidationOn));
+ airpcap_if_get_fcs_presence(ad,&(fake_if_info->IsFcsPresent));
+ airpcap_if_get_link_type(ad,&(fake_if_info->linkType));
+ airpcap_if_get_device_channel_ex(ad,&(fake_if_info->channelInfo));
+ airpcap_if_turn_led_on(ad, 0);
+ fake_if_info->led = true;
+ fake_if_info->blinking = false;
+ fake_if_info->saved = true; /* NO NEED TO BE SAVED */
+
+ /* get the keys, if everything is ok, close the adapter */
+ if (airpcap_if_load_driver_keys(ad,fake_if_info))
+ {
+ airpcap_if_close(ad);
+ }
+ }
+
+ return fake_if_info;
+}
+
+#ifdef AIRPCAP_DEBUG
+/*
+ * USED FOR DEBUG ONLY... PRINTS AN AirPcap ADAPTER STRUCTURE in a fancy way.
+ */
+void
+airpcap_if_info_print(airpcap_if_info_t* if_info)
+{
+ unsigned i;
+ if (if_info == NULL)
+ {
+ g_print("\nWARNING : AirPcap Interface pointer is NULL.\n");
+ return;
+ }
+
+ g_print("\n----------------- AirPcap Interface \n");
+ g_print(" NAME: %s\n",if_info->name);
+ g_print(" DESCRIPTION: %s\n",if_info->description);
+ g_print(" BLINKING: %s\n",if_info->blinking ? "TRUE" : "FALSE");
+ g_print(" channelInfo.Frequency: %u\n",if_info->channelInfo.Frequency);
+ g_print(" channelInfo.ExtChannel: %d\n",if_info->channelInfo.ExtChannel);
+ g_print(" CRCVALIDATION: %s\n",if_info->CrcValidationOn ? "ON" : "OFF");
+ g_print(" DECRYPTION: %s\n",if_info->DecryptionOn ? "ON" : "OFF");
+ g_print(" IP ADDR: %s\n",if_info->ip_addr!=NULL ? "NOT NULL" : "NULL");
+ g_print(" FCSPRESENT: %s\n",if_info->IsFcsPresent ? "TRUE" : "FALSE");
+ g_print(" KEYSCOLLECTION: %s\n",if_info->keysCollection!=NULL ? "NOT NULL" : "NULL");
+ g_print(" KEYSCOLLECTIONSIZE: %u\n",if_info->keysCollectionSize);
+ g_print(" LED: %s\n",if_info->led ? "ON" : "OFF");
+ g_print(" LINKTYPE: %d\n",if_info->linkType);
+ g_print(" LOOPBACK: %s\n",if_info->loopback ? "YES" : "NO");
+ g_print(" (GTK) TAG: %d\n",if_info->tag);
+ g_print("SUPPORTED CHANNELS POINTER: %p\n",if_info->pSupportedChannels);
+ g_print(" NUM SUPPORTED CHANNELS: %u\n",if_info->numSupportedChannels);
+
+ for(i=0; i<(if_info->numSupportedChannels); i++){
+ g_print("\n SUPPORTED CHANNEL #%u\n",i+1);
+ g_print(" CHANNEL: %u\n",if_info->pSupportedChannels[i].Channel);
+ g_print(" FREQUENCY: %u\n",if_info->pSupportedChannels[i].Frequency);
+ g_print(" FLAGS: %u\n",if_info->pSupportedChannels[i].Flags);
+ }
+ g_print("\n\n");
+}
+#endif /* AIRPCAP_DEBUG */
+
+/*
+ * Function used to load the WEP keys for a selected interface
+ */
+bool
+airpcap_if_load_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info)
+{
+ if (!if_info) return false;
+
+ if_info->keysCollectionSize = 0;
+ if_info->keysCollection = NULL;
+
+ if (!airpcap_if_get_device_keys(ad, NULL, &(if_info->keysCollectionSize)))
+ {
+ if (if_info->keysCollectionSize == 0)
+ {
+ if_info->keysCollection = NULL;
+ airpcap_if_close(ad);
+ return false;
+ }
+
+ if_info->keysCollection = (PAirpcapKeysCollection)g_malloc(if_info->keysCollectionSize);
+ if (!if_info->keysCollection)
+ {
+ if_info->keysCollectionSize = 0;
+ if_info->keysCollection = NULL;
+ airpcap_if_close(ad);
+ return false;
+ }
+
+ airpcap_if_get_device_keys(ad, if_info->keysCollection, &(if_info->keysCollectionSize));
+ return true;
+ }
+
+ airpcap_if_close(ad);
+ return false;
+}
+
+/*
+ * Function used to load the WEP keys for a selected interface
+ */
+bool
+airpcap_if_load_driver_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info)
+{
+ if_info->keysCollectionSize = 0;
+ if_info->keysCollection = NULL;
+
+ if (!airpcap_if_get_driver_keys(ad, NULL, &(if_info->keysCollectionSize)))
+ {
+ if (if_info->keysCollectionSize == 0)
+ {
+ if_info->keysCollection = NULL;
+ airpcap_if_close(ad);
+ return false;
+ }
+
+ if_info->keysCollection = (PAirpcapKeysCollection)g_malloc(if_info->keysCollectionSize);
+ if (!if_info->keysCollection)
+ {
+ if_info->keysCollectionSize = 0;
+ if_info->keysCollection = NULL;
+ airpcap_if_close(ad);
+ return false;
+ }
+
+ airpcap_if_get_driver_keys(ad, if_info->keysCollection, &(if_info->keysCollectionSize));
+ return true;
+ }
+
+ airpcap_if_close(ad);
+ return false;
+}
+
+/*
+ * Function used to save the WEP keys for a selected interface
+ */
+void
+airpcap_if_save_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info)
+{
+ if (!if_info || !AirpcapLoaded) return;
+
+ if (if_info->keysCollection != NULL)
+ g_PAirpcapSetDeviceKeys(ad,if_info->keysCollection);
+}
+
+/*
+ * Function used to save the WEP keys for a selected interface
+ */
+void
+airpcap_if_save_driver_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info)
+{
+ if (if_info->keysCollection != NULL)
+ airpcap_if_set_driver_keys(ad,if_info->keysCollection);
+}
+
+/*
+ * Callback used to free an instance of airpcap_if_info_t
+ */
+static void
+free_airpcap_if_cb(void * data, void * user_data _U_)
+{
+ airpcap_if_info_t *if_info = (airpcap_if_info_t *)data;
+
+ if (NULL == if_info)
+ return;
+
+ g_free(if_info->name);
+
+ g_free(if_info->description);
+
+ /* XXX - FREE THE WEP KEY LIST HERE!!!*/
+ if (if_info->keysCollection != NULL)
+ {
+ g_free(if_info->keysCollection);
+ if_info->keysCollection = NULL;
+ }
+
+ g_slist_free(if_info->ip_addr);
+
+ g_free(if_info);
+}
+
+/*
+ * Function used to free the airpcap interface list
+ */
+void
+free_airpcap_interface_list(GList *if_list)
+{
+ g_list_foreach(if_list, free_airpcap_if_cb, NULL);
+ g_list_free(if_list);
+}
+
+/*
+ * This function will use the airpcap.dll to find all the airpcap devices.
+ * Will return null if no device is found.
+ */
+GList*
+get_airpcap_interface_list(int *err, char **err_str)
+{
+ GList *il = NULL;
+ airpcap_if_info_t *if_info;
+ int n_adapts;
+ AirpcapDeviceDescription *devsList, *adListEntry;
+ char errbuf[AIRPCAP_ERRBUF_SIZE];
+
+ *err = 0;
+
+ if (!AirpcapLoaded)
+ {
+ *err = AIRPCAP_NOT_LOADED;
+ return il;
+ }
+
+ if (!g_PAirpcapGetDeviceList(&devsList, errbuf))
+ {
+ /* No interfaces, return il = NULL; */
+ *err = CANT_GET_AIRPCAP_INTERFACE_LIST;
+ if (err_str != NULL)
+ *err_str = cant_get_airpcap_if_list_error_message(errbuf);
+ return il;
+ }
+
+ /*
+ * Count the adapters
+ */
+ adListEntry = devsList;
+ n_adapts = 0;
+ while(adListEntry)
+ {
+ n_adapts++;
+ adListEntry = adListEntry->next;
+ }
+
+ if (n_adapts == 0)
+ {
+ /* No interfaces, return il= NULL */
+ g_PAirpcapFreeDeviceList(devsList);
+ *err = NO_AIRPCAP_INTERFACES_FOUND;
+ if (err_str != NULL)
+ *err_str = NULL;
+ return il;
+ }
+
+ /*
+ * Insert the adapters in our list
+ */
+ adListEntry = devsList;
+ while(adListEntry)
+ {
+ if_info = airpcap_if_info_new(adListEntry->Name, adListEntry->Description);
+ if (if_info != NULL){
+ il = g_list_append(il, if_info);
+ }
+
+ adListEntry = adListEntry->next;
+ }
+
+ g_PAirpcapFreeDeviceList(devsList);
+
+ return il;
+}
+
+/*
+ * Used to retrieve the interface given the name
+ * (the name is used in AirpcapOpen)
+ */
+airpcap_if_info_t* get_airpcap_if_from_name(GList* if_list, const char* name)
+{
+ GList* curr;
+ airpcap_if_info_t* if_info;
+
+ for (curr = g_list_first(if_list); curr; curr = g_list_next(curr)) {
+ if_info = (airpcap_if_info_t *)curr->data;
+ if (if_info && (g_ascii_strcasecmp(if_info->name, name) == 0)) {
+ return (if_info);
+ }
+ /* Try the name without the "\\.\" prefix. */
+ if (strlen(if_info->name) > 4 && (g_ascii_strcasecmp(if_info->name + 4, name) == 0)) {
+ return (if_info);
+ }
+ }
+ return (NULL);
+}
+
+/*
+ * Clear keys and decryption status for the specified interface
+ */
+void
+airpcap_if_clear_decryption_settings(airpcap_if_info_t* info_if)
+{
+ if (info_if != NULL)
+ {
+ if (info_if->keysCollection != NULL)
+ {
+ g_free(info_if->keysCollection);
+ info_if->keysCollection = NULL;
+ }
+
+ info_if->keysCollectionSize = 0;
+
+ info_if->DecryptionOn = AIRPCAP_DECRYPTION_OFF;
+ info_if->saved = false;
+ }
+}
+
+/*
+ * Used to retrieve the two chars string from interface
+ */
+char*
+airpcap_get_if_string_number(airpcap_if_info_t* if_info)
+{
+ char* number;
+ unsigned n;
+ int a;
+
+ a = sscanf(if_info->name,AIRPCAP_DEVICE_NUMBER_EXTRACT_STRING,&n);
+
+ /* If sscanf() returned 1, it means that has read a number, so interface is not "Any"
+ * Otherwise, check if it is the "Any" adapter...
+ */
+ if (a == 0)
+ {
+ if (g_ascii_strcasecmp(if_info->name,AIRPCAP_DEVICE_ANY_EXTRACT_STRING)!=0)
+ number = g_strdup("??");
+ else
+ number = g_strdup(AIRPCAP_CHANNEL_ANY_NAME);
+ }
+ else
+ {
+ number = ws_strdup_printf("%.2u",n);
+ }
+
+ return number;
+}
+
+/*
+ * Used to retrieve the two chars string from interface
+ */
+char*
+airpcap_get_if_string_number_from_description(char* description)
+{
+ char* number;
+ char* pointer;
+
+ number = g_new(char, 3);
+
+ pointer = g_strrstr(description,"#\0");
+
+ number[0] = *(pointer+1);
+ number[1] = *(pointer+2);
+ number[2] = '\0';
+
+ return number;
+}
+
+/*
+ * Load the configuration for the specified interface
+ */
+void
+airpcap_load_selected_if_configuration(airpcap_if_info_t* if_info)
+{
+ char ebuf[AIRPCAP_ERRBUF_SIZE];
+ PAirpcapHandle ad;
+
+ if (if_info != NULL)
+ {
+ ad = airpcap_if_open(if_info->name, ebuf);
+
+ if (ad)
+ {
+ /* Stop blinking (if it was blinking!)*/
+ if (if_info->blinking)
+ {
+ /* Turn on the light (if it was off) */
+ if (!(if_info->led)) airpcap_if_turn_led_on(ad, 0);
+ }
+
+ /* Apply settings... */
+ airpcap_if_get_device_channel_ex(ad,&(if_info->channelInfo));
+ airpcap_if_get_fcs_validation(ad,&(if_info->CrcValidationOn));
+ airpcap_if_get_fcs_presence(ad,&(if_info->IsFcsPresent));
+ airpcap_if_get_link_type(ad,&(if_info->linkType));
+ airpcap_if_get_decryption_state(ad, &(if_info->DecryptionOn));
+ /* get the keys, if everything is ok, close the adapter */
+ if (airpcap_if_load_keys(ad,if_info))
+ airpcap_if_close(ad);
+
+ if_info->saved = true;
+ }
+#if 0
+ else
+ {
+ simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, " Error in opening adapter for %s",if_info->description);
+ }
+#endif
+ }
+}
+
+/*
+ * Save the configuration for the specified interface
+ */
+void
+airpcap_save_selected_if_configuration(airpcap_if_info_t* if_info)
+{
+ char ebuf[AIRPCAP_ERRBUF_SIZE];
+ PAirpcapHandle ad;
+
+ if (if_info != NULL)
+ {
+ ad = airpcap_if_open(if_info->name, ebuf);
+
+ if (ad)
+ {
+ /* Stop blinking (if it was blinking!)*/
+ if (if_info->blinking)
+ {
+ /* Turn on the light (if it was off) */
+ if (!(if_info->led)) airpcap_if_turn_led_on(ad, 0);
+ }
+
+ /* Apply settings... */
+ airpcap_if_set_device_channel_ex(ad,if_info->channelInfo);
+ airpcap_if_set_fcs_validation(ad,if_info->CrcValidationOn);
+ airpcap_if_set_fcs_presence(ad,if_info->IsFcsPresent);
+ airpcap_if_set_link_type(ad,if_info->linkType);
+ airpcap_if_set_decryption_state(ad, if_info->DecryptionOn);
+ airpcap_if_save_keys(ad,if_info);
+
+ /* ... and save them */
+ if (!airpcap_if_store_cur_config_as_adapter_default(ad))
+ {
+#if 0
+ simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "Cannot save Wireless configuration!!!\nRemember that in order to store the configuration in the registry you have to:\n\n- Close all the airpcap-based applications.\n- Be sure to have administrative privileges.");
+#endif
+ if_info->saved = false;
+ airpcap_if_close(ad);
+ return;
+ }
+
+ if_info->saved = true;
+ airpcap_if_close(ad);
+ }
+#if 0
+ else
+ {
+ simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, " Error in opening adapter for %s",if_info->description);
+ }
+#endif
+ }
+}
+
+/*
+ * Save the configuration for the specified interface
+ */
+void
+airpcap_save_driver_if_configuration(airpcap_if_info_t* fake_if_info)
+{
+ char ebuf[AIRPCAP_ERRBUF_SIZE];
+ PAirpcapHandle ad;
+
+ if (fake_if_info != NULL)
+ {
+ ad = airpcap_if_open(fake_if_info->name, ebuf);
+
+ if (ad)
+ {
+ /* Apply decryption settings... */
+ airpcap_if_set_driver_decryption_state(ad, fake_if_info->DecryptionOn);
+ airpcap_if_save_driver_keys(ad,fake_if_info);
+ airpcap_if_close(ad);
+ }
+#if 0
+ else
+ {
+ simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, " Error in opening adapter for %s",fake_if_info->description);
+ }
+#endif
+ }
+
+ return;
+}
+
+/*
+ * Free an instance of airpcap_if_info_t
+ */
+void
+airpcap_if_info_free(airpcap_if_info_t *if_info)
+{
+ if (if_info != NULL)
+ {
+ g_free(if_info->name);
+
+ g_free(if_info->description);
+
+ if (if_info->keysCollection != NULL)
+ {
+ g_free(if_info->keysCollection);
+ if_info->keysCollection = NULL;
+ }
+
+ if (if_info->ip_addr != NULL)
+ {
+ g_slist_free(if_info->ip_addr);
+ if_info->ip_addr = NULL;
+ }
+
+ g_free(if_info);
+ }
+}
+
+
+/* DYNAMIC LIBRARY LOADER */
+/*
+ * Used to dynamically load the airpcap library in order link it only when
+ * it's present on the system
+ */
+int load_airpcap(void)
+{
+#ifdef _WIN32
+ bool base_functions = true;
+ bool eleven_n_functions = true;
+
+ if ((AirpcapLib = ws_load_library("airpcap.dll")) == NULL)
+ {
+ /* Report the error but go on */
+ AirpcapVersion = AIRPCAP_DLL_NOT_FOUND;
+ return AirpcapVersion;
+ }
+ else
+ {
+ if ((g_PAirpcapGetLastError = (AirpcapGetLastErrorHandler) GetProcAddress(AirpcapLib, "AirpcapGetLastError")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetDeviceList = (AirpcapGetDeviceListHandler) GetProcAddress(AirpcapLib, "AirpcapGetDeviceList")) == NULL) base_functions = false;
+ if ((g_PAirpcapFreeDeviceList = (AirpcapFreeDeviceListHandler) GetProcAddress(AirpcapLib, "AirpcapFreeDeviceList")) == NULL) base_functions = false;
+ if ((g_PAirpcapOpen = (AirpcapOpenHandler) GetProcAddress(AirpcapLib, "AirpcapOpen")) == NULL) base_functions = false;
+ if ((g_PAirpcapClose = (AirpcapCloseHandler) GetProcAddress(AirpcapLib, "AirpcapClose")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetLinkType = (AirpcapGetLinkTypeHandler) GetProcAddress(AirpcapLib, "AirpcapGetLinkType")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetLinkType = (AirpcapSetLinkTypeHandler) GetProcAddress(AirpcapLib, "AirpcapSetLinkType")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetKernelBuffer = (AirpcapSetKernelBufferHandler) GetProcAddress(AirpcapLib, "AirpcapSetKernelBuffer")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetFilter = (AirpcapSetFilterHandler) GetProcAddress(AirpcapLib, "AirpcapSetFilter")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetMacAddress = (AirpcapGetMacAddressHandler) GetProcAddress(AirpcapLib, "AirpcapGetMacAddress")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetMinToCopy = (AirpcapSetMinToCopyHandler) GetProcAddress(AirpcapLib, "AirpcapSetMinToCopy")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetReadEvent = (AirpcapGetReadEventHandler) GetProcAddress(AirpcapLib, "AirpcapGetReadEvent")) == NULL) base_functions = false;
+ if ((g_PAirpcapRead = (AirpcapReadHandler) GetProcAddress(AirpcapLib, "AirpcapRead")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetStats = (AirpcapGetStatsHandler) GetProcAddress(AirpcapLib, "AirpcapGetStats")) == NULL) base_functions = false;
+ if ((g_PAirpcapTurnLedOn = (AirpcapTurnLedOnHandler) GetProcAddress(AirpcapLib, "AirpcapTurnLedOn")) == NULL) base_functions = false;
+ if ((g_PAirpcapTurnLedOff = (AirpcapTurnLedOffHandler) GetProcAddress(AirpcapLib, "AirpcapTurnLedOff")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetDeviceChannel = (AirpcapGetDeviceChannelHandler) GetProcAddress(AirpcapLib, "AirpcapGetDeviceChannel")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetDeviceChannel = (AirpcapSetDeviceChannelHandler) GetProcAddress(AirpcapLib, "AirpcapSetDeviceChannel")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetFcsPresence = (AirpcapGetFcsPresenceHandler) GetProcAddress(AirpcapLib, "AirpcapGetFcsPresence")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetFcsPresence = (AirpcapSetFcsPresenceHandler) GetProcAddress(AirpcapLib, "AirpcapSetFcsPresence")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetFcsValidation = (AirpcapGetFcsValidationHandler) GetProcAddress(AirpcapLib, "AirpcapGetFcsValidation")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetFcsValidation = (AirpcapSetFcsValidationHandler) GetProcAddress(AirpcapLib, "AirpcapSetFcsValidation")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetDeviceKeys = (AirpcapGetDeviceKeysHandler) GetProcAddress(AirpcapLib, "AirpcapGetDeviceKeys")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetDeviceKeys = (AirpcapSetDeviceKeysHandler) GetProcAddress(AirpcapLib, "AirpcapSetDeviceKeys")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetDecryptionState = (AirpcapGetDecryptionStateHandler) GetProcAddress(AirpcapLib, "AirpcapGetDecryptionState")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetDecryptionState = (AirpcapSetDecryptionStateHandler) GetProcAddress(AirpcapLib, "AirpcapSetDecryptionState")) == NULL) base_functions = false;
+ if ((g_PAirpcapStoreCurConfigAsAdapterDefault = (AirpcapStoreCurConfigAsAdapterDefaultHandler) GetProcAddress(AirpcapLib, "AirpcapStoreCurConfigAsAdapterDefault")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetVersion = (AirpcapGetVersionHandler) GetProcAddress(AirpcapLib, "AirpcapGetVersion")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetDriverDecryptionState = (AirpcapGetDriverDecryptionStateHandler) GetProcAddress(AirpcapLib, "AirpcapGetDriverDecryptionState")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetDriverDecryptionState = (AirpcapSetDriverDecryptionStateHandler) GetProcAddress(AirpcapLib, "AirpcapSetDriverDecryptionState")) == NULL) base_functions = false;
+ if ((g_PAirpcapGetDriverKeys = (AirpcapGetDriverKeysHandler) GetProcAddress(AirpcapLib, "AirpcapGetDriverKeys")) == NULL) base_functions = false;
+ if ((g_PAirpcapSetDriverKeys = (AirpcapSetDriverKeysHandler) GetProcAddress(AirpcapLib, "AirpcapSetDriverKeys")) == NULL) base_functions = false;
+
+ /* TEST IF AIRPCAP SUPPORTS 11N */
+ if ((g_PAirpcapSetDeviceChannelEx = (AirpcapSetDeviceChannelExHandler) GetProcAddress(AirpcapLib, "AirpcapSetDeviceChannelEx")) == NULL) eleven_n_functions = false;
+ if ((g_PAirpcapGetDeviceChannelEx = (AirpcapGetDeviceChannelExHandler) GetProcAddress(AirpcapLib, "AirpcapGetDeviceChannelEx")) == NULL) eleven_n_functions = false;
+ if ((g_PAirpcapGetDeviceSupportedChannels = (AirpcapGetDeviceSupportedChannelsHandler) GetProcAddress(AirpcapLib, "AirpcapGetDeviceSupportedChannels")) == NULL) eleven_n_functions = false;
+
+ if (base_functions && eleven_n_functions){
+ AirpcapLoaded = true;
+ AirpcapVersion = AIRPCAP_DLL_OK;
+ } else if (base_functions){
+ AirpcapLoaded = true;
+ AirpcapVersion = AIRPCAP_DLL_OLD;
+ return AIRPCAP_DLL_OK;
+ }else{
+ AirpcapLoaded = false;
+ AirpcapVersion = AIRPCAP_DLL_ERROR;
+ }
+ }
+ return AirpcapVersion;
+#else /* _WIN32 */
+ return AIRPCAP_DLL_NOT_FOUND;
+#endif /* _WIN32 */
+}
+
+/*
+ * Append the version of AirPcap with which we were compiled to a GString.
+ */
+void
+gather_airpcap_compile_info(feature_list l)
+{
+ with_feature(l, "AirPcap");
+}
+
+/*
+ * Append the version of AirPcap with which we're running to a GString.
+ */
+void
+gather_airpcap_runtime_info(feature_list l)
+{
+ unsigned vmaj, vmin, vrev, build;
+
+ /* See if the DLL has been loaded successfully. Bail if it hasn't */
+ if (AirpcapLoaded == false) {
+ without_feature(l, "AirPcap");
+ return;
+ }
+
+ g_PAirpcapGetVersion(&vmaj, &vmin, &vrev, &build);
+ with_feature(l, "AirPcap %d.%d.%d build %d", vmaj, vmin, vrev, build);
+}
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 4
+ * tab-width: 8
+ * indent-tabs-mode: nil
+ * End:
+ *
+ * vi: set shiftwidth=4 tabstop=8 expandtab:
+ * :indentSize=4:tabSize=8:noTabs=true:
+ */
diff --git a/capture/airpcap_loader.h b/capture/airpcap_loader.h
new file mode 100644
index 00000000..9ca9fa45
--- /dev/null
+++ b/capture/airpcap_loader.h
@@ -0,0 +1,409 @@
+/** @file
+ *
+ * Declarations of routines for the "About" dialog
+ *
+ * Giorgio Tino <giorgio.tino@cacetech.com>
+ * Copyright (c) CACE Technologies, LLC 2006
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef __AIRPCAP_LOADER_H__
+#define __AIRPCAP_LOADER_H__
+
+#include <epan/crypt/dot11decrypt_system.h>
+#include <wsutil/feature_list.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/* Error values from "get_airpcap_interface_list()". */
+#define CANT_GET_AIRPCAP_INTERFACE_LIST 0 /* error getting list */
+#define NO_AIRPCAP_INTERFACES_FOUND 1 /* list is empty */
+#define AIRPCAP_NOT_LOADED 2 /* Airpcap DLL not loaded */
+
+#define AIRPCAP_CHANNEL_ANY_NAME "ANY"
+
+#define AIRPCAP_WEP_KEY_STRING "WEP"
+/*
+ * XXX - WPA_PWD is the passphrase+ssid and WPA-PSK is the hexadecimal key
+ */
+#define AIRPCAP_WPA_PWD_KEY_STRING "WPA-PWD"
+#define AIRPCAP_WPA_BIN_KEY_STRING "WPA-PSK"
+
+#define AIRPCAP_DLL_OK 0
+#define AIRPCAP_DLL_OLD 1
+#define AIRPCAP_DLL_ERROR 2
+#define AIRPCAP_DLL_NOT_FOUND 3
+
+/* #define AIRPCAP_DEBUG 1 */
+
+typedef char * (*AirpcapGetLastErrorHandler)(PAirpcapHandle AdapterHandle);
+typedef bool (*AirpcapGetDeviceListHandler)(PAirpcapDeviceDescription *PPAllDevs, char * Ebuf);
+typedef void (*AirpcapFreeDeviceListHandler)(PAirpcapDeviceDescription PAllDevs);
+typedef PAirpcapHandle (*AirpcapOpenHandler)(char * DeviceName, char * Ebuf);
+typedef void (*AirpcapCloseHandler)(PAirpcapHandle AdapterHandle);
+typedef bool (*AirpcapGetLinkTypeHandler)(PAirpcapHandle AdapterHandle, PAirpcapLinkType PLinkType);
+typedef bool (*AirpcapSetLinkTypeHandler)(PAirpcapHandle AdapterHandle, AirpcapLinkType NewLinkType);
+typedef bool (*AirpcapSetKernelBufferHandler)(PAirpcapHandle AdapterHandle, unsigned BufferSize);
+typedef bool (*AirpcapSetFilterHandler)(PAirpcapHandle AdapterHandle, void * Instructions, unsigned Len);
+typedef bool (*AirpcapGetMacAddressHandler)(PAirpcapHandle AdapterHandle, PAirpcapMacAddress PMacAddress);
+typedef bool (*AirpcapSetMinToCopyHandler)(PAirpcapHandle AdapterHandle, unsigned MinToCopy);
+typedef bool (*AirpcapGetReadEventHandler)(PAirpcapHandle AdapterHandle, void *** PReadEvent);
+typedef bool (*AirpcapReadHandler)(PAirpcapHandle AdapterHandle, uint8_t * Buffer, unsigned BufSize, unsigned * PReceievedBytes);
+typedef bool (*AirpcapGetStatsHandler)(PAirpcapHandle AdapterHandle, PAirpcapStats PStats);
+typedef bool (*AirpcapTurnLedOnHandler)(PAirpcapHandle AdapterHandle, unsigned LedNumber);
+typedef bool (*AirpcapTurnLedOffHandler)(PAirpcapHandle AdapterHandle, unsigned LedNumber);
+typedef bool (*AirpcapSetDeviceChannelHandler)(PAirpcapHandle AdapterHandle, unsigned Channel);
+typedef bool (*AirpcapGetDeviceChannelHandler)(PAirpcapHandle AdapterHandle, unsigned * PChannel);
+typedef bool (*AirpcapSetFcsPresenceHandler)(PAirpcapHandle AdapterHandle, bool IsFcsPresent);
+typedef bool (*AirpcapGetFcsPresenceHandler)(PAirpcapHandle AdapterHandle, bool * PIsFcsPresent);
+typedef bool (*AirpcapSetFcsValidationHandler)(PAirpcapHandle AdapterHandle, AirpcapValidationType ValidationType);
+typedef bool (*AirpcapGetFcsValidationHandler)(PAirpcapHandle AdapterHandle, PAirpcapValidationType PValidationType);
+typedef bool (*AirpcapSetDeviceKeysHandler)(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection);
+typedef bool (*AirpcapGetDeviceKeysHandler)(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize);
+typedef bool (*AirpcapSetDriverKeysHandler)(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection);
+typedef bool (*AirpcapGetDriverKeysHandler)(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize);
+typedef bool (*AirpcapSetDecryptionStateHandler)(PAirpcapHandle AdapterHandle, AirpcapDecryptionState Enable);
+typedef bool (*AirpcapGetDecryptionStateHandler)(PAirpcapHandle AdapterHandle, PAirpcapDecryptionState PEnable);
+typedef bool (*AirpcapSetDriverDecryptionStateHandler)(PAirpcapHandle AdapterHandle, AirpcapDecryptionState Enable);
+typedef bool (*AirpcapGetDriverDecryptionStateHandler)(PAirpcapHandle AdapterHandle, PAirpcapDecryptionState PEnable);
+typedef bool (*AirpcapStoreCurConfigAsAdapterDefaultHandler)(PAirpcapHandle AdapterHandle);
+typedef void (*AirpcapGetVersionHandler)(unsigned * VersionMajor, unsigned * VersionMinor, unsigned * VersionRev, unsigned * VersionBuild);
+typedef bool (*AirpcapSetDeviceChannelExHandler)(PAirpcapHandle AdapterHandle, AirpcapChannelInfo ChannelInfo);
+typedef bool (*AirpcapGetDeviceChannelExHandler)(PAirpcapHandle AdapterHandle, PAirpcapChannelInfo PChannelInfo);
+typedef bool (*AirpcapGetDeviceSupportedChannelsHandler)(PAirpcapHandle AdapterHandle, AirpcapChannelInfo **ppChannelInfo, uint32_t * pNumChannelInfo);
+
+#define FLAG_CAN_BE_LOW 0x00000001
+#define FLAG_CAN_BE_HIGH 0x00000002
+#define FLAG_IS_BG_CHANNEL 0x00000004
+#define FLAG_IS_A_CHANNEL 0x00000008
+
+typedef struct _Dot11Channel
+{
+ unsigned Channel;
+ uint32_t Frequency;
+ uint32_t Flags;
+} Dot11Channel;
+
+/*
+ * The list of interfaces returned by "get_airpcap_interface_list()" is
+ * a list of these structures.
+ */
+typedef struct {
+ char *name; /* e.g. "eth0" */
+ char *description; /* from OS, e.g. "Local Area Connection" or NULL */
+ GSList *ip_addr; /* containing address values of if_addr_t */
+ bool loopback; /* true if loopback, false otherwise */
+ AirpcapLinkType linkType; /* The link layer type */
+ AirpcapChannelInfo channelInfo; /* Channel Information */
+ bool IsFcsPresent; /* Include 802.11 CRC in frames */
+ AirpcapValidationType CrcValidationOn; /* Capture Frames with Wrong CRC */
+ AirpcapDecryptionState DecryptionOn; /* true if decryption is on, false otherwise */
+ PAirpcapKeysCollection keysCollection; /* WEP Key collection for the adapter */
+ unsigned keysCollectionSize; /* Size of the key collection */
+ bool blinking; /* true if is blinkng, false otherwise */
+ bool led; /* true if on, false if off */
+ bool saved; /* true if current configuration has been saved, false otherwise */
+ int tag; /* int for the gtk blinking callback */
+ Dot11Channel *pSupportedChannels;
+ uint32_t numSupportedChannels;
+} airpcap_if_info_t;
+
+/*
+ * Struct used to store infos to pass to the preferences manager callbacks
+ */
+typedef struct {
+ GList *list;
+ int current_index;
+ int number_of_keys;
+} keys_cb_data_t;
+
+/* Airpcap interface list */
+extern GList *g_airpcap_if_list;
+
+/* Airpcap current selected interface */
+extern airpcap_if_info_t *airpcap_if_selected;
+
+/* Airpcap current active interface */
+extern airpcap_if_info_t *airpcap_if_active;
+
+#ifdef AIRPCAP_DEBUG
+/*
+ * USED FOR DEBUG ONLY... PRINTS AN AirPcap ADAPTER STRUCTURE in a fancy way.
+ */
+void
+airpcap_if_info_print(airpcap_if_info_t* if_info);
+#endif
+
+/*
+ * Used to retrieve the two chars string from interface
+ */
+char*
+airpcap_get_if_string_number_from_description(char* description);
+
+/*
+ * Function used to free the airpcap interface list
+ */
+void
+free_airpcap_interface_list(GList *if_list);
+
+/*
+ * Used to retrieve the interface given the name
+ * (the name is used in AirpcapOpen).
+ */
+airpcap_if_info_t* get_airpcap_if_from_name(GList* if_list, const char* name);
+
+/*
+ * Airpcap wrapper, used to store the current settings for the selected adapter
+ */
+bool
+airpcap_if_store_cur_config_as_adapter_default(PAirpcapHandle ah);
+
+/*
+ * Function used to load the WEP keys for a selected interface
+ */
+bool
+airpcap_if_load_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info);
+
+/*
+ * Function used to load the WEP keys from the global driver list
+ */
+bool
+airpcap_if_load_driver_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info);
+
+/*
+ * Function used to save the WEP keys for a selected interface
+ */
+void
+airpcap_if_save_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info);
+
+/*
+ * Function used to save the WEP keys for a selected interface
+ */
+void
+airpcap_if_save_driver_keys(PAirpcapHandle ad, airpcap_if_info_t *if_info);
+
+/*
+ * Airpcap wrapper, used to get the fcs validation of an airpcap adapter
+ */
+bool
+airpcap_if_get_fcs_validation(PAirpcapHandle ah, PAirpcapValidationType val);
+
+/*
+ * Airpcap wrapper, used to set the fcs validation of an airpcap adapter
+ */
+bool
+airpcap_if_set_fcs_validation(PAirpcapHandle ah, AirpcapValidationType val);
+
+/* Many of these are GTK+ only. */
+/*
+ * Airpcap wrapper, used to get the decryption enabling of an airpcap adapter
+ */
+bool
+airpcap_if_get_decryption_state(PAirpcapHandle ah, PAirpcapDecryptionState val);
+
+/*
+ * Airpcap wrapper, used to set the decryption enabling of an airpcap adapter
+ */
+bool
+airpcap_if_set_decryption_state(PAirpcapHandle ah, AirpcapDecryptionState val);
+
+/*
+ * Airpcap wrapper, used to get the fcs presence of an airpcap adapter
+ */
+bool
+airpcap_if_get_fcs_presence(PAirpcapHandle ah, bool * ch);
+
+/*
+ * Airpcap wrapper, used to set the fcs presence of an airpcap adapter
+ */
+bool
+airpcap_if_set_fcs_presence(PAirpcapHandle ah, bool ch);
+
+/*
+ * Airpcap wrapper, used to get the link type of an airpcap adapter
+ */
+bool
+airpcap_if_get_link_type(PAirpcapHandle ah, PAirpcapLinkType lt);
+
+/*
+ * Airpcap wrapper, used to set the link type of an airpcap adapter
+ */
+bool
+airpcap_if_set_link_type(PAirpcapHandle ah, AirpcapLinkType lt);
+
+/*
+ * Airpcap wrapper, used to get the channel of an airpcap adapter
+ */
+bool
+airpcap_if_get_device_channel(PAirpcapHandle ah, unsigned * ch);
+
+/*
+ * Airpcap wrapper, get the channels supported by the adapter
+ */
+bool
+airpcap_if_get_device_supported_channels(PAirpcapHandle ah, AirpcapChannelInfo **cInfo, uint32_t * nInfo);
+
+/*
+ * Airpcap wrapper, get supported channels formatted into an array
+ */
+Dot11Channel*
+airpcap_if_get_device_supported_channels_array(PAirpcapHandle ah, uint32_t * pNumSupportedChannels);
+
+/*
+ * Airpcap wrapper, used to set the channel of an airpcap adapter
+ */
+bool
+airpcap_if_set_device_channel(PAirpcapHandle ah, unsigned ch);
+
+/*
+ * Airpcap wrapper, used to get the frequency of an airpcap adapter
+ */
+bool
+airpcap_if_get_device_channel_ex(PAirpcapHandle ah, PAirpcapChannelInfo pChannelInfo);
+
+/*
+ * Airpcap wrapper, used to set the frequency of an airpcap adapter
+ */
+bool
+airpcap_if_set_device_channel_ex(PAirpcapHandle ah, AirpcapChannelInfo ChannelInfo);
+
+/*
+ * Airpcap wrapper, used to open an airpcap adapter
+ */
+PAirpcapHandle airpcap_if_open(char * name, char * err);
+
+/*
+ * Airpcap wrapper, used to close an airpcap adapter
+ */
+void airpcap_if_close(PAirpcapHandle handle);
+
+/*
+ * Retrieve the state of the Airpcap DLL
+ */
+int
+airpcap_get_dll_state(void);
+
+/*
+ * Airpcap wrapper, used to turn on the led of an airpcap adapter
+ */
+bool airpcap_if_turn_led_on(PAirpcapHandle AdapterHandle, unsigned LedNumber);
+
+/*
+ * Airpcap wrapper, used to turn off the led of an airpcap adapter
+ */
+bool airpcap_if_turn_led_off(PAirpcapHandle AdapterHandle, unsigned LedNumber);
+
+/*
+ * This function will create a new airpcap_if_info_t using a name and a description
+ */
+airpcap_if_info_t* airpcap_if_info_new(char *name, char *description);
+
+/*
+ * This function will create a new fake drivers' interface, to load global keys...
+ */
+airpcap_if_info_t* airpcap_driver_fake_if_info_new(void);
+
+/*
+ * Used to dinamically load the airpcap library in order link it only when
+ * it's present on the system.
+ */
+int load_airpcap(void);
+
+/*
+ * This function will use the airpcap.dll to find all the airpcap devices.
+ * Will return null if no device is found.
+ */
+GList* get_airpcap_interface_list(int *err, char **err_str);
+
+/*
+ * Load the configuration for the specified interface
+ */
+void
+airpcap_load_selected_if_configuration(airpcap_if_info_t* if_info);
+
+/*
+ * Save the configuration for the specified interface
+ */
+void
+airpcap_save_selected_if_configuration(airpcap_if_info_t* if_info);
+
+/*
+ * Used to retrieve the two chars string from interface description
+ */
+char*
+airpcap_get_if_string_number(airpcap_if_info_t* if_info);
+
+/*
+ * Airpcap wrapper, used to save the settings for the selected_if
+ */
+bool
+airpcap_if_set_device_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection);
+
+/*
+ * Airpcap wrapper, used to save the settings for the selected_if
+ */
+bool
+airpcap_if_get_device_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize);
+
+/*
+ * Airpcap wrapper, used to save the settings for the selected_if
+ */
+bool
+airpcap_if_set_driver_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection);
+
+/*
+ * Airpcap wrapper, used to save the settings for the selected_if
+ */
+bool
+airpcap_if_get_driver_keys(PAirpcapHandle AdapterHandle, PAirpcapKeysCollection KeysCollection, unsigned * PKeysCollectionSize);
+
+/*
+ * Airpcap wrapper, used to get the decryption enabling of an airpcap driver
+ */
+bool
+airpcap_if_get_driver_decryption_state(PAirpcapHandle ah, PAirpcapDecryptionState PEnable);
+/*
+ * Airpcap wrapper, used to set the decryption enabling of an airpcap driver
+ */
+bool
+airpcap_if_set_driver_decryption_state(PAirpcapHandle ah, AirpcapDecryptionState Enable);
+
+/*
+ * Save the configuration for the specified interface
+ */
+void
+airpcap_save_driver_if_configuration(airpcap_if_info_t* fake_if_info);
+
+/*
+ * Free an instance of airpcap_if_info_t
+ */
+void
+airpcap_if_info_free(airpcap_if_info_t *if_info);
+
+/*
+ * Clear keys and decryption status for the specified interface
+ */
+void
+airpcap_if_clear_decryption_settings(airpcap_if_info_t* info_if);
+
+/*
+ * Adds compiled version string to str
+ */
+void
+gather_airpcap_compile_info(feature_list l);
+
+void
+gather_airpcap_runtime_info(feature_list l);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __AIRPCAP_LOADER_H__ */
diff --git a/capture/capture-pcap-util-int.h b/capture/capture-pcap-util-int.h
new file mode 100644
index 00000000..b148fb77
--- /dev/null
+++ b/capture/capture-pcap-util-int.h
@@ -0,0 +1,63 @@
+/** @file
+ *
+ * Definitions of routines internal to the libpcap/WinPcap/Npcap utilities
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef __PCAP_UTIL_INT_H__
+#define __PCAP_UTIL_INT_H__
+
+extern if_info_t *if_info_new(const char *name, const char *description,
+ bool loopback);
+extern void if_info_add_address(if_info_t *if_info, struct sockaddr *addr);
+#ifdef HAVE_PCAP_REMOTE
+extern GList *get_interface_list_findalldevs_ex(const char *hostname,
+ const char *port, int auth_type, const char *username, const char *passwd,
+ int *err, char **err_str);
+#endif /* HAVE_PCAP_REMOTE */
+extern GList *get_interface_list_findalldevs(int *err, char **err_str);
+
+#ifdef HAVE_PCAP_SET_TSTAMP_PRECISION
+/*
+ * Request that a pcap_t provide high-resolution (nanosecond) time
+ * stamps; if that request fails, we'll just silently continue to
+ * use the microsecond-resolution time stamps, and our caller will
+ * find out, when they call have_high_resolution_timestamp(), that
+ * we don't have high-resolution time stamps.
+ */
+extern void request_high_resolution_timestamp(pcap_t *pcap_h);
+#endif
+
+extern if_capabilities_t *get_if_capabilities_local(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str);
+extern pcap_t *open_capture_device_local(capture_options *capture_opts,
+ interface_options *interface_opts, int timeout,
+ cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE]);
+#ifdef HAVE_PCAP_CREATE
+extern if_capabilities_t *get_if_capabilities_pcap_create(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str);
+extern pcap_t *open_capture_device_pcap_create(capture_options *capture_opts,
+ interface_options *interface_opts, int timeout,
+ cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE]);
+#endif /* HAVE_PCAP_CREATE */
+extern if_capabilities_t *get_if_capabilities_pcap_open_live(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str);
+extern pcap_t *open_capture_device_pcap_open_live(interface_options *interface_opts,
+ int timeout, cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE]);
+
+/*
+ * Get an error message string for a CANT_GET_INTERFACE_LIST error from
+ * "get_interface_list()". This is used to let the error message string
+ * be platform-dependent.
+ */
+extern char *cant_get_if_list_error_message(const char *err_str);
+
+#endif /* __PCAP_UTIL_INT_H__ */
diff --git a/capture/capture-pcap-util-unix.c b/capture/capture-pcap-util-unix.c
new file mode 100644
index 00000000..8fa94306
--- /dev/null
+++ b/capture/capture-pcap-util-unix.c
@@ -0,0 +1,209 @@
+/* capture-pcap-util-unix.c
+ * UN*X-specific utility routines for packet capture
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+
+#include <wireshark.h>
+
+#include <string.h>
+#include <ws_attributes.h>
+#include <wsutil/feature_list.h>
+
+#ifdef HAVE_LIBPCAP
+
+#include <pcap.h>
+
+#ifdef HAVE_LIBCAP
+# include <sys/capability.h>
+#endif
+
+#include "capture/capture_ifinfo.h"
+#include "capture/capture-pcap-util.h"
+#include "capture/capture-pcap-util-int.h"
+
+#ifdef HAVE_PCAP_REMOTE
+GList *
+get_remote_interface_list(const char *hostname, const char *port,
+ int auth_type, const char *username,
+ const char *passwd, int *err, char **err_str)
+{
+ return get_interface_list_findalldevs_ex(hostname, port, auth_type,
+ username, passwd, err, err_str);
+}
+#endif
+
+GList *
+get_interface_list(int *err, char **err_str)
+{
+ return get_interface_list_findalldevs(err, err_str);
+}
+
+/*
+ * Get an error message string for a CANT_GET_INTERFACE_LIST error from
+ * "get_interface_list()".
+ */
+char *
+cant_get_if_list_error_message(const char *err_str)
+{
+ return ws_strdup_printf("Can't get list of interfaces: %s", err_str);
+}
+
+if_capabilities_t *
+get_if_capabilities_local(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str)
+{
+#ifdef HAVE_PCAP_CREATE
+ return get_if_capabilities_pcap_create(interface_opts, status,
+ status_str);
+#else
+ return get_if_capabilities_pcap_open_live(interface_opts, status,
+ status_str);
+#endif
+}
+
+pcap_t *
+open_capture_device_local(capture_options *capture_opts
+#ifndef HAVE_PCAP_CREATE
+ _U_
+#endif
+ ,
+ interface_options *interface_opts, int timeout,
+ cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE])
+{
+ /*
+ * We're not opening a remote device; use pcap_create() and
+ * pcap_activate() if we have them, so that we can set various
+ * options, otherwise use pcap_open_live().
+ */
+#ifdef HAVE_PCAP_CREATE
+ return open_capture_device_pcap_create(capture_opts,
+ interface_opts, timeout, open_status, open_status_str);
+#else
+ return open_capture_device_pcap_open_live(interface_opts, timeout,
+ open_status, open_status_str);
+#endif
+}
+
+/*
+ * Get the versions of libpcap, libpcap, and libnl with which we were
+ * compiled, and append them to a GString.
+ */
+void
+gather_caplibs_compile_info(feature_list l)
+{
+ /*
+ * NOTE: in *some* flavors of UN*X, the data from a shared
+ * library might be linked into executable images that are
+ * linked with that shared library, in which case you could
+ * look at pcap_version[] to get the version with which
+ * the program was compiled.
+ *
+ * In other flavors of UN*X, that doesn't happen, so
+ * pcap_version[] gives you the version the program is
+ * running with, not the version it was built with, and,
+ * in at least some of them, if the length of a data item
+ * referred to by the executable - such as the pcap_version[]
+ * string - isn't the same in the version of the library
+ * with which the program was built and the version with
+ * which it was run, the run-time linker will complain,
+ * which is Not Good.
+ *
+ * So, for now, we just give up on reporting the version
+ * of libpcap with which we were compiled.
+ */
+#ifdef HAVE_PCAP_REMOTE
+ /*
+ * We have remote pcap support in libpcap.
+ */
+ with_feature(l, "libpcap (including remote capture support)");
+#else
+ with_feature(l, "libpcap");
+#endif
+
+ /*
+ * XXX - these libraries are actually used only by dumpcap,
+ * but we mention them here so that a user reporting a bug
+ * can get information about dumpcap's libraries without
+ * having to run dumpcap.
+ */
+ /* LIBCAP */
+#ifdef HAVE_LIBCAP
+#ifdef _LINUX_CAPABILITY_VERSION
+ with_feature(l, "POSIX capabilities (Linux)");
+#else /* _LINUX_CAPABILITY_VERSION */
+ with_feature(l, "POSIX capabilities");
+#endif /* _LINUX_CAPABILITY_VERSION */
+#else /* HAVE_LIBCAP */
+ without_feature(l, "POSIX capabilities");
+#endif /* HAVE_LIBCAP */
+
+#ifdef __linux__
+ /* This is a Linux-specific library. */
+ /* LIBNL */
+#if defined(HAVE_LIBNL1)
+ with_feature(l, "libnl 1");
+#elif defined(HAVE_LIBNL2)
+ with_feature(l, "libnl 2");
+#elif defined(HAVE_LIBNL3)
+ with_feature(l, "libnl 3");
+#else /* no libnl */
+ without_feature(l, "libnl");
+#endif /* libnl version */
+#endif /* __linux__ */
+}
+
+void
+gather_caplibs_runtime_info(feature_list l)
+{
+ const char *vstr = pcap_lib_version();
+
+ /*
+ * Remove the substring "version" from the output of pcap_lib_version()
+ * to be consistent with our format.
+ */
+ if (g_str_has_prefix(vstr, "libpcap version ")) /* Sanity check */
+ with_feature(l, "libpcap %s", vstr + strlen("libpcap version "));
+ else
+ with_feature(l, "%s", vstr);
+}
+
+#else /* HAVE_LIBPCAP */
+
+/*
+ * Append an indication that we were not compiled with libpcap
+ * to a GString. Don't even bother mentioning the other
+ * libraries.
+ */
+void
+gather_caplibs_compile_info(feature_list l)
+{
+ without_feature(l, "libpcap");
+}
+
+void
+gather_caplibs_runtime_info(feature_list l _U_)
+{
+}
+
+#endif /* HAVE_LIBPCAP */
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 8
+ * tab-width: 8
+ * indent-tabs-mode: t
+ * End:
+ *
+ * vi: set shiftwidth=8 tabstop=8 noexpandtab:
+ * :indentSize=8:tabSize=8:noTabs=false:
+ */
diff --git a/capture/capture-pcap-util.c b/capture/capture-pcap-util.c
new file mode 100644
index 00000000..ac54bb36
--- /dev/null
+++ b/capture/capture-pcap-util.c
@@ -0,0 +1,1826 @@
+/* capture-pcap-util.c
+ * Utility routines for packet capture
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+#define WS_LOG_DOMAIN LOG_DOMAIN_CAPCHILD
+
+#ifdef HAVE_LIBPCAP
+
+#include <wireshark.h>
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <limits.h>
+#include <string.h>
+
+#include <sys/types.h>
+
+#ifdef HAVE_SYS_SOCKET_H
+#include <sys/socket.h>
+#endif
+
+#ifdef __APPLE__
+#include <dlfcn.h>
+#endif
+
+#include "ws_attributes.h"
+
+/*
+ * Linux bonding devices mishandle unknown ioctls; they fail
+ * with ENODEV rather than ENOTSUP, EOPNOTSUPP, or ENOTTY,
+ * so pcap_can_set_rfmon() returns a "no such device" indication
+ * if we try to do SIOCGIWMODE on them.
+ *
+ * So, on Linux, we check for bonding devices, if we can, before
+ * trying pcap_can_set_rfmon(), as pcap_can_set_rfmon() will
+ * end up trying SIOCGIWMODE on the device if that ioctl exists.
+ */
+#if defined(HAVE_PCAP_CREATE) && defined(__linux__)
+
+#include <sys/ioctl.h>
+
+/*
+ * If we're building for a Linux version that supports bonding,
+ * HAVE_BONDING will be defined.
+ */
+
+#ifdef HAVE_LINUX_SOCKIOS_H
+#include <linux/sockios.h>
+#endif
+
+#ifdef HAVE_LINUX_IF_BONDING_H
+#include <linux/if_bonding.h>
+#endif
+
+#if defined(BOND_INFO_QUERY_OLD) || defined(SIOCBONDINFOQUERY)
+#define HAVE_BONDING
+#endif
+
+#endif /* defined(HAVE_PCAP_CREATE) && defined(__linux__) */
+
+#include "capture/capture_ifinfo.h"
+#include "capture/capture-pcap-util.h"
+#include "capture/capture-pcap-util-int.h"
+#ifdef _WIN32
+#include "capture/capture-wpcap.h"
+#else
+#define ws_pcap_findalldevs_ex pcap_findalldevs_ex
+#endif
+
+#include <wsutil/file_util.h>
+#include <wsutil/please_report_bug.h>
+#include <wsutil/wslog.h>
+
+#ifndef _WIN32
+#include <netinet/in.h>
+#else
+#include <ws2tcpip.h>
+#endif
+
+#ifdef _WIN32
+#include "capture/capture_win_ifnames.h" /* windows friendly interface names */
+#endif
+
+#if defined(__FreeBSD__) || defined(__OpenBSD__)
+/*
+ * Needed for the code to get a device description.
+ */
+#include <errno.h>
+#include <net/if.h>
+#include <sys/sockio.h>
+#include <sys/ioctl.h>
+#endif
+
+/*
+ * Given an interface name, find the "friendly name" and interface
+ * type for the interface.
+ */
+
+#if defined(HAVE_MACOS_FRAMEWORKS)
+
+#include <CoreFoundation/CoreFoundation.h>
+#include <SystemConfiguration/SystemConfiguration.h>
+
+#include <wsutil/cfutils.h>
+
+/*
+ * On macOS, we get the "friendly name" and interface type for the interface
+ * from the System Configuration framework.
+ *
+ * To find the System Configuration framework information for the
+ * interface, we get all the interfaces that the System Configuration
+ * framework knows about and look for the one with a "BSD name" matching
+ * the interface name.
+ *
+ * If we find it, we use its "localized display name", if it has one, as
+ * the "friendly name".
+ *
+ * As for the interface type:
+ *
+ * Yes, fetching all the network addresses for an interface gets you an
+ * AF_LINK address, of type "struct sockaddr_dl", and, yes, that includes
+ * an SNMP MIB-II ifType value.
+ *
+ * However, it's IFT_ETHER, i.e. Ethernet, for AirPort interfaces,
+ * not IFT_IEEE80211 (which isn't defined in macOS in any case).
+ *
+ * Perhaps some other BSD-flavored OSes won't make this mistake;
+ * however, FreeBSD 7.0 and OpenBSD 4.2, at least, appear to have
+ * made the same mistake, at least for my Belkin ZyDAS stick.
+ *
+ * SCNetworkInterfaceGetInterfaceType() will get the interface
+ * type. The interface type is a CFString, and:
+ *
+ * kSCNetworkInterfaceTypeIEEE80211 means IF_WIRELESS;
+ * kSCNetworkInterfaceTypeBluetooth means IF_BLUETOOTH;
+ * kSCNetworkInterfaceTypeModem or
+ * kSCNetworkInterfaceTypePPP or
+ * maybe kSCNetworkInterfaceTypeWWAN means IF_DIALUP
+ */
+static void
+add_unix_interface_ifinfo(if_info_t *if_info, const char *name,
+ const char *description _U_)
+{
+ CFStringRef name_CFString;
+ CFArrayRef interfaces;
+ CFIndex num_interfaces;
+ CFIndex i;
+ SCNetworkInterfaceRef interface;
+ CFStringRef bsdname_CFString;
+ CFStringRef friendly_name_CFString;
+ CFStringRef interface_type_CFString;
+
+ interfaces = SCNetworkInterfaceCopyAll();
+ if (interfaces == NULL) {
+ /*
+ * Couldn't get a list of interfaces.
+ */
+ return;
+ }
+
+ name_CFString = CFStringCreateWithCString(kCFAllocatorDefault,
+ name, kCFStringEncodingUTF8);
+ if (name_CFString == NULL) {
+ /*
+ * Couldn't convert the interface name to a CFString.
+ */
+ CFRelease(interfaces);
+ return;
+ }
+
+ num_interfaces = CFArrayGetCount(interfaces);
+ for (i = 0; i < num_interfaces; i++) {
+ interface = (SCNetworkInterfaceRef)CFArrayGetValueAtIndex(interfaces, i);
+ bsdname_CFString = SCNetworkInterfaceGetBSDName(interface);
+ if (bsdname_CFString == NULL) {
+ /*
+ * This interface has no BSD name, so it's not
+ * a regular network interface.
+ */
+ continue;
+ }
+ if (CFStringCompare(name_CFString, bsdname_CFString, 0) == 0) {
+ /*
+ * This is the interface.
+ * First, get the friendly name.
+ */
+ friendly_name_CFString = SCNetworkInterfaceGetLocalizedDisplayName(interface);
+ if (friendly_name_CFString != NULL)
+ if_info->friendly_name = CFString_to_C_string(friendly_name_CFString);
+
+ /*
+ * Now get the interface type.
+ */
+ interface_type_CFString = SCNetworkInterfaceGetInterfaceType(interface);
+ if (CFStringCompare(interface_type_CFString,
+ kSCNetworkInterfaceTypeIEEE80211, 0) == kCFCompareEqualTo)
+ if_info->type = IF_WIRELESS;
+ else if (CFStringCompare(interface_type_CFString,
+ kSCNetworkInterfaceTypeBluetooth, 0) == kCFCompareEqualTo)
+ if_info->type = IF_BLUETOOTH;
+ else if (CFStringCompare(interface_type_CFString,
+ kSCNetworkInterfaceTypeModem, 0) == kCFCompareEqualTo)
+ if_info->type = IF_DIALUP;
+ else if (CFStringCompare(interface_type_CFString,
+ kSCNetworkInterfaceTypePPP, 0) == kCFCompareEqualTo)
+ if_info->type = IF_DIALUP;
+ else if (CFStringCompare(interface_type_CFString,
+ kSCNetworkInterfaceTypeWWAN, 0) == kCFCompareEqualTo)
+ if_info->type = IF_DIALUP;
+ else
+ if_info->type = IF_WIRED;
+ break;
+ }
+ }
+
+ CFRelease(interfaces);
+ CFRelease(name_CFString);
+}
+#elif defined(__linux__)
+/*
+ * Linux doesn't offer any form of "friendly name", but you can
+ * determine an interface type to some degree.
+ */
+static void
+add_unix_interface_ifinfo(if_info_t *if_info, const char *name,
+ const char *description _U_)
+{
+ char *wireless_path;
+ ws_statb64 statb;
+
+ /*
+ * Look for /sys/class/net/{device}/wireless. If it exists,
+ * it's a wireless interface.
+ */
+ wireless_path = ws_strdup_printf("/sys/class/net/%s/wireless", name);
+ if (wireless_path != NULL) {
+ if (ws_stat64(wireless_path, &statb) == 0)
+ if_info->type = IF_WIRELESS;
+ g_free(wireless_path);
+ }
+ if (if_info->type == IF_WIRED) {
+ /*
+ * We still don't know what it is. Check for
+ * Bluetooth and USB devices.
+ */
+ if (strstr(name, "bluetooth") != NULL) {
+ /*
+ * XXX - this is for raw Bluetooth capture; what
+ * about IP-over-Bluetooth devices?
+ */
+ if_info->type = IF_BLUETOOTH;
+ } else if (strstr(name, "usbmon") != NULL)
+ if_info->type = IF_USB;
+ }
+}
+#elif !defined(_WIN32)
+/*
+ * On other UN*Xes, if there is a description, it's a friendly
+ * name, and there is no vendor description. ("Other UN*Xes"
+ * currently means "FreeBSD and OpenBSD".)
+ */
+static void
+add_unix_interface_ifinfo(if_info_t *if_info, const char *name _U_,
+ const char *description)
+{
+ if_info->friendly_name = g_strdup(description);
+}
+#endif
+
+if_info_t *
+if_info_get(const char *name)
+{
+ char *description = NULL;
+ if_info_t *if_info;
+#ifdef SIOCGIFDESCR
+ /*
+ * Try to fetch the description of this interface.
+ * XXX - this is only here because libpcap has no API to
+ * get the description of a *single* interface; it really
+ * needs both an API to get pcapng-IDB-style attributes
+ * for a single interface and to get a list of interfaces
+ * with pcapng-IDB-style attributes for each interface.
+ */
+ int s;
+ struct ifreq ifrdesc;
+#ifndef IFDESCRSIZE
+ size_t descrlen = 64;
+#else
+ size_t descrlen = IFDESCRSIZE;
+#endif /* IFDESCRSIZE */
+
+ /*
+ * Get the description for the interface.
+ */
+ memset(&ifrdesc, 0, sizeof ifrdesc);
+ (void) g_strlcpy(ifrdesc.ifr_name, name, sizeof ifrdesc.ifr_name);
+ s = socket(AF_INET, SOCK_DGRAM, 0);
+ if (s >= 0) {
+#ifdef __FreeBSD__
+ /*
+ * On FreeBSD, if the buffer isn't big enough for the
+ * description, the ioctl succeeds, but the description
+ * isn't copied, ifr_buffer.length is set to the description
+ * length, and ifr_buffer.buffer is set to NULL.
+ */
+ for (;;) {
+ g_free(description);
+ if ((description = (char*)g_malloc(descrlen)) != NULL) {
+ ifrdesc.ifr_buffer.buffer = description;
+ ifrdesc.ifr_buffer.length = descrlen;
+ if (ioctl(s, SIOCGIFDESCR, &ifrdesc) == 0) {
+ if (ifrdesc.ifr_buffer.buffer ==
+ description)
+ break;
+ else
+ descrlen = ifrdesc.ifr_buffer.length;
+ } else {
+ /*
+ * Failed to get interface description.
+ */
+ g_free(description);
+ description = NULL;
+ break;
+ }
+ } else
+ break;
+ }
+#else /* __FreeBSD__ */
+ /*
+ * The only other OS that currently supports
+ * SIOCGIFDESCR is OpenBSD, and it has no way
+ * to get the description length - it's clamped
+ * to a maximum of IFDESCRSIZE.
+ */
+ if ((description = (char*)g_malloc(descrlen)) != NULL) {
+ ifrdesc.ifr_data = (caddr_t)description;
+ if (ioctl(s, SIOCGIFDESCR, &ifrdesc) != 0) {
+ /*
+ * Failed to get interface description.
+ */
+ g_free(description);
+ description = NULL;
+ }
+ }
+#endif /* __FreeBSD__ */
+ close(s);
+ if (description != NULL && strlen(description) == 0) {
+ /*
+ * Description is empty, so discard it.
+ */
+ g_free(description);
+ description = NULL;
+ }
+ }
+
+#ifdef __FreeBSD__
+ /*
+ * For FreeBSD, if we didn't get a description, and this is
+ * a device with a name of the form usbusN, label it as a USB
+ * bus.
+ */
+ if (description == NULL) {
+ if (strncmp(name, "usbus", 5) == 0) {
+ /*
+ * OK, it begins with "usbus".
+ */
+ long busnum;
+ char *p;
+
+ errno = 0;
+ busnum = strtol(name + 5, &p, 10);
+ if (errno == 0 && p != name + 5 && *p == '\0' &&
+ busnum >= 0 && busnum <= INT_MAX) {
+ /*
+ * OK, it's a valid number that's not
+ * bigger than INT_MAX. Construct
+ * a description from it.
+ */
+ static const char descr_prefix[] = "USB bus number ";
+ size_t descr_size;
+
+ /*
+ * Allow enough room for a 32-bit bus number.
+ * sizeof (descr_prefix) includes the
+ * terminating NUL.
+ */
+ descr_size = sizeof (descr_prefix) + 10;
+ description = g_malloc(descr_size);
+ if (description != NULL) {
+ snprintf(description, descr_size,
+ "%s%ld", descr_prefix, busnum);
+ }
+ }
+ }
+ }
+#endif /* __FreeBSD__ */
+#endif /* SIOCGIFDESCR */
+ if_info = if_info_new(name, description, false);
+ g_free(description);
+ return if_info;
+}
+
+void
+if_info_free(if_info_t *if_info)
+{
+ g_free(if_info->name);
+ g_free(if_info->friendly_name);
+ g_free(if_info->vendor_description);
+ g_free(if_info->extcap);
+ g_slist_free_full(if_info->addrs, g_free);
+ g_free(if_info);
+}
+
+if_info_t *
+if_info_new(const char *name, const char *description, bool loopback)
+{
+ if_info_t *if_info;
+#ifdef _WIN32
+ const char *guid_text;
+ GUID guid;
+#endif
+
+ if_info = g_new(if_info_t, 1);
+ if_info->name = g_strdup(name);
+ if_info->friendly_name = NULL; /* default - unknown */
+ if_info->vendor_description = NULL;
+ if_info->type = IF_WIRED; /* default */
+ if_info->extcap = g_strdup("");
+#ifdef _WIN32
+ /*
+ * Get the interface type.
+ *
+ * Much digging failed to reveal any obvious way to get something
+ * such as the SNMP MIB-II ifType value for an interface:
+ *
+ * https://www.iana.org/assignments/ianaiftype-mib/ianaiftype-mib
+ *
+ * by making some NDIS request. And even if there were such
+ * a way, there's no guarantee that the ifType reflects an
+ * interface type that a user would view as correct (for
+ * example, some systems report Wi-Fi interfaces as
+ * Ethernet interfaces).
+ *
+ * So we look for keywords in the vendor's interface
+ * description.
+ */
+ if (description && (strstr(description, "generic dialup") != NULL ||
+ strstr(description, "PPP/SLIP") != NULL)) {
+ if_info->type = IF_DIALUP;
+ } else if (description && (strstr(description, "Wireless") != NULL ||
+ strstr(description,"802.11") != NULL)) {
+ if_info->type = IF_WIRELESS;
+ } else if (description && (strstr(description, "AirPcap") != NULL ||
+ strstr(name, "airpcap") != NULL)) {
+ if_info->type = IF_AIRPCAP;
+ } else if (description && strstr(description, "Bluetooth") != NULL ) {
+ if_info->type = IF_BLUETOOTH;
+ } else if (description && strstr(description, "VMware") != NULL) {
+ /*
+ * Bridge, NAT, or host-only interface on a VMware host.
+ *
+ * XXX - what about guest interfaces?
+ */
+ if_info->type = IF_VIRTUAL;
+ }
+
+ /*
+ * On Windows, the "description" is a vendor description,
+ * and the friendly name isn't returned by Npcap/WinPcap.
+ * Fetch it ourselves.
+ */
+
+ /*
+ * Skip over the "\Device\NPF_" prefix in the device name,
+ * if present.
+ */
+ if (strncmp("\\Device\\NPF_", name, 12) == 0)
+ guid_text = name + 12;
+ else
+ guid_text = name;
+
+ /* Now try to parse what remains as a GUID. */
+ if (parse_as_guid(guid_text, &guid)) {
+ /*
+ * Success. Try to get a friendly name using the GUID.
+ * As this is a regular interface, the description is a
+ * vendor description.
+ */
+ if_info->friendly_name = get_interface_friendly_name_from_device_guid(&guid);
+ if_info->vendor_description = g_strdup(description);
+ } else {
+ /*
+ * This is probably not a regular interface; we only
+ * support NT 5 (W2K) and later, so all regular interfaces
+ * should have GUIDs at the end of the name. Therefore,
+ * the description, if supplied, is a friendly name
+ * provided by WinPcap, and there is no vendor
+ * description.
+ */
+ if_info->friendly_name = g_strdup(description);
+ if_info->vendor_description = NULL;
+ }
+#else
+ /*
+ * On UN*X, if there is a description, it's a friendly
+ * name, and there is no vendor description.
+ *
+ * Try the platform's way of getting a friendly name and
+ * interface type first.
+ *
+ * If that fails, then, for a loopback interface, give it the
+ * friendly name "Loopback" and, for VMware interfaces,
+ * give them the type IF_VIRTUAL.
+ */
+ add_unix_interface_ifinfo(if_info, name, description);
+ if (if_info->type == IF_WIRED) {
+ /*
+ * This is the default interface type.
+ *
+ * Bridge, NAT, or host-only interfaces on VMWare hosts
+ * have the name vmnet[0-9]+. Guests might use a native
+ * (LANCE or E1000) driver or the vmxnet driver. Check
+ * the name.
+ */
+ if (g_ascii_strncasecmp(name, "vmnet", 5) == 0)
+ if_info->type = IF_VIRTUAL;
+ else if (g_ascii_strncasecmp(name, "vmxnet", 6) == 0)
+ if_info->type = IF_VIRTUAL;
+ }
+ if (if_info->friendly_name == NULL) {
+ /*
+ * We couldn't get interface information using platform-
+ * dependent calls.
+ *
+ * If this is a loopback interface, give it a
+ * "friendly name" of "Loopback".
+ */
+ if (loopback)
+ if_info->friendly_name = g_strdup("Loopback");
+ }
+ if_info->vendor_description = NULL;
+#endif
+ if_info->loopback = loopback;
+ if_info->addrs = NULL;
+ return if_info;
+}
+
+void
+if_info_add_address(if_info_t *if_info, struct sockaddr *addr)
+{
+ if_addr_t *if_addr;
+ struct sockaddr_in *ai;
+ struct sockaddr_in6 *ai6;
+
+ switch (addr->sa_family) {
+
+ case AF_INET:
+ ai = (struct sockaddr_in *)(void *)addr;
+ if_addr = (if_addr_t *)g_malloc(sizeof(*if_addr));
+ if_addr->ifat_type = IF_AT_IPv4;
+ if_addr->addr.ip4_addr = ai->sin_addr.s_addr;
+ if_info->addrs = g_slist_prepend(if_info->addrs, if_addr);
+ break;
+
+ case AF_INET6:
+ ai6 = (struct sockaddr_in6 *)(void *)addr;
+ if_addr = (if_addr_t *)g_malloc(sizeof(*if_addr));
+ if_addr->ifat_type = IF_AT_IPv6;
+ memcpy((void *)&if_addr->addr.ip6_addr,
+ (void *)&ai6->sin6_addr.s6_addr,
+ sizeof if_addr->addr.ip6_addr);
+ if_info->addrs = g_slist_prepend(if_info->addrs, if_addr);
+ break;
+ }
+}
+
+/*
+ * Get all IP address information for the given interface.
+ */
+static void
+if_info_ip(if_info_t *if_info, pcap_if_t *d)
+{
+ pcap_addr_t *a;
+
+ /* All addresses */
+ for (a = d->addresses; a != NULL; a = a->next) {
+ if (a->addr != NULL)
+ if_info_add_address(if_info, a->addr);
+ }
+
+ if(if_info->addrs){
+ if_info->addrs = g_slist_reverse(if_info->addrs);
+ }
+}
+
+#ifdef HAVE_PCAP_REMOTE
+GList *
+get_interface_list_findalldevs_ex(const char *hostname, const char *port,
+ int auth_type, const char *username,
+ const char *passwd, int *err, char **err_str)
+{
+ char source[PCAP_BUF_SIZE];
+ struct pcap_rmtauth auth;
+ GList *il = NULL;
+ pcap_if_t *alldevs, *dev;
+ if_info_t *if_info;
+ /*
+ * WinPcap can overflow PCAP_ERRBUF_SIZE if the host is unreachable.
+ * Fudge a larger size.
+ */
+ char errbuf[PCAP_ERRBUF_SIZE*4];
+
+ if (pcap_createsrcstr(source, PCAP_SRC_IFREMOTE, hostname, port,
+ NULL, errbuf) == -1) {
+ *err = CANT_GET_INTERFACE_LIST;
+ if (strcmp(errbuf, "not supported") == 0) {
+ /*
+ * macOS 14's pcap_createsrcstr(), which is a
+ * stub that always returns -1 with an error
+ * message of "not supported".
+ *
+ * In this case, as we passed it an rpcap://
+ * URL, treat that as meaning "remote capture
+ * not supported".
+ */
+ g_strlcpy(errbuf, "Remote capture not supported",
+ PCAP_ERRBUF_SIZE);
+ }
+ if (err_str != NULL)
+ *err_str = cant_get_if_list_error_message(errbuf);
+ return NULL;
+ }
+
+ auth.type = auth_type;
+ auth.username = g_strdup(username);
+ auth.password = g_strdup(passwd);
+
+ if (ws_pcap_findalldevs_ex(source, &auth, &alldevs, errbuf) == -1) {
+ *err = CANT_GET_INTERFACE_LIST;
+ if (strcmp(errbuf, "not supported") == 0) {
+ /*
+ * macOS 14's pcap_findalldevs_ex(), which is a
+ * stub that always returns -1 with an error
+ * message of "not supported".
+ *
+ * In this case, as we passed it an rpcap://
+ * URL, treat that as meaning "remote capture
+ * not supported".
+ */
+ g_strlcpy(errbuf, "Remote capture not supported",
+ PCAP_ERRBUF_SIZE);
+ }
+ if (err_str != NULL)
+ *err_str = cant_get_if_list_error_message(errbuf);
+ g_free(auth.username);
+ g_free(auth.password);
+ return NULL;
+ }
+
+ if (alldevs == NULL) {
+ /*
+ * No interfaces found.
+ */
+ *err = 0;
+ if (err_str != NULL)
+ *err_str = NULL;
+ g_free(auth.username);
+ g_free(auth.password);
+ return NULL;
+ }
+
+ for (dev = alldevs; dev != NULL; dev = dev->next) {
+ if_info = if_info_new(dev->name, dev->description,
+ (dev->flags & PCAP_IF_LOOPBACK) ? true : false);
+ il = g_list_append(il, if_info);
+ if_info_ip(if_info, dev);
+ }
+ pcap_freealldevs(alldevs);
+ g_free(auth.username);
+ g_free(auth.password);
+
+ return il;
+}
+#endif
+
+GList *
+get_interface_list_findalldevs(int *err, char **err_str)
+{
+ GList *il = NULL;
+ pcap_if_t *alldevs, *dev;
+ if_info_t *if_info;
+ char errbuf[PCAP_ERRBUF_SIZE];
+
+ if (pcap_findalldevs(&alldevs, errbuf) == -1) {
+ *err = CANT_GET_INTERFACE_LIST;
+ if (err_str != NULL)
+ *err_str = cant_get_if_list_error_message(errbuf);
+ return NULL;
+ }
+
+ if (alldevs == NULL) {
+ /*
+ * No interfaces found.
+ */
+ *err = 0;
+ if (err_str != NULL)
+ *err_str = NULL;
+ return NULL;
+ }
+
+ for (dev = alldevs; dev != NULL; dev = dev->next) {
+ if_info = if_info_new(dev->name, dev->description,
+ (dev->flags & PCAP_IF_LOOPBACK) ? true : false);
+ il = g_list_append(il, if_info);
+ if_info_ip(if_info, dev);
+ }
+ pcap_freealldevs(alldevs);
+
+ return il;
+}
+
+static void
+free_if_cb(void * data, void * user_data _U_)
+{
+ if_info_free((if_info_t *)data);
+}
+
+void
+free_interface_list(GList *if_list)
+{
+ g_list_foreach(if_list, free_if_cb, NULL);
+ g_list_free(if_list);
+}
+
+static void
+free_linktype_cb(void * data, void * user_data _U_)
+{
+ data_link_info_t *linktype_info = (data_link_info_t *)data;
+
+ g_free(linktype_info->name);
+ g_free(linktype_info->description);
+ g_free(linktype_info);
+}
+
+static void
+free_timestamp_cb(void * data, void * user_data _U_)
+{
+ timestamp_info_t *timestamp_info = (timestamp_info_t *)data;
+
+ g_free(timestamp_info->name);
+ g_free(timestamp_info->description);
+ g_free(data);
+}
+
+void
+free_if_capabilities(if_capabilities_t *caps)
+{
+ g_list_foreach(caps->data_link_types, free_linktype_cb, NULL);
+ g_list_free(caps->data_link_types);
+
+ g_list_foreach(caps->timestamp_types, free_timestamp_cb, NULL);
+ g_list_free(caps->timestamp_types);
+
+ g_free(caps);
+}
+
+const char *
+linktype_val_to_name(int dlt)
+{
+ return pcap_datalink_val_to_name(dlt);
+}
+
+int
+linktype_name_to_val(const char *linktype)
+{
+ return pcap_datalink_name_to_val(linktype);
+}
+
+/*
+ * Get the data-link type for a libpcap device.
+ * This works around AIX 5.x's non-standard and incompatible-with-the-
+ * rest-of-the-universe libpcap.
+ */
+int
+get_pcap_datalink(pcap_t *pch,
+#ifdef _AIX
+ const char* devicename
+#else
+ const char* devicename _U_
+#endif
+ )
+{
+ int datalink;
+#ifdef _AIX
+ const char *ifacename;
+#endif
+
+ datalink = pcap_datalink(pch);
+#ifdef _AIX
+
+ /*
+ * The libpcap that comes with AIX 5.x uses RFC 1573 ifType values
+ * rather than DLT_ values for link-layer types; the ifType values
+ * for LAN devices are:
+ *
+ * Ethernet 6
+ * 802.3 7
+ * Token Ring 9
+ * FDDI 15
+ *
+ * and the ifType value for a loopback device is 24.
+ *
+ * The AIX names for LAN devices begin with:
+ *
+ * Ethernet en
+ * 802.3 et
+ * Token Ring tr
+ * FDDI fi
+ *
+ * and the AIX names for loopback devices begin with "lo".
+ *
+ * (The difference between "Ethernet" and "802.3" is presumably
+ * whether packets have an Ethernet header, with a packet type,
+ * or an 802.3 header, with a packet length, followed by an 802.2
+ * header and possibly a SNAP header.)
+ *
+ * If the device name matches "datalink" interpreted as an ifType
+ * value, rather than as a DLT_ value, we will assume this is AIX's
+ * non-standard, incompatible libpcap, rather than a standard libpcap,
+ * and will map the link-layer type to the standard DLT_ value for
+ * that link-layer type, as that's what the rest of Wireshark expects.
+ *
+ * (This means the capture files won't be readable by a tcpdump
+ * linked with AIX's non-standard libpcap, but so it goes. They
+ * *will* be readable by standard versions of tcpdump, Wireshark,
+ * and so on.)
+ *
+ * XXX - if we conclude we're using AIX libpcap, should we also
+ * set a flag to cause us to assume the time stamps are in
+ * seconds-and-nanoseconds form, and to convert them to
+ * seconds-and-microseconds form before processing them and
+ * writing them out?
+ */
+
+ /*
+ * Find the last component of the device name, which is the
+ * interface name.
+ */
+ ifacename = strchr(devicename, '/');
+ if (ifacename == NULL)
+ ifacename = devicename;
+
+ /* See if it matches any of the LAN device names. */
+ if (strncmp(ifacename, "en", 2) == 0) {
+ if (datalink == 6) {
+ /*
+ * That's the RFC 1573 value for Ethernet;
+ * map it to DLT_EN10MB.
+ */
+ datalink = 1;
+ }
+ } else if (strncmp(ifacename, "et", 2) == 0) {
+ if (datalink == 7) {
+ /*
+ * That's the RFC 1573 value for 802.3;
+ * map it to DLT_EN10MB.
+ *
+ * (libpcap, tcpdump, Wireshark, etc. don't
+ * care if it's Ethernet or 802.3.)
+ */
+ datalink = 1;
+ }
+ } else if (strncmp(ifacename, "tr", 2) == 0) {
+ if (datalink == 9) {
+ /*
+ * That's the RFC 1573 value for 802.5 (Token Ring);
+ * map it to DLT_IEEE802, which is what's used for
+ * Token Ring.
+ */
+ datalink = 6;
+ }
+ } else if (strncmp(ifacename, "fi", 2) == 0) {
+ if (datalink == 15) {
+ /*
+ * That's the RFC 1573 value for FDDI;
+ * map it to DLT_FDDI.
+ */
+ datalink = 10;
+ }
+ } else if (strncmp(ifacename, "lo", 2) == 0) {
+ if (datalink == 24) {
+ /*
+ * That's the RFC 1573 value for "software loopback"
+ * devices; map it to DLT_NULL, which is what's used
+ * for loopback devices on BSD.
+ */
+ datalink = 0;
+ }
+ }
+#endif
+
+ return datalink;
+}
+
+/* Set the data link type on a pcap. */
+bool
+set_pcap_datalink(pcap_t *pcap_h, int datalink, char *name,
+ char *errmsg, size_t errmsg_len,
+ char *secondary_errmsg, size_t secondary_errmsg_len)
+{
+ char *set_datalink_err_str;
+
+ if (datalink == -1)
+ return true; /* just use the default */
+ if (pcap_set_datalink(pcap_h, datalink) == 0)
+ return true; /* no error */
+ set_datalink_err_str = pcap_geterr(pcap_h);
+ snprintf(errmsg, errmsg_len, "Unable to set data link type on interface '%s' (%s).",
+ name, set_datalink_err_str);
+ /*
+ * If the error isn't "XXX is not one of the DLTs supported by this device",
+ * tell the user to tell the Wireshark developers about it.
+ */
+ if (strstr(set_datalink_err_str, "is not one of the DLTs supported by this device") == NULL)
+ snprintf(secondary_errmsg, secondary_errmsg_len,
+ "%s", please_report_bug());
+ else
+ secondary_errmsg[0] = '\0';
+ return false;
+}
+
+static data_link_info_t *
+create_data_link_info(int dlt)
+{
+ data_link_info_t *data_link_info;
+ const char *text;
+
+ data_link_info = g_new(data_link_info_t, 1);
+ data_link_info->dlt = dlt;
+ text = pcap_datalink_val_to_name(dlt);
+ if (text != NULL)
+ data_link_info->name = g_strdup(text);
+ else
+ data_link_info->name = ws_strdup_printf("DLT %d", dlt);
+ text = pcap_datalink_val_to_description(dlt);
+ data_link_info->description = g_strdup(text);
+ return data_link_info;
+}
+
+static GList *
+get_data_link_types(pcap_t *pch, interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str)
+{
+ GList *data_link_types;
+ int deflt;
+ int *linktypes;
+ int i, nlt;
+ data_link_info_t *data_link_info;
+
+ deflt = get_pcap_datalink(pch, interface_opts->name);
+ nlt = pcap_list_datalinks(pch, &linktypes);
+ if (nlt < 0) {
+ /*
+ * A negative return is an error.
+ */
+#ifdef HAVE_PCAP_CREATE
+ /*
+ * If we have pcap_create(), we have
+ * pcap_statustostr(), and we can get back errors
+ * other than PCAP_ERROR (-1), such as
+ * PCAP_ERROR_NOT_ACTIVATED. and we should report
+ * them properly.
+ */
+ switch (nlt) {
+
+ case PCAP_ERROR:
+ *status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *status_str = ws_strdup_printf("pcap_list_datalinks() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ default:
+ /*
+ * This "shouldn't happen".
+ */
+ *status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *status_str = ws_strdup_printf("pcap_list_datalinks() failed: %s - %s",
+ pcap_statustostr(nlt), pcap_geterr(pch));
+ break;
+ }
+#else /* HAVE_PCAP_CREATE */
+ *status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *status_str = ws_strdup_printf("pcap_list_datalinks() failed: %s",
+ pcap_geterr(pch));
+#endif /* HAVE_PCAP_CREATE */
+ return NULL;
+ }
+ data_link_types = NULL;
+ for (i = 0; i < nlt; i++) {
+ data_link_info = create_data_link_info(linktypes[i]);
+
+ /*
+ * XXX - for 802.11, make the most detailed 802.11
+ * version the default, rather than the one the
+ * device has as the default?
+ */
+ if (linktypes[i] == deflt)
+ data_link_types = g_list_prepend(data_link_types,
+ data_link_info);
+ else
+ data_link_types = g_list_append(data_link_types,
+ data_link_info);
+ }
+#ifdef HAVE_PCAP_FREE_DATALINKS
+ pcap_free_datalinks(linktypes);
+#else
+ /*
+ * In Windows, there's no guarantee that if you have a library
+ * built with one version of the MSVC++ run-time library, and
+ * it returns a pointer to allocated data, you can free that
+ * data from a program linked with another version of the
+ * MSVC++ run-time library.
+ *
+ * This is not an issue on UN*X.
+ *
+ * See the mail threads starting at
+ *
+ * https://www.winpcap.org/pipermail/winpcap-users/2006-September/001421.html
+ *
+ * and
+ *
+ * https://www.winpcap.org/pipermail/winpcap-users/2008-May/002498.html
+ */
+#ifndef _WIN32
+#define xx_free free /* hack so checkAPIs doesn't complain */
+ xx_free(linktypes);
+#endif /* _WIN32 */
+#endif /* HAVE_PCAP_FREE_DATALINKS */
+
+ *status_str = NULL;
+ return data_link_types;
+}
+
+/* Get supported timestamp types for a libpcap device. */
+static GList*
+get_pcap_timestamp_types(pcap_t *pch _U_, char **err_str _U_)
+{
+ GList *list = NULL;
+#ifdef HAVE_PCAP_SET_TSTAMP_TYPE
+ int *types;
+ int ntypes = pcap_list_tstamp_types(pch, &types);
+
+ if (err_str)
+ *err_str = ntypes < 0 ? pcap_geterr(pch) : NULL;
+
+ if (ntypes <= 0)
+ return NULL;
+
+ while (ntypes--) {
+ timestamp_info_t *info = (timestamp_info_t *)g_malloc(sizeof *info);
+ info->name = g_strdup(pcap_tstamp_type_val_to_name(types[ntypes]));
+ info->description = g_strdup(pcap_tstamp_type_val_to_description(types[ntypes]));
+ list = g_list_prepend(list, info);
+ }
+
+ pcap_free_tstamp_types(types);
+#endif
+ return list;
+}
+
+#ifdef HAVE_PCAP_SET_TSTAMP_PRECISION
+/*
+ * Request high-resolution time stamps.
+ *
+ * We don't check for errors - if this fails, we just live with boring old
+ * microsecond-resolution time stamps. The only errors pcap_set_tstamp_precision()
+ * is documenting as returning are PCAP_ERROR_TSTAMP_PRECISION_NOTSUP, which just
+ * means we can't do nanosecond precision on this adapter, in which case we
+ * just live with whatever resolution we get by default, and
+ * PCAP_ERROR_ACTIVATED, which shouldn't happen as we shouldn't call this
+ * after we've activated the pcap_t.
+ */
+void
+request_high_resolution_timestamp(pcap_t *pcap_h)
+{
+#ifdef __APPLE__
+ /*
+ * On macOS, if you build with a newer SDK, pcap_set_tstamp_precision()
+ * is available, so the code will be built with it.
+ *
+ * However, if you then try to run on an older release that
+ * doesn't have pcap_set_tstamp_precision(), the dynamic linker
+ * will fail, as it won't find pcap_set_tstamp_precision().
+ *
+ * libpcap doesn't use macOS "weak linking" for new routines,
+ * so we can't just check whether a pointer to
+ * pcap_set_tstamp_precision() is null and, if it is, not
+ * call it. We have to, instead, use dlopen() to load
+ * libpcap, and dlsym() to find a pointer to pcap_set_tstamp_precision(),
+ * and if we find the pointer, call it.
+ */
+ static bool initialized = false;
+ static int (*p_pcap_set_tstamp_precision)(pcap_t *, int);
+
+ if (!initialized) {
+ p_pcap_set_tstamp_precision =
+ (int (*)(pcap_t *, int))
+ dlsym(RTLD_NEXT, "pcap_set_tstamp_precision");
+ initialized = true;
+ }
+ if (p_pcap_set_tstamp_precision != NULL)
+ (*p_pcap_set_tstamp_precision)(pcap_h, PCAP_TSTAMP_PRECISION_NANO);
+#else /* __APPLE__ */
+ /*
+ * On other UN*Xes we require that we be run on an OS version
+ * with a libpcap equal to or later than the version with which
+ * we were built.
+ */
+ pcap_set_tstamp_precision(pcap_h, PCAP_TSTAMP_PRECISION_NANO);
+#endif /* __APPLE__ */
+}
+
+/*
+ * Return true if the pcap_t in question is set up for high-precision
+ * time stamps, false otherwise.
+ */
+bool
+have_high_resolution_timestamp(pcap_t *pcap_h)
+{
+#ifdef __APPLE__
+ /*
+ * See above.
+ */
+ static bool initialized = false;
+ static int (*p_pcap_get_tstamp_precision)(pcap_t *);
+
+ if (!initialized) {
+ p_pcap_get_tstamp_precision =
+ (int (*)(pcap_t *))
+ dlsym(RTLD_NEXT, "pcap_get_tstamp_precision");
+ initialized = true;
+ }
+ if (p_pcap_get_tstamp_precision != NULL)
+ return (*p_pcap_get_tstamp_precision)(pcap_h) == PCAP_TSTAMP_PRECISION_NANO;
+ else
+ return false; /* Can't get implies couldn't set */
+#else /* __APPLE__ */
+ /*
+ * On other UN*Xes we require that we be run on an OS version
+ * with a libpcap equal to or later than the version with which
+ * we were built.
+ */
+ return pcap_get_tstamp_precision(pcap_h) == PCAP_TSTAMP_PRECISION_NANO;
+#endif /* __APPLE__ */
+}
+
+#endif /* HAVE_PCAP_SET_TSTAMP_PRECISION */
+
+#ifdef HAVE_PCAP_CREATE
+#ifdef HAVE_BONDING
+static bool
+is_linux_bonding_device(const char *ifname)
+{
+ int fd;
+ struct ifreq ifr;
+ ifbond ifb;
+
+ fd = socket(PF_INET, SOCK_DGRAM, 0);
+ if (fd == -1)
+ return false;
+
+ memset(&ifr, 0, sizeof ifr);
+ (void) g_strlcpy(ifr.ifr_name, ifname, sizeof ifr.ifr_name);
+ memset(&ifb, 0, sizeof ifb);
+ ifr.ifr_data = (caddr_t)&ifb;
+#if defined(SIOCBONDINFOQUERY)
+ if (ioctl(fd, SIOCBONDINFOQUERY, &ifr) == 0) {
+ close(fd);
+ return true;
+ }
+#else
+ if (ioctl(fd, BOND_INFO_QUERY_OLD, &ifr) == 0) {
+ close(fd);
+ return true;
+ }
+#endif
+
+ close(fd);
+ return false;
+}
+#else
+static bool
+is_linux_bonding_device(const char *ifname _U_)
+{
+ return false;
+}
+#endif
+
+if_capabilities_t *
+get_if_capabilities_pcap_create(interface_options *interface_opts,
+ cap_device_open_status *open_status, char **open_status_str)
+{
+ if_capabilities_t *caps;
+ char errbuf[PCAP_ERRBUF_SIZE];
+ pcap_t *pch;
+ int status;
+
+ pch = pcap_create(interface_opts->name, errbuf);
+ if (pch == NULL) {
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = g_strdup(errbuf);
+ return NULL;
+ }
+
+ if (is_linux_bonding_device(interface_opts->name)) {
+ /*
+ * Linux bonding device; not Wi-Fi, so no monitor mode, and
+ * calling pcap_can_set_rfmon() might get a "no such device"
+ * error.
+ */
+ status = 0;
+ } else {
+ /*
+ * Not a Linux bonding device, so go ahead.
+ */
+ status = pcap_can_set_rfmon(pch);
+ }
+ if (status < 0) {
+ /* Error. */
+ switch (status) {
+
+ case PCAP_ERROR_NO_SUCH_DEVICE:
+ *open_status = CAP_DEVICE_OPEN_ERROR_NO_SUCH_DEVICE;
+ *open_status_str = ws_strdup_printf("pcap_can_set_rfmon() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ case PCAP_ERROR_PERM_DENIED:
+ *open_status = CAP_DEVICE_OPEN_ERROR_PERM_DENIED;
+ *open_status_str = ws_strdup_printf("pcap_can_set_rfmon() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ case PCAP_ERROR:
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = ws_strdup_printf("pcap_can_set_rfmon() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ default:
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = ws_strdup_printf("pcap_can_set_rfmon() failed: %s - %s",
+ pcap_statustostr(status), pcap_geterr(pch));
+ break;
+ }
+ pcap_close(pch);
+ return NULL;
+ }
+ caps = (if_capabilities_t *)g_malloc(sizeof *caps);
+ if (status == 0)
+ caps->can_set_rfmon = false;
+ else if (status == 1) {
+ caps->can_set_rfmon = true;
+ if (interface_opts->monitor_mode)
+ pcap_set_rfmon(pch, 1);
+ } else {
+ /*
+ * This "should not happen".
+ */
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = ws_strdup_printf("pcap_can_set_rfmon() returned %d",
+ status);
+ pcap_close(pch);
+ g_free(caps);
+ return NULL;
+ }
+
+ status = pcap_activate(pch);
+ if (status < 0) {
+ /* Error. */
+ switch (status) {
+
+ case PCAP_ERROR_NO_SUCH_DEVICE:
+ *open_status = CAP_DEVICE_OPEN_ERROR_NO_SUCH_DEVICE;
+ *open_status_str = ws_strdup_printf("pcap_activate() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ case PCAP_ERROR_PERM_DENIED:
+ *open_status = CAP_DEVICE_OPEN_ERROR_PERM_DENIED;
+ *open_status_str = ws_strdup_printf("pcap_activate() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ case PCAP_ERROR_IFACE_NOT_UP:
+ *open_status = CAP_DEVICE_OPEN_ERROR_IFACE_NOT_UP;
+ *open_status_str = ws_strdup_printf("pcap_activate() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ case PCAP_ERROR:
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = ws_strdup_printf("pcap_activate() failed: %s",
+ pcap_geterr(pch));
+ break;
+
+ default:
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = ws_strdup_printf("pcap_activate() failed: %s - %s",
+ pcap_statustostr(status), pcap_geterr(pch));
+ break;
+ }
+ pcap_close(pch);
+ g_free(caps);
+ return NULL;
+ }
+
+ caps->data_link_types = get_data_link_types(pch, interface_opts,
+ open_status, open_status_str);
+ if (caps->data_link_types == NULL) {
+ pcap_close(pch);
+ g_free(caps);
+ return NULL;
+ }
+
+ caps->timestamp_types = get_pcap_timestamp_types(pch, NULL);
+
+ pcap_close(pch);
+
+ if (open_status_str != NULL)
+ *open_status_str = NULL;
+ return caps;
+}
+
+pcap_t *
+open_capture_device_pcap_create(
+#if defined(HAVE_PCAP_SET_TSTAMP_PRECISION)
+ capture_options* capture_opts,
+#else
+ capture_options* capture_opts _U_,
+#endif
+ interface_options *interface_opts, int timeout,
+ cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE])
+{
+ pcap_t *pcap_h;
+ int status;
+
+ ws_debug("Calling pcap_create() using %s.", interface_opts->name);
+ pcap_h = pcap_create(interface_opts->name, *open_status_str);
+ ws_debug("pcap_create() returned %p.", (void *)pcap_h);
+ if (pcap_h == NULL) {
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ return NULL;
+ }
+ if (interface_opts->has_snaplen) {
+ ws_debug("Calling pcap_set_snaplen() with snaplen %d.",
+ interface_opts->snaplen);
+ pcap_set_snaplen(pcap_h, interface_opts->snaplen);
+ }
+ ws_debug("Calling pcap_set_promisc() with promisc_mode %d.",
+ interface_opts->promisc_mode);
+ pcap_set_promisc(pcap_h, interface_opts->promisc_mode);
+ pcap_set_timeout(pcap_h, timeout);
+
+#ifdef HAVE_PCAP_SET_TSTAMP_PRECISION
+ /*
+ * If we're writing pcapng files, try to enable
+ * nanosecond-resolution capture; any code that
+ * can read pcapng files must be able to handle
+ * nanosecond-resolution time stamps. We don't
+ * care whether it succeeds or fails - if it fails,
+ * we just use the microsecond-precision time stamps
+ * we get.
+ *
+ * If we're writing pcap files, don't try to enable
+ * nanosecond-resolution capture, as not all code
+ * that reads pcap files recognizes the nanosecond-
+ * resolution pcap file magic number.
+ * We don't care whether this succeeds or fails; if it
+ * fails (because we don't have pcap_set_tstamp_precision(),
+ * or because we do but the OS or device doesn't support
+ * nanosecond resolution timing), we just use microsecond-
+ * resolution time stamps.
+ */
+ if (capture_opts->use_pcapng)
+ request_high_resolution_timestamp(pcap_h);
+#endif /* HAVE_PCAP_SET_TSTAMP_PRECISION */
+
+#ifdef HAVE_PCAP_SET_TSTAMP_TYPE
+ if (interface_opts->timestamp_type) {
+ status = pcap_set_tstamp_type(pcap_h, interface_opts->timestamp_type_id);
+ /*
+ * XXX - what if it fails because that time stamp type
+ * isn't supported?
+ */
+ if (status == PCAP_ERROR) {
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ pcap_close(pcap_h);
+ return NULL;
+ }
+ }
+#endif /* HAVE_PCAP_SET_TSTAMP_PRECISION */
+
+ ws_debug("buffersize %d.", interface_opts->buffer_size);
+ if (interface_opts->buffer_size != 0)
+ pcap_set_buffer_size(pcap_h,
+ interface_opts->buffer_size * 1024 * 1024);
+ ws_debug("monitor_mode %d.", interface_opts->monitor_mode);
+ if (interface_opts->monitor_mode)
+ pcap_set_rfmon(pcap_h, 1);
+ status = pcap_activate(pcap_h);
+ ws_debug("pcap_activate() returned %d.", status);
+ if (status < 0) {
+ /* Failed to activate, set to NULL */
+ switch (status) {
+
+ case PCAP_ERROR_NO_SUCH_DEVICE:
+ *open_status = CAP_DEVICE_OPEN_ERROR_NO_SUCH_DEVICE;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+ case PCAP_ERROR_PERM_DENIED:
+ *open_status = CAP_DEVICE_OPEN_ERROR_PERM_DENIED;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+#ifdef HAVE_PCAP_ERROR_PROMISC_PERM_DENIED
+ case PCAP_ERROR_PROMISC_PERM_DENIED:
+ *open_status = CAP_DEVICE_OPEN_ERROR_PROMISC_PERM_DENIED;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+#endif
+
+ case PCAP_ERROR_RFMON_NOTSUP:
+ *open_status = CAP_DEVICE_OPEN_ERROR_RFMON_NOTSUP;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+ case PCAP_ERROR_IFACE_NOT_UP:
+ *open_status = CAP_DEVICE_OPEN_ERROR_IFACE_NOT_UP;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+ case PCAP_ERROR:
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+ default:
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ snprintf(*open_status_str, sizeof *open_status_str,
+ "%s - %s", pcap_statustostr(status), pcap_geterr(pcap_h));
+ break;
+ }
+ pcap_close(pcap_h);
+ return NULL;
+ }
+ if (status > 0) {
+ /*
+ * Warning. The call succeeded, but something happened
+ * that the user might want to know.
+ */
+ switch (status) {
+
+ case PCAP_WARNING_PROMISC_NOTSUP:
+ *open_status = CAP_DEVICE_OPEN_WARNING_PROMISC_NOTSUP;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+#ifdef HAVE_PCAP_WARNING_TSTAMP_TYPE_NOTSUP
+ case PCAP_WARNING_TSTAMP_TYPE_NOTSUP:
+ *open_status = CAP_DEVICE_OPEN_WARNING_TSTAMP_TYPE_NOTSUP;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+#endif
+
+ case PCAP_WARNING:
+ *open_status = CAP_DEVICE_OPEN_WARNING_OTHER;
+ (void) g_strlcpy(*open_status_str, pcap_geterr(pcap_h),
+ sizeof *open_status_str);
+ break;
+
+ default:
+ *open_status = CAP_DEVICE_OPEN_WARNING_OTHER;
+ snprintf(*open_status_str, sizeof *open_status_str,
+ "%s - %s", pcap_statustostr(status), pcap_geterr(pcap_h));
+ break;
+ }
+ } else {
+ /*
+ * No warning issued.
+ */
+ *open_status = CAP_DEVICE_OPEN_NO_ERR;
+ }
+ return pcap_h;
+}
+#endif /* HAVE_PCAP_CREATE */
+
+if_capabilities_t *
+get_if_capabilities_pcap_open_live(interface_options *interface_opts,
+ cap_device_open_status *open_status, char **open_status_str)
+{
+ if_capabilities_t *caps;
+ char errbuf[PCAP_ERRBUF_SIZE];
+ pcap_t *pch;
+
+ pch = pcap_open_live(interface_opts->name, MIN_PACKET_SIZE, 0, 0,
+ errbuf);
+ if (pch == NULL) {
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ *open_status_str = g_strdup(errbuf[0] == '\0' ? "Unknown error (pcap bug; actual error cause not reported)" : errbuf);
+ return NULL;
+ }
+
+ caps = (if_capabilities_t *)g_malloc(sizeof *caps);
+ caps->can_set_rfmon = false;
+ caps->data_link_types = get_data_link_types(pch, interface_opts,
+ open_status, open_status_str);
+ if (caps->data_link_types == NULL) {
+ pcap_close(pch);
+ g_free(caps);
+ return NULL;
+ }
+
+ caps->timestamp_types = get_pcap_timestamp_types(pch, NULL);
+
+ pcap_close(pch);
+
+ *open_status = CAP_DEVICE_OPEN_NO_ERR;
+ *open_status_str = NULL;
+ return caps;
+}
+
+pcap_t *
+open_capture_device_pcap_open_live(interface_options *interface_opts,
+ int timeout, cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE])
+{
+ pcap_t *pcap_h;
+ int snaplen;
+
+ if (interface_opts->has_snaplen)
+ snaplen = interface_opts->snaplen;
+ else {
+ /*
+ * Default - use the non-D-Bus maximum snapshot length of
+ * 256KB, which should be big enough (libpcap didn't get
+ * D-Bus support until after it goet pcap_create() and
+ * pcap_activate(), so we don't have D-Bus support and
+ * don't have to worry about really huge packets).
+ */
+ snaplen = 256*1024;
+ }
+ ws_debug("pcap_open_live() calling using name %s, snaplen %d, promisc_mode %d.",
+ interface_opts->name, snaplen, interface_opts->promisc_mode);
+ /*
+ * This might succeed but put a messsage in *open_status_str;
+ * that means that a warning was issued.
+ *
+ * Clear the error message buffer, so that if it's not an empty
+ * string after the call, we know a warning was issued.
+ */
+ (*open_status_str)[0] = '\0';
+ pcap_h = pcap_open_live(interface_opts->name, snaplen,
+ interface_opts->promisc_mode, timeout, *open_status_str);
+ ws_debug("pcap_open_live() returned %p.", (void *)pcap_h);
+ if (pcap_h == NULL) {
+ *open_status = CAP_DEVICE_OPEN_ERROR_OTHER;
+ return NULL;
+ }
+ if ((*open_status_str)[0] != '\0') {
+ /*
+ * Warning. The call succeeded, but something happened
+ * that the user might want to know.
+ */
+ *open_status = CAP_DEVICE_OPEN_WARNING_OTHER;
+ } else {
+ /*
+ * No warning issued.
+ */
+ *open_status = CAP_DEVICE_OPEN_NO_ERR;
+ }
+
+#ifdef _WIN32
+ /* Try to set the capture buffer size. */
+ if (interface_opts->buffer_size > 1) {
+ /*
+ * We have no mechanism to report a warning if this
+ * fails; we just keep capturing with the smaller buffer,
+ * as is the case on systems with BPF and pcap_create()
+ * and pcap_set_buffer_size(), where pcap_activate() just
+ * silently clamps the buffer size to the maximum.
+ */
+ pcap_setbuff(pcap_h, interface_opts->buffer_size * 1024 * 1024);
+ }
+#endif
+
+ return pcap_h;
+}
+
+/*
+ * Get the capabilities of a network device.
+ */
+if_capabilities_t *
+get_if_capabilities(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str)
+{
+#if defined(HAVE_PCAP_OPEN) && defined(HAVE_PCAP_REMOTE)
+ if_capabilities_t *caps;
+ char errbuf[PCAP_ERRBUF_SIZE];
+ pcap_t *pch;
+ int deflt;
+ data_link_info_t *data_link_info;
+
+ if (strncmp (interface_opts->name, "rpcap://", 8) == 0) {
+ struct pcap_rmtauth auth;
+
+ auth.type = interface_opts->auth_type == CAPTURE_AUTH_PWD ?
+ RPCAP_RMTAUTH_PWD : RPCAP_RMTAUTH_NULL;
+ auth.username = interface_opts->auth_username;
+ auth.password = interface_opts->auth_password;
+
+ /*
+ * WinPcap 4.1.2, and possibly earlier versions, have a bug
+ * wherein, when an open with an rpcap: URL fails, the error
+ * message for the error is not copied to errbuf and whatever
+ * on-the-stack junk is in errbuf is treated as the error
+ * message.
+ *
+ * To work around that (and any other bugs of that sort), we
+ * initialize errbuf to an empty string. If we get an error
+ * and the string is empty, we report it as an unknown error.
+ * (If we *don't* get an error, and the string is *non*-empty,
+ * that could be a warning returned, such as "can't turn
+ * promiscuous mode on"; we currently don't do so.)
+ */
+ errbuf[0] = '\0';
+ pch = pcap_open(interface_opts->name, MIN_PACKET_SIZE, 0, 0, &auth,
+ errbuf);
+ if (pch == NULL) {
+ /*
+ * We don't know whether it's a permission error or not.
+ * And, if it is, the user will either have to ask for
+ * permission for their own remote account or will have
+ * to use an account that *does* have permissions.
+ */
+ *status = CAP_DEVICE_OPEN_ERROR_GENERIC;
+ if (strcmp(errbuf, "not supported") == 0) {
+ /*
+ * macOS 14's pcap_open(), which is a stub that
+ * always returns NULL with an error message of
+ * "not supported".
+ *
+ * In this case, as we passed it an rpcap://
+ * URL, treat that as meaning "remote capture
+ * not supported".
+ */
+ g_strlcpy(errbuf, "Remote capture not supported",
+ PCAP_ERRBUF_SIZE);
+ }
+ *status_str = g_strdup(errbuf[0] == '\0' ? "Unknown error (pcap bug; actual error cause not reported)" : errbuf);
+ return NULL;
+ }
+
+ caps = (if_capabilities_t *)g_malloc(sizeof *caps);
+ caps->can_set_rfmon = false;
+ caps->data_link_types = NULL;
+ deflt = get_pcap_datalink(pch, interface_opts->name);
+ data_link_info = create_data_link_info(deflt);
+ caps->data_link_types = g_list_append(caps->data_link_types, data_link_info);
+ caps->timestamp_types = get_pcap_timestamp_types(pch, NULL);
+ pcap_close(pch);
+
+ /*
+ * This doesn't return warnings for remote devices, and
+ * we don't use it for local devices.
+ */
+ *status = CAP_DEVICE_OPEN_NO_ERR;
+ *status_str = NULL;
+ return caps;
+ }
+#endif /* defined(HAVE_PCAP_OPEN) && defined(HAVE_PCAP_REMOTE) */
+
+ /*
+ * Local interface.
+ */
+ return get_if_capabilities_local(interface_opts, status, status_str);
+}
+
+pcap_t *
+open_capture_device(capture_options *capture_opts,
+ interface_options *interface_opts, int timeout,
+ cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE])
+{
+ pcap_t *pcap_h;
+#if defined(HAVE_PCAP_OPEN) && defined(HAVE_PCAP_REMOTE)
+ struct pcap_rmtauth auth;
+#endif
+
+ /* Open the network interface to capture from it.
+ Some versions of libpcap may put warnings into the error buffer
+ if they succeed; to tell if that's happened, we have to clear
+ the error buffer, and check if it's still a null string. */
+ ws_debug("Entering open_capture_device().");
+ *open_status = CAP_DEVICE_OPEN_NO_ERR;
+ (*open_status_str)[0] = '\0';
+#if defined(HAVE_PCAP_OPEN) && defined(HAVE_PCAP_REMOTE)
+ /*
+ * If we're opening a remote device, use pcap_open(); that's currently
+ * the only open routine that supports remote devices.
+ */
+ if (strncmp (interface_opts->name, "rpcap://", 8) == 0) {
+ int snaplen;
+
+ auth.type = interface_opts->auth_type == CAPTURE_AUTH_PWD ?
+ RPCAP_RMTAUTH_PWD : RPCAP_RMTAUTH_NULL;
+ auth.username = interface_opts->auth_username;
+ auth.password = interface_opts->auth_password;
+
+ if (interface_opts->has_snaplen)
+ snaplen = interface_opts->snaplen;
+ else {
+ /*
+ * Default - use the non-D-Bus maximum snapshot length,
+ * which should be big enough, except for D-Bus.
+ */
+ snaplen = 256*1024;
+ }
+ ws_debug("Calling pcap_open() using name %s, snaplen %d, promisc_mode %d, datatx_udp %d, nocap_rpcap %d.",
+ interface_opts->name, snaplen,
+ interface_opts->promisc_mode, interface_opts->datatx_udp,
+ interface_opts->nocap_rpcap);
+ pcap_h = pcap_open(interface_opts->name, snaplen,
+ /* flags */
+ (interface_opts->promisc_mode ? PCAP_OPENFLAG_PROMISCUOUS : 0) |
+ (interface_opts->datatx_udp ? PCAP_OPENFLAG_DATATX_UDP : 0) |
+ (interface_opts->nocap_rpcap ? PCAP_OPENFLAG_NOCAPTURE_RPCAP : 0),
+ timeout, &auth, *open_status_str);
+ if (pcap_h == NULL) {
+ /*
+ * Error.
+ *
+ * We don't know whether it's a permission error
+ * or not.
+ * (If it is, maybe we can give ourselves permission
+ * or maybe we just have to ask politely for
+ * permission.)
+ */
+ *open_status = CAP_DEVICE_OPEN_ERROR_GENERIC;
+ if (strcmp(*open_status_str, "not supported") == 0) {
+ /*
+ * macOS 14's pcap_open(), which is a stub
+ * that always returns NULL with an error
+ * message of "not supported".
+ *
+ * In this case, as we passed it an rpcap://
+ * URL, treat that as meaning "remote capture
+ * not supported".
+ */
+ g_strlcpy(*open_status_str,
+ "Remote capture not supported",
+ PCAP_ERRBUF_SIZE);
+ }
+
+ /* Did pcap actually supply an error message? */
+ if ((*open_status_str)[0] == '\0') {
+ /*
+ * Work around known WinPcap bug wherein
+ * no error message is filled in on a
+ * failure to open an rpcap: URL.
+ */
+ (void) g_strlcpy(*open_status_str,
+ "Unknown error (pcap bug; actual error cause not reported)",
+ sizeof *open_status_str);
+ }
+ }
+ ws_debug("pcap_open() returned %p.", (void *)pcap_h);
+ ws_debug("open_capture_device %s : %s", pcap_h ? "SUCCESS" : "FAILURE", interface_opts->name);
+ /*
+ * This doesn't return warnings for remote devices, and
+ * we don't use it for local devices.
+ */
+ *open_status = CAP_DEVICE_OPEN_NO_ERR;
+ return pcap_h;
+ }
+#endif
+
+ pcap_h = open_capture_device_local(capture_opts, interface_opts,
+ timeout, open_status, open_status_str);
+ ws_debug("open_capture_device %s : %s", pcap_h ? "SUCCESS" : "FAILURE", interface_opts->name);
+ return pcap_h;
+}
+
+#endif /* HAVE_LIBPCAP */
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 8
+ * tab-width: 8
+ * indent-tabs-mode: t
+ * End:
+ *
+ * vi: set shiftwidth=8 tabstop=8 noexpandtab:
+ * :indentSize=8:tabSize=8:noTabs=false:
+ */
diff --git a/capture/capture-pcap-util.h b/capture/capture-pcap-util.h
new file mode 100644
index 00000000..14a5e034
--- /dev/null
+++ b/capture/capture-pcap-util.h
@@ -0,0 +1,120 @@
+/** @file
+ *
+ * Utility definitions for packet capture
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef __CAPTURE_PCAP_UTIL_H__
+#define __CAPTURE_PCAP_UTIL_H__
+
+#include <wsutil/feature_list.h>
+
+#ifdef HAVE_LIBPCAP
+
+#include <pcap.h>
+
+#include "capture_opts.h"
+
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+#ifdef HAVE_LIBPCAP
+/*
+ * A snapshot length of 0 is useless - and libpcap/WinPcap/Npcap don't guarantee
+ * that a snapshot length of 0 will work, and, on some platforms, it won't
+ * (with BPF, for example, the kernel is told the snapshot length via the
+ * return value of the BPF program, and a return value of 0 means "drop
+ * the packet"), so the minimum packet size is 1 byte.
+ */
+#define MIN_PACKET_SIZE 1 /* minimum amount of packet data we can read */
+
+GList *get_interface_list(int *err, char **err_str);
+#ifdef HAVE_PCAP_REMOTE
+GList *get_remote_interface_list(const char *hostname, const char *port,
+ int auth_type, const char *username,
+ const char *passwd, int *err, char **err_str);
+#endif /* HAVE_PCAP_REMOTE */
+
+const char *linktype_val_to_name(int dlt);
+int linktype_name_to_val(const char *linktype);
+
+int get_pcap_datalink(pcap_t *pch, const char *devicename);
+
+bool set_pcap_datalink(pcap_t *pcap_h, int datalink, char *name,
+ char *errmsg, size_t errmsg_len,
+ char *secondary_errmsg, size_t secondary_errmsg_len);
+
+#ifdef HAVE_PCAP_SET_TSTAMP_PRECISION
+/*
+ * Return true if the pcap_t in question is set up for high-precision
+ * time stamps, false otherwise.
+ */
+bool have_high_resolution_timestamp(pcap_t *pcap_h);
+#endif /* HAVE_PCAP_SET_TSTAMP_PRECISION */
+
+/*
+ * Capture device open status values.
+ */
+typedef enum {
+ /* No error and no warning */
+ CAP_DEVICE_OPEN_NO_ERR,
+
+ /* Errors corresponding to libpcap errors */
+ CAP_DEVICE_OPEN_ERROR_NO_SUCH_DEVICE,
+ CAP_DEVICE_OPEN_ERROR_RFMON_NOTSUP,
+ CAP_DEVICE_OPEN_ERROR_PERM_DENIED,
+ CAP_DEVICE_OPEN_ERROR_IFACE_NOT_UP,
+ CAP_DEVICE_OPEN_ERROR_PROMISC_PERM_DENIED,
+
+ /* Error, none of the above */
+ CAP_DEVICE_OPEN_ERROR_OTHER,
+
+ /* Error from pcap_open_live() or pcap_open() rather than pcap_activate() */
+ CAP_DEVICE_OPEN_ERROR_GENERIC,
+
+ /* Warnings corresponding to libpcap warnings */
+ CAP_DEVICE_OPEN_WARNING_PROMISC_NOTSUP,
+ CAP_DEVICE_OPEN_WARNING_TSTAMP_TYPE_NOTSUP,
+
+ /* Warning, none of the above */
+ CAP_DEVICE_OPEN_WARNING_OTHER
+} cap_device_open_status;
+extern if_capabilities_t *get_if_capabilities(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str);
+extern pcap_t *open_capture_device(capture_options *capture_opts,
+ interface_options *interface_opts,
+ int timeout, cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE]);
+
+#endif /* HAVE_LIBPCAP */
+
+extern void gather_caplibs_compile_info(feature_list l);
+
+/*
+ * Append to a GString an indication of the version of capture libraries
+ * with which we're running, or an indication that we're not running
+ * with capture libraries, if we were compiled with WinPcap or Npcap but
+ * WinPcap/Npcap wasn't loaded, or nothing, if we weren't compiled with
+ * libpcap/WinPcap/Npcap.
+ */
+extern void gather_caplibs_runtime_info(feature_list l);
+
+#ifdef _WIN32
+extern bool caplibs_have_npcap(void);
+extern bool caplibs_get_npcap_version(unsigned int *major,
+ unsigned int *minor);
+#endif
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* __CAPTURE_PCAP_UTIL_H__ */
diff --git a/capture/capture-wpcap.c b/capture/capture-wpcap.c
new file mode 100644
index 00000000..4f7d1dca
--- /dev/null
+++ b/capture/capture-wpcap.c
@@ -0,0 +1,967 @@
+/* capture-wpcap.c
+ * WinPcap/Npcap-specific interfaces for capturing. We load WinPcap/Npcap
+ * at run time, so that we only need one Wireshark binary and one TShark
+ * binary for Windows, regardless of whether WinPcap/Npcap is installed
+ * or not.
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 2001 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+
+#include <wireshark.h>
+
+#include <windows.h>
+#include <wchar.h>
+#include <tchar.h>
+
+#include <stdio.h>
+
+#include <ws_attributes.h>
+
+#include "capture/capture-wpcap.h"
+#include <wsutil/feature_list.h>
+
+bool has_wpcap = false;
+
+#ifdef HAVE_LIBPCAP
+
+#include <gmodule.h>
+
+#include <epan/strutil.h>
+
+#include "capture/capture_ifinfo.h"
+#include "capture/capture-pcap-util.h"
+#include "capture/capture-pcap-util-int.h"
+
+#include <wsutil/file_util.h>
+#include <wsutil/strtoi.h>
+#include <wsutil/ws_assert.h>
+
+#define MAX_WIN_IF_NAME_LEN 511
+
+static void (*p_pcap_close) (pcap_t *);
+static int (*p_pcap_stats) (pcap_t *, struct pcap_stat *);
+static int (*p_pcap_dispatch) (pcap_t *, int, pcap_handler, unsigned char *);
+static int (*p_pcap_snapshot) (pcap_t *);
+static int (*p_pcap_datalink) (pcap_t *);
+static int (*p_pcap_setfilter) (pcap_t *, struct bpf_program *);
+static char* (*p_pcap_geterr) (pcap_t *);
+static int (*p_pcap_compile) (pcap_t *, struct bpf_program *, const char *, int,
+ bpf_u_int32);
+static int (*p_pcap_compile_nopcap) (int, int, struct bpf_program *, const char *, int,
+ bpf_u_int32);
+static int (*p_pcap_lookupnet) (const char *, bpf_u_int32 *, bpf_u_int32 *,
+ char *);
+static pcap_t* (*p_pcap_open_live) (const char *, int, int, int, char *);
+static int (*p_pcap_loop) (pcap_t *, int, pcap_handler, unsigned char *);
+static pcap_t* (*p_pcap_open_dead) (int, int);
+static void (*p_pcap_freecode) (struct bpf_program *);
+static int (*p_pcap_findalldevs) (pcap_if_t **, char *);
+static void (*p_pcap_freealldevs) (pcap_if_t *);
+static int (*p_pcap_datalink_name_to_val) (const char *);
+static const char *(*p_pcap_datalink_val_to_name) (int);
+static const char *(*p_pcap_datalink_val_to_description) (int);
+static void (*p_pcap_breakloop) (pcap_t *);
+static const char *(*p_pcap_lib_version) (void);
+static int (*p_pcap_setbuff) (pcap_t *, int dim);
+static int (*p_pcap_next_ex) (pcap_t *, struct pcap_pkthdr **pkt_header, const u_char **pkt_data);
+#ifdef HAVE_PCAP_REMOTE
+static pcap_t* (*p_pcap_open) (const char *, int, int, int,
+ struct pcap_rmtauth *, char *);
+static int (*p_pcap_findalldevs_ex) (const char *, struct pcap_rmtauth *,
+ pcap_if_t **, char *);
+static int (*p_pcap_createsrcstr) (char *, int, const char *, const char *,
+ const char *, char *);
+#endif
+#ifdef HAVE_PCAP_SETSAMPLING
+static struct pcap_samp* (*p_pcap_setsampling)(pcap_t *);
+#endif
+
+static int (*p_pcap_list_datalinks)(pcap_t *, int **);
+static int (*p_pcap_set_datalink)(pcap_t *, int);
+
+#ifdef HAVE_PCAP_FREE_DATALINKS
+static int (*p_pcap_free_datalinks)(int *);
+#endif
+
+static char *(*p_bpf_image)(const struct bpf_insn *, int);
+
+#ifdef HAVE_PCAP_CREATE
+static pcap_t *(*p_pcap_create)(const char *, char *);
+static int (*p_pcap_set_snaplen)(pcap_t *, int);
+static int (*p_pcap_set_promisc)(pcap_t *, int);
+static int (*p_pcap_can_set_rfmon)(pcap_t *);
+static int (*p_pcap_set_rfmon)(pcap_t *, int);
+static int (*p_pcap_set_timeout)(pcap_t *, int);
+static int (*p_pcap_set_buffer_size)(pcap_t *, int);
+static int (*p_pcap_activate)(pcap_t *);
+static const char *(*p_pcap_statustostr)(int);
+#endif
+
+#ifdef HAVE_PCAP_SET_TSTAMP_TYPE
+static int (*p_pcap_set_tstamp_type)(pcap_t *, int);
+static int (*p_pcap_set_tstamp_precision)(pcap_t *, int);
+static int (*p_pcap_get_tstamp_precision)(pcap_t *);
+static int (*p_pcap_list_tstamp_types)(pcap_t *, int **);
+static void (*p_pcap_free_tstamp_types)(int *);
+static int (*p_pcap_tstamp_type_name_to_val)(const char *);
+static const char * (*p_pcap_tstamp_type_val_to_name)(int);
+static const char * (*p_pcap_tstamp_type_val_to_description)(int);
+#endif
+
+typedef struct {
+ const char *name;
+ void * *ptr;
+ bool optional;
+} symbol_table_t;
+
+#define SYM(x, y) { G_STRINGIFY(x) , (void *) &G_PASTE(p_,x), y }
+
+void
+load_wpcap(void)
+{
+
+ /* These are the symbols I need or want from Wpcap */
+ static const symbol_table_t symbols[] = {
+ SYM(pcap_close, false),
+ SYM(pcap_stats, false),
+ SYM(pcap_dispatch, false),
+ SYM(pcap_snapshot, false),
+ SYM(pcap_datalink, false),
+ SYM(pcap_setfilter, false),
+ SYM(pcap_geterr, false),
+ SYM(pcap_compile, false),
+ SYM(pcap_compile_nopcap, false),
+ SYM(pcap_lookupnet, false),
+#ifdef HAVE_PCAP_REMOTE
+ SYM(pcap_open, false),
+ SYM(pcap_findalldevs_ex, false),
+ SYM(pcap_createsrcstr, false),
+#endif
+ SYM(pcap_open_live, false),
+ SYM(pcap_open_dead, false),
+#ifdef HAVE_PCAP_SETSAMPLING
+ SYM(pcap_setsampling, true),
+#endif
+ SYM(pcap_loop, false),
+ SYM(pcap_freecode, false),
+ SYM(pcap_findalldevs, false),
+ SYM(pcap_freealldevs, false),
+ SYM(pcap_datalink_name_to_val, false),
+ SYM(pcap_datalink_val_to_name, false),
+ SYM(pcap_datalink_val_to_description, false),
+ SYM(pcap_breakloop, false),
+ SYM(pcap_lib_version, false),
+ SYM(pcap_setbuff, true),
+ SYM(pcap_next_ex, true),
+ SYM(pcap_list_datalinks, false),
+ SYM(pcap_set_datalink, false),
+#ifdef HAVE_PCAP_FREE_DATALINKS
+ SYM(pcap_free_datalinks, true),
+#endif
+ SYM(bpf_image, false),
+#ifdef HAVE_PCAP_CREATE
+ SYM(pcap_create, true),
+ SYM(pcap_set_snaplen, true),
+ SYM(pcap_set_promisc, true),
+ SYM(pcap_can_set_rfmon, true),
+ SYM(pcap_set_rfmon, true),
+ SYM(pcap_set_timeout, false),
+ SYM(pcap_set_buffer_size, false),
+ SYM(pcap_activate, true),
+ SYM(pcap_statustostr, true),
+#endif
+#ifdef HAVE_PCAP_SET_TSTAMP_TYPE
+ SYM(pcap_set_tstamp_type, true),
+ SYM(pcap_set_tstamp_precision, true),
+ SYM(pcap_get_tstamp_precision, true),
+ SYM(pcap_list_tstamp_types, true),
+ SYM(pcap_free_tstamp_types, true),
+ SYM(pcap_tstamp_type_name_to_val, true),
+ SYM(pcap_tstamp_type_val_to_name, true),
+ SYM(pcap_tstamp_type_val_to_description, true),
+#endif
+ { NULL, NULL, false }
+ };
+
+ GModule *wh; /* wpcap handle */
+ const symbol_table_t *sym;
+
+ wh = load_wpcap_module();
+
+ if (!wh) {
+ return;
+ }
+
+ sym = symbols;
+ while (sym->name) {
+ if (!g_module_symbol(wh, sym->name, sym->ptr)) {
+ if (sym->optional) {
+ /*
+ * We don't care if it's missing; we just
+ * don't use it.
+ */
+ *sym->ptr = NULL;
+ } else {
+ /*
+ * We require this symbol.
+ */
+ return;
+ }
+ }
+ sym++;
+ }
+
+
+ has_wpcap = true;
+}
+
+bool
+caplibs_have_npcap(void)
+{
+ return has_wpcap && g_str_has_prefix(p_pcap_lib_version(), "Npcap");
+}
+
+bool
+caplibs_get_npcap_version(unsigned int *major, unsigned int *minor)
+{
+ const char *version;
+ static const char prefix[] = "Npcap version ";
+
+ if (!has_wpcap)
+ return false; /* we don't have any pcap */
+
+ version = p_pcap_lib_version();
+ if (!g_str_has_prefix(version, prefix))
+ return false; /* we have it, but it's not Npcap */
+
+ /*
+ * This is Npcap; return the major and minor version numbers.
+ * First, skip pas the "Npcap version " prefix.
+ */
+ const char *major_version_number;
+ const char *minor_version_number;
+ const char *p;
+
+ /*
+ * Get the major version number.
+ */
+ major_version_number = version + sizeof prefix - 1;
+ if (!ws_strtou(major_version_number, &p, major))
+ return false; /* not a number */
+ if (*p != '.')
+ return false; /* not followed by a "." */
+ p++; /* skip over the '.' */
+
+ /*
+ * Get the minor version number.
+ */
+ minor_version_number = p;
+ if (!ws_strtou(minor_version_number, &p, minor))
+ return false; /* not a number */
+ if (*p != ',' && *p != '.' && *p != '\0') {
+ /*
+ * Not followed by a comma (to separate from "based on
+ * libpcap ..."), not followed by a period (in case Npcap
+ * ever has a dot-dot release), and not followed by a
+ * '\0' (in case it has only the Npcap version number).
+ */
+ return false;
+ }
+ return true;
+}
+
+static char *
+local_code_page_str_to_utf8(char *str)
+{
+ ULONG utf16_len;
+ wchar_t *utf16_str;
+ char *utf8_str;
+
+ if (str == NULL) {
+ return NULL;
+ }
+
+ utf16_len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
+ utf16_str = g_malloc_n(utf16_len, sizeof(wchar_t));
+ MultiByteToWideChar(CP_ACP, 0, str, -1, utf16_str, utf16_len);
+
+ utf8_str = g_utf16_to_utf8(utf16_str, -1, NULL, NULL, NULL);
+
+ g_free(utf16_str);
+ return utf8_str;
+}
+
+static void
+prepare_errbuf(char *errbuf)
+{
+ ws_assert(errbuf);
+ errbuf[0] = '\0';
+}
+
+static void
+convert_errbuf_to_utf8(char *errbuf)
+{
+ char *utf8_err;
+ if (errbuf[0] == '\0') {
+ return;
+ }
+ errbuf[PCAP_ERRBUF_SIZE - 1] = '\0';
+ utf8_err = local_code_page_str_to_utf8(errbuf);
+ snprintf(errbuf, PCAP_ERRBUF_SIZE, "%s", utf8_err);
+ g_free(utf8_err);
+}
+
+static char *
+cant_load_winpcap_err(const char *app_name)
+{
+ return ws_strdup_printf(
+"Unable to load Npcap or WinPcap (wpcap.dll); %s will not be able to\n"
+"capture packets.\n"
+"\n"
+"In order to capture packets Npcap or WinPcap must be installed. See\n"
+"\n"
+" https://npcap.com/\n"
+"\n"
+"for a downloadable version of Npcap and for instructions on how to\n"
+"install it.",
+ app_name);
+}
+
+void
+pcap_close(pcap_t *a)
+{
+ ws_assert(has_wpcap);
+ p_pcap_close(a);
+}
+
+int
+pcap_stats(pcap_t *a, struct pcap_stat *b)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_stats(a, b);
+}
+
+int
+pcap_dispatch(pcap_t *a, int b, pcap_handler c, unsigned char *d)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_dispatch(a, b, c, d);
+}
+
+int
+pcap_snapshot(pcap_t *a)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_snapshot(a);
+}
+
+int
+pcap_datalink(pcap_t *a)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_datalink(a);
+}
+
+int
+pcap_set_datalink(pcap_t *p, int dlt)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_set_datalink(p, dlt);
+}
+
+int
+pcap_setfilter(pcap_t *a, struct bpf_program *b)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_setfilter(a, b);
+}
+
+char*
+pcap_geterr(pcap_t *a)
+{
+ char *errbuf;
+ ws_assert(has_wpcap);
+ errbuf = p_pcap_geterr(a);
+ convert_errbuf_to_utf8(errbuf);
+ return errbuf;
+}
+
+int
+pcap_compile(pcap_t *a, struct bpf_program *b, const char *c, int d,
+ bpf_u_int32 e)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_compile(a, b, c, d, e);
+}
+
+int
+pcap_compile_nopcap(int a, int b, struct bpf_program *c, const char *d, int e,
+ bpf_u_int32 f)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_compile_nopcap(a, b, c, d, e, f);
+}
+
+int
+pcap_lookupnet(const char *a, bpf_u_int32 *b, bpf_u_int32 *c, char *errbuf)
+{
+ int ret;
+ ws_assert(has_wpcap);
+ ret = p_pcap_lookupnet(a, b, c, errbuf);
+ if (ret == -1)
+ convert_errbuf_to_utf8(errbuf);
+ return ret;
+}
+
+pcap_t*
+pcap_open_live(const char *a, int b, int c, int d, char *errbuf)
+{
+ pcap_t *p;
+ if (!has_wpcap) {
+ snprintf(errbuf, PCAP_ERRBUF_SIZE,
+ "unable to load Npcap or WinPcap (wpcap.dll); can't open %s to capture",
+ a);
+ return NULL;
+ }
+ prepare_errbuf(errbuf);
+ p = p_pcap_open_live(a, b, c, d, errbuf);
+ convert_errbuf_to_utf8(errbuf);
+ return p;
+}
+
+pcap_t*
+pcap_open_dead(int a, int b)
+{
+ if (!has_wpcap) {
+ return NULL;
+ }
+ return p_pcap_open_dead(a, b);
+}
+
+char *
+bpf_image(const struct bpf_insn *a, int b)
+{
+ if (!has_wpcap) {
+ return NULL;
+ }
+ return p_bpf_image(a, b);
+}
+
+#ifdef HAVE_PCAP_REMOTE
+pcap_t*
+pcap_open(const char *a, int b, int c, int d, struct pcap_rmtauth *e, char *errbuf)
+{
+ pcap_t *ret;
+ if (!has_wpcap) {
+ snprintf(errbuf, PCAP_ERRBUF_SIZE,
+ "unable to load Npcap or WinPcap (wpcap.dll); can't open %s to capture",
+ a);
+ return NULL;
+ }
+ prepare_errbuf(errbuf);
+ ret = p_pcap_open(a, b, c, d, e, errbuf);
+ convert_errbuf_to_utf8(errbuf);
+ return ret;
+}
+
+int
+ws_pcap_findalldevs_ex(const char *a, struct pcap_rmtauth *b, pcap_if_t **c, char *errbuf)
+{
+ int ret;
+ ws_assert(has_wpcap);
+ ret = p_pcap_findalldevs_ex(a, b, c, errbuf);
+ if (ret == -1)
+ convert_errbuf_to_utf8(errbuf);
+ return ret;
+}
+
+int
+pcap_createsrcstr(char *a, int b, const char *c, const char *d, const char *e,
+ char *errbuf)
+{
+ int ret;
+ ws_assert(has_wpcap);
+ ret = p_pcap_createsrcstr(a, b, c, d, e, errbuf);
+ if (ret == -1)
+ convert_errbuf_to_utf8(errbuf);
+ return ret;
+}
+#endif
+
+#ifdef HAVE_PCAP_SETSAMPLING
+struct pcap_samp *
+pcap_setsampling(pcap_t *a)
+{
+ ws_assert(has_wpcap);
+ if (p_pcap_setsampling != NULL) {
+ return p_pcap_setsampling(a);
+ }
+ return NULL;
+}
+#endif
+
+int
+pcap_loop(pcap_t *a, int b, pcap_handler c, unsigned char *d)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_loop(a, b, c, d);
+}
+
+void
+pcap_freecode(struct bpf_program *a)
+{
+ ws_assert(has_wpcap);
+ p_pcap_freecode(a);
+}
+
+int
+pcap_findalldevs(pcap_if_t **a, char *errbuf)
+{
+ int ret;
+ ws_assert(has_wpcap);
+ ret = p_pcap_findalldevs(a, errbuf);
+ if (ret == -1)
+ convert_errbuf_to_utf8(errbuf);
+ return ret;
+}
+
+void
+pcap_freealldevs(pcap_if_t *a)
+{
+ ws_assert(has_wpcap);
+ p_pcap_freealldevs(a);
+}
+
+#ifdef HAVE_PCAP_CREATE
+pcap_t *
+pcap_create(const char *a, char *errbuf)
+{
+ pcap_t *p;
+ ws_assert(has_wpcap && p_pcap_create != NULL);
+ p = p_pcap_create(a, errbuf);
+ if (p == NULL)
+ convert_errbuf_to_utf8(errbuf);
+ return p;
+}
+
+int
+pcap_set_snaplen(pcap_t *a, int b)
+{
+ ws_assert(has_wpcap && p_pcap_set_snaplen != NULL);
+ return p_pcap_set_snaplen(a, b);
+}
+
+int
+pcap_set_promisc(pcap_t *a, int b)
+{
+ ws_assert(has_wpcap && p_pcap_set_promisc != NULL);
+ return p_pcap_set_promisc(a, b);
+}
+
+int
+pcap_can_set_rfmon(pcap_t *a)
+{
+ ws_assert(has_wpcap);
+ if (p_pcap_can_set_rfmon != NULL) {
+ return p_pcap_can_set_rfmon(a);
+ }
+ return 0;
+}
+
+int
+pcap_set_rfmon(pcap_t *a, int b)
+{
+ ws_assert(has_wpcap && p_pcap_set_rfmon != NULL);
+ return p_pcap_set_rfmon(a, b);
+}
+
+int
+pcap_set_timeout(pcap_t *a, int b)
+{
+ ws_assert(has_wpcap && p_pcap_set_timeout != NULL);
+ return p_pcap_set_timeout(a, b);
+}
+int
+pcap_set_buffer_size(pcap_t *a, int b)
+{
+ ws_assert(has_wpcap && p_pcap_set_buffer_size != NULL);
+ return p_pcap_set_buffer_size(a, b);
+}
+
+int
+pcap_activate(pcap_t *a)
+{
+ ws_assert(has_wpcap && p_pcap_activate != NULL);
+ return p_pcap_activate(a);
+
+}
+
+const char *
+pcap_statustostr(int a)
+{
+ static char ebuf[15 + 10 + 1];
+
+ ws_assert(has_wpcap);
+ if (p_pcap_statustostr != NULL) {
+ return p_pcap_statustostr(a);
+ }
+
+ /* XXX copy routine from pcap.c ??? */
+ (void)snprintf(ebuf, sizeof ebuf, "Don't have pcap_statustostr(), can't translate error: %d", a);
+ return(ebuf);
+
+}
+#endif
+
+#ifdef HAVE_PCAP_SET_TSTAMP_TYPE
+int
+pcap_set_tstamp_type(pcap_t *a, int b) {
+ ws_assert(has_wpcap);
+ if (p_pcap_set_tstamp_type != NULL) {
+ return p_pcap_set_tstamp_type(a, b);
+ }
+ return PCAP_ERROR_CANTSET_TSTAMP_TYPE;
+}
+
+int
+pcap_set_tstamp_precision(pcap_t *a, int b) {
+ ws_assert(has_wpcap);
+ if (p_pcap_set_tstamp_precision != NULL) {
+ return p_pcap_set_tstamp_precision(a, b);
+ }
+ // No error code defined so return NOTSUP.
+ return PCAP_ERROR_TSTAMP_PRECISION_NOTSUP;
+}
+
+int
+pcap_get_tstamp_precision(pcap_t *a) {
+ ws_assert(has_wpcap);
+ if (p_pcap_get_tstamp_precision != NULL) {
+ return p_pcap_get_tstamp_precision(a);
+ }
+ // No error code defined so return MICRO.
+ return PCAP_TSTAMP_PRECISION_MICRO;
+}
+
+int
+pcap_list_tstamp_types(pcap_t *a, int **b) {
+ ws_assert(has_wpcap);
+ if (p_pcap_list_tstamp_types != NULL) {
+ return p_pcap_list_tstamp_types(a, b);
+ }
+ return PCAP_ERROR;
+}
+
+void
+pcap_free_tstamp_types(int *a) {
+ ws_assert(has_wpcap);
+ if (p_pcap_free_tstamp_types != NULL) {
+ p_pcap_free_tstamp_types(a);
+ }
+}
+
+int
+pcap_tstamp_type_name_to_val(const char *a) {
+ ws_assert(has_wpcap);
+ if (p_pcap_tstamp_type_name_to_val != NULL) {
+ return p_pcap_tstamp_type_name_to_val(a);
+ }
+ return PCAP_ERROR;
+}
+
+const char *
+pcap_tstamp_type_val_to_name(int a) {
+ ws_assert(has_wpcap);
+ if (p_pcap_tstamp_type_val_to_name != NULL) {
+ return p_pcap_tstamp_type_val_to_name(a);
+ }
+ return NULL;
+}
+
+const char *
+pcap_tstamp_type_val_to_description(int a) {
+ ws_assert(has_wpcap);
+ if (p_pcap_tstamp_type_val_to_description != NULL) {
+ return p_pcap_tstamp_type_val_to_description(a);
+ }
+ return NULL;
+}
+#endif
+
+int
+pcap_datalink_name_to_val(const char *name)
+{
+ if (has_wpcap)
+ return p_pcap_datalink_name_to_val(name);
+ else
+ return -1;
+}
+
+int
+pcap_list_datalinks(pcap_t *p, int **ddlt)
+{
+ if (has_wpcap)
+ return p_pcap_list_datalinks(p, ddlt);
+ else
+ return -1;
+}
+
+#ifdef HAVE_PCAP_FREE_DATALINKS
+void
+pcap_free_datalinks(int *ddlt)
+{
+ ws_assert(has_wpcap);
+
+ /*
+ * If we don't have pcap_free_datalinks() in WinPcap,
+ * we don't free the memory - we can't use free(), as
+ * we might not have been built with the same version
+ * of the C runtime library as WinPcap was, and, if we're
+ * not, free() isn't guaranteed to work on something
+ * allocated by WinPcap.
+ */
+ if (p_pcap_free_datalinks != NULL)
+ p_pcap_free_datalinks(ddlt);
+}
+#endif
+
+const char *
+pcap_datalink_val_to_name(int dlt)
+{
+ if (has_wpcap)
+ return p_pcap_datalink_val_to_name(dlt);
+ else
+ return NULL;
+}
+
+const char *
+pcap_datalink_val_to_description(int dlt)
+{
+ if (has_wpcap)
+ return p_pcap_datalink_val_to_description(dlt);
+ return NULL;
+}
+
+void pcap_breakloop(pcap_t *a)
+{
+ p_pcap_breakloop(a);
+}
+
+/* setbuff is win32 specific! */
+int pcap_setbuff(pcap_t *a, int b)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_setbuff(a, b);
+}
+
+int pcap_next_ex(pcap_t *a, struct pcap_pkthdr **b, const u_char **c)
+{
+ ws_assert(has_wpcap);
+ return p_pcap_next_ex(a, b, c);
+}
+
+#ifdef HAVE_PCAP_REMOTE
+GList *
+get_remote_interface_list(const char *hostname, const char *port,
+ int auth_type, const char *username,
+ const char *passwd, int *err, char **err_str)
+{
+ if (!has_wpcap) {
+ /*
+ * We don't have Npcap or WinPcap, so we can't get a list of
+ * interfaces.
+ */
+ *err = DONT_HAVE_PCAP;
+ if (err_str != NULL)
+ *err_str = cant_load_winpcap_err("you");
+ return NULL;
+ }
+
+ return get_interface_list_findalldevs_ex(hostname, port, auth_type,
+ username, passwd, err, err_str);
+}
+#endif
+
+GList *
+get_interface_list(int *err, char **err_str)
+{
+ if (!has_wpcap) {
+ /*
+ * We don't have Npcap or WinPcap, so we can't get a list of
+ * interfaces.
+ */
+ *err = DONT_HAVE_PCAP;
+ if (err_str != NULL)
+ *err_str = cant_load_winpcap_err("you");
+ return NULL;
+ }
+
+ return get_interface_list_findalldevs(err, err_str);
+}
+
+/*
+ * Get an error message string for a CANT_GET_INTERFACE_LIST error from
+ * "get_interface_list()".
+ */
+char *
+cant_get_if_list_error_message(const char *err_str)
+{
+ /*
+ * If the error message includes "Not enough storage is available
+ * to process this command" or "The operation completed successfully",
+ * suggest that they install a WinPcap version later than 3.0.
+ */
+ if (strstr(err_str, "Not enough storage is available to process this command") != NULL ||
+ strstr(err_str, "The operation completed successfully") != NULL) {
+ return ws_strdup_printf("Can't get list of interfaces: %s\n"
+"This might be a problem with WinPcap 3.0. You should try updating to\n"
+"Npcap. See https://npcap.com/ for more information.",
+ err_str);
+ }
+ return ws_strdup_printf("Can't get list of interfaces: %s", err_str);
+}
+
+if_capabilities_t *
+get_if_capabilities_local(interface_options *interface_opts,
+ cap_device_open_status *status, char **status_str)
+{
+ /*
+ * We're not getting capaibilities for a remote device; use
+ * pcap_create() and pcap_activate() if we have them, so that
+ * we can set various options, otherwise use pcap_open_live().
+ */
+#ifdef HAVE_PCAP_CREATE
+ if (p_pcap_create != NULL)
+ return get_if_capabilities_pcap_create(interface_opts, status,
+ status_str);
+#endif
+ return get_if_capabilities_pcap_open_live(interface_opts, status,
+ status_str);
+}
+
+pcap_t *
+open_capture_device_local(capture_options *capture_opts,
+ interface_options *interface_opts, int timeout,
+ cap_device_open_status *open_status,
+ char (*open_status_str)[PCAP_ERRBUF_SIZE])
+{
+ /*
+ * We're not opening a remote device; use pcap_create() and
+ * pcap_activate() if we have them, so that we can set various
+ * options, otherwise use pcap_open_live().
+ */
+#ifdef HAVE_PCAP_CREATE
+ if (p_pcap_create != NULL)
+ return open_capture_device_pcap_create(capture_opts,
+ interface_opts, timeout, open_status, open_status_str);
+#endif
+ return open_capture_device_pcap_open_live(interface_opts, timeout,
+ open_status, open_status_str);
+}
+
+/*
+ * Append the WinPcap or Npcap SDK version with which we were compiled to a GString.
+ */
+void
+gather_caplibs_compile_info(feature_list l)
+{
+ with_feature(l, "libpcap");
+}
+
+void
+gather_caplibs_runtime_info(feature_list l)
+{
+ /*
+ * On Windows, we might have been compiled with WinPcap/Npcap but
+ * might not have it loaded; indicate whether we have it or
+ * not and, if we have it, what version we have.
+ */
+ if (has_wpcap) {
+ with_feature(l, "%s", p_pcap_lib_version());
+ } else
+ without_feature(l, "Npcap or WinPcap");
+}
+
+/*
+ * If npf.sys is running, return true.
+ */
+bool
+npf_sys_is_running(void)
+{
+ SC_HANDLE h_scm, h_serv;
+ SERVICE_STATUS ss;
+
+ h_scm = OpenSCManager(NULL, NULL, 0);
+ if (!h_scm)
+ return false;
+
+ h_serv = OpenService(h_scm, _T("npcap"), SC_MANAGER_CONNECT|SERVICE_QUERY_STATUS);
+ if (!h_serv) {
+ h_serv = OpenService(h_scm, _T("npf"), SC_MANAGER_CONNECT|SERVICE_QUERY_STATUS);
+ if (!h_serv) {
+ CloseServiceHandle(h_scm);
+ return false;
+ }
+ }
+
+ if (QueryServiceStatus(h_serv, &ss)) {
+ if (ss.dwCurrentState & SERVICE_RUNNING) {
+ CloseServiceHandle(h_serv);
+ CloseServiceHandle(h_scm);
+ return true;
+ }
+ }
+ CloseServiceHandle(h_serv);
+ CloseServiceHandle(h_scm);
+ return false;
+}
+
+#else /* HAVE_LIBPCAP */
+
+void
+load_wpcap(void)
+{
+ return;
+}
+
+/*
+ * Append an indication that we were not compiled with WinPcap
+ * to a GString.
+ */
+void
+gather_caplibs_compile_info(feature_list l)
+{
+ without_feature(l, "libpcap");
+}
+
+void
+gather_caplibs_runtime_info(feature_list l _U_)
+{
+}
+
+bool
+caplibs_have_npcap(void)
+{
+ return false;
+}
+
+#endif /* HAVE_LIBPCAP */
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 8
+ * tab-width: 8
+ * indent-tabs-mode: t
+ * End:
+ *
+ * vi: set shiftwidth=8 tabstop=8 noexpandtab:
+ * :indentSize=8:tabSize=8:noTabs=false:
+ */
diff --git a/capture/capture-wpcap.h b/capture/capture-wpcap.h
new file mode 100644
index 00000000..f3397821
--- /dev/null
+++ b/capture/capture-wpcap.h
@@ -0,0 +1,44 @@
+/** @file
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 2001 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef CAPTURE_WPCAP_H
+#define CAPTURE_WPCAP_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+#ifdef HAVE_LIBPCAP
+#ifdef __MINGW32__
+#include <_bsd_types.h>
+#endif
+#include <pcap.h>
+#endif
+
+extern bool has_wpcap;
+
+extern void load_wpcap(void);
+
+/**
+ * Check to see if npf.sys is running.
+ * @return true if npf.sys is running, false if it's not or if there was
+ * an error checking its status.
+ */
+bool npf_sys_is_running(void);
+
+#ifdef HAVE_LIBPCAP
+int
+ws_pcap_findalldevs_ex(const char *a, struct pcap_rmtauth *b, pcap_if_t **c, char *errbuf);
+#endif
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* CAPTURE_WPCAP_H */
diff --git a/capture/capture_ifinfo.c b/capture/capture_ifinfo.c
new file mode 100644
index 00000000..45201b88
--- /dev/null
+++ b/capture/capture_ifinfo.c
@@ -0,0 +1,359 @@
+/* capture_ifinfo.c
+ * Routines for getting interface information from dumpcap
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+#define WS_LOG_DOMAIN LOG_DOMAIN_CAPTURE
+
+#ifdef HAVE_LIBPCAP
+
+#include <wireshark.h>
+
+#include <stdlib.h>
+#include <stdio.h>
+
+#include "capture_opts.h"
+
+#include "capture/capture_session.h"
+#include "capture/capture_sync.h"
+#include "extcap.h"
+
+#include <capture/capture_ifinfo.h>
+#include <wsutil/inet_addr.h>
+
+#ifdef HAVE_PCAP_REMOTE
+static GList *remote_interface_list = NULL;
+
+static GList * append_remote_list(GList *iflist)
+{
+ GSList *list;
+ GList *rlist;
+ if_addr_t *if_addr, *temp_addr;
+ if_info_t *if_info, *temp;
+
+ for (rlist = g_list_nth(remote_interface_list, 0); rlist != NULL; rlist = g_list_next(rlist)) {
+ if_info = (if_info_t *)rlist->data;
+ temp = g_new0(if_info_t, 1);
+ temp->name = g_strdup(if_info->name);
+ temp->friendly_name = g_strdup(if_info->friendly_name);
+ temp->vendor_description = g_strdup(if_info->vendor_description);
+ for (list = g_slist_nth(if_info->addrs, 0); list != NULL; list = g_slist_next(list)) {
+ temp_addr = g_new0(if_addr_t, 1);
+ if_addr = (if_addr_t *)list->data;
+ if (if_addr) {
+ temp_addr->ifat_type = if_addr->ifat_type;
+ if (temp_addr->ifat_type == IF_AT_IPv4) {
+ temp_addr->addr.ip4_addr = if_addr->addr.ip4_addr;
+ } else {
+ memcpy(temp_addr->addr.ip6_addr, if_addr->addr.ip6_addr, sizeof(if_addr->addr));
+ }
+ } else {
+ g_free(temp_addr);
+ temp_addr = NULL;
+ }
+ if (temp_addr) {
+ temp->addrs = g_slist_append(temp->addrs, temp_addr);
+ }
+ }
+ temp->loopback = if_info->loopback;
+ iflist = g_list_append(iflist, temp);
+ }
+ return iflist;
+}
+#endif
+
+/**
+ * Fetch the interface list from a child process (dumpcap).
+ *
+ * @return A GList containing if_info_t structs if successful, NULL (with err and possibly err_str set) otherwise.
+ *
+ */
+
+/* XXX - We parse simple text output to get our interface list. Should
+ * we use "real" data serialization instead, e.g. via XML? */
+GList *
+capture_interface_list(int *err, char **err_str, void (*update_cb)(void))
+{
+ int ret;
+ GList *if_list = NULL;
+ int i, j;
+ char *data, *primary_msg, *secondary_msg;
+ char **raw_list, **if_parts, **addr_parts;
+ char *name;
+ if_info_t *if_info;
+ if_addr_t *if_addr;
+
+ *err = 0;
+ if (err_str) {
+ *err_str = NULL;
+ }
+
+ /* Try to get the local interface list */
+ ret = sync_interface_list_open(&data, &primary_msg, &secondary_msg, update_cb);
+ if (ret != 0) {
+ ws_info("sync_interface_list_open() failed. %s (%s)",
+ primary_msg ? primary_msg : "no message",
+ secondary_msg ? secondary_msg : "no secondary message");
+ if (err_str) {
+ *err_str = primary_msg;
+ } else {
+ g_free(primary_msg);
+ }
+ g_free(secondary_msg);
+ *err = CANT_GET_INTERFACE_LIST;
+
+ /*
+ * Add the extcap interfaces that can exist; they may exist
+ * even if no native interfaces have been found.
+ */
+ ws_info("Loading External Capture Interface List ...");
+ if_list = append_extcap_interface_list(if_list);
+ return if_list;
+ }
+
+ /* Split our lines */
+#ifdef _WIN32
+ raw_list = g_strsplit(data, "\r\n", 0);
+#else
+ raw_list = g_strsplit(data, "\n", 0);
+#endif
+ g_free(data);
+
+ for (i = 0; raw_list[i] != NULL; i++) {
+ if_parts = g_strsplit(raw_list[i], "\t", 7);
+ if (if_parts[0] == NULL || if_parts[1] == NULL || if_parts[2] == NULL ||
+ if_parts[3] == NULL || if_parts[4] == NULL || if_parts[5] == NULL ||
+ if_parts[6] == NULL) {
+ g_strfreev(if_parts);
+ continue;
+ }
+
+ /* Number followed by the name, e.g "1. eth0" */
+ name = strchr(if_parts[0], ' ');
+ if (name) {
+ name++;
+ } else {
+ g_strfreev(if_parts);
+ continue;
+ }
+
+ if_info = g_new0(if_info_t,1);
+ if_info->name = g_strdup(name);
+ if (strlen(if_parts[1]) > 0)
+ if_info->vendor_description = g_strdup(if_parts[1]);
+ if (strlen(if_parts[2]) > 0)
+ if_info->friendly_name = g_strdup(if_parts[2]);
+ if_info->type = (interface_type)(int)strtol(if_parts[3], NULL, 10);
+ addr_parts = g_strsplit(if_parts[4], ",", 0);
+ for (j = 0; addr_parts[j] != NULL; j++) {
+ if_addr = g_new0(if_addr_t,1);
+ if (ws_inet_pton4(addr_parts[j], &if_addr->addr.ip4_addr)) {
+ if_addr->ifat_type = IF_AT_IPv4;
+ } else if (ws_inet_pton6(addr_parts[j], (ws_in6_addr *)&if_addr->addr.ip6_addr)) {
+ if_addr->ifat_type = IF_AT_IPv6;
+ } else {
+ g_free(if_addr);
+ if_addr = NULL;
+ }
+ if (if_addr) {
+ if_info->addrs = g_slist_append(if_info->addrs, if_addr);
+ }
+ }
+ if (strcmp(if_parts[5], "loopback") == 0)
+ if_info->loopback = true;
+ if_info->extcap = g_strdup(if_parts[6]);
+ g_strfreev(if_parts);
+ g_strfreev(addr_parts);
+ if_list = g_list_append(if_list, if_info);
+ }
+ g_strfreev(raw_list);
+
+#ifdef HAVE_PCAP_REMOTE
+ /* Add the remote interface list */
+ if (remote_interface_list && g_list_length(remote_interface_list) > 0) {
+ if_list = append_remote_list(if_list);
+ }
+#endif
+
+ /* Add the extcap interfaces after the native and remote interfaces */
+ ws_info("Loading External Capture Interface List ...");
+ if_list = append_extcap_interface_list(if_list);
+
+ return if_list;
+}
+
+/* XXX - We parse simple text output to get our interface list. Should
+ * we use "real" data serialization instead, e.g. via XML? */
+if_capabilities_t *
+capture_get_if_capabilities(const char *ifname, bool monitor_mode,
+ const char *auth_string,
+ char **err_primary_msg, char **err_secondary_msg,
+ void (*update_cb)(void))
+{
+ if_capabilities_t *caps;
+ GList *linktype_list = NULL, *timestamp_list = NULL;
+ int err, i;
+ char *data, *primary_msg, *secondary_msg;
+ char **raw_list;
+
+ /* see if the interface is from extcap */
+ caps = extcap_get_if_dlts(ifname, err_primary_msg);
+ if (caps != NULL)
+ return caps;
+
+ /* return if the extcap interface generated an error */
+ if (err_primary_msg != NULL && *err_primary_msg != NULL)
+ return NULL;
+
+ /* Try to get our interface list */
+ err = sync_if_capabilities_open(ifname, monitor_mode, auth_string, &data,
+ &primary_msg, &secondary_msg, update_cb);
+ if (err != 0) {
+ ws_info("Capture Interface Capabilities failed. Error %d, %s",
+ err, primary_msg ? primary_msg : "no message");
+ if (err_primary_msg)
+ *err_primary_msg = primary_msg;
+ else
+ g_free(primary_msg);
+ if (err_secondary_msg)
+ *err_secondary_msg = secondary_msg;
+ else
+ g_free(secondary_msg);
+ return NULL;
+ }
+
+ /* Split our lines */
+#ifdef _WIN32
+ raw_list = g_strsplit(data, "\r\n", 0);
+#else
+ raw_list = g_strsplit(data, "\n", 0);
+#endif
+ g_free(data);
+
+ /*
+ * First line is 0 if monitor mode isn't supported, 1 if it is.
+ */
+ if (raw_list[0] == NULL || *raw_list[0] == '\0') {
+ ws_info("Capture Interface Capabilities returned no information.");
+ if (err_primary_msg) {
+ *err_primary_msg = g_strdup("Dumpcap returned no interface capability information");
+ }
+ g_strfreev(raw_list);
+ return NULL;
+ }
+
+ /*
+ * Allocate the interface capabilities structure.
+ */
+ caps = (if_capabilities_t *)g_malloc(sizeof *caps);
+ switch (*raw_list[0]) {
+
+ case '0':
+ caps->can_set_rfmon = false;
+ break;
+
+ case '1':
+ caps->can_set_rfmon = true;
+ break;
+
+ default:
+ ws_info("Capture Interface Capabilities returned bad information.");
+ if (err_primary_msg) {
+ *err_primary_msg = ws_strdup_printf("Dumpcap returned \"%s\" for monitor-mode capability",
+ raw_list[0]);
+ }
+ g_free(caps);
+ g_strfreev(raw_list);
+ return NULL;
+ }
+
+ /*
+ * The following are link-layer types.
+ */
+ for (i = 1; raw_list[i] != NULL && *raw_list[i] != '\0'; i++) {
+ data_link_info_t *data_link_info;
+ /* ...and what if the interface name has a tab in it, Mr. Clever Programmer? */
+ char **lt_parts = g_strsplit(raw_list[i], "\t", 3);
+ if (lt_parts[0] == NULL || lt_parts[1] == NULL || lt_parts[2] == NULL) {
+ g_strfreev(lt_parts);
+ continue;
+ }
+
+ data_link_info = g_new(data_link_info_t,1);
+ data_link_info->dlt = (int) strtol(lt_parts[0], NULL, 10);
+ data_link_info->name = g_strdup(lt_parts[1]);
+ if (strcmp(lt_parts[2], "(not supported)") != 0)
+ data_link_info->description = g_strdup(lt_parts[2]);
+ else
+ data_link_info->description = NULL;
+ g_strfreev(lt_parts);
+
+ linktype_list = g_list_append(linktype_list, data_link_info);
+ }
+
+ if (raw_list[i]) { /* Oh, timestamp types! */
+ for (i++; raw_list[i] != NULL && *raw_list[i] != '\0'; i++) {
+ timestamp_info_t *timestamp_info;
+ char **tt_parts = g_strsplit(raw_list[i], "\t", 2);
+ if (tt_parts[0] == NULL || tt_parts[1] == NULL) {
+ g_strfreev(tt_parts);
+ continue;
+ }
+
+ timestamp_info = g_new(timestamp_info_t,1);
+ timestamp_info->name = g_strdup(tt_parts[0]);
+ timestamp_info->description = g_strdup(tt_parts[1]);
+ g_strfreev(tt_parts);
+
+ timestamp_list = g_list_append(timestamp_list, timestamp_info);
+ }
+ }
+
+ g_strfreev(raw_list);
+
+ caps->data_link_types = linktype_list;
+ /* Might be NULL. Not all systems report timestamp types */
+ caps->timestamp_types = timestamp_list;
+
+ return caps;
+}
+
+#ifdef HAVE_PCAP_REMOTE
+void add_interface_to_remote_list(if_info_t *if_info)
+{
+ GSList *list;
+ if_addr_t *if_addr, *temp_addr;
+
+ if_info_t *temp = g_new0(if_info_t, 1);
+ temp->name = g_strdup(if_info->name);
+ temp->friendly_name = g_strdup(if_info->friendly_name);
+ temp->vendor_description = g_strdup(if_info->vendor_description);
+ for (list = g_slist_nth(if_info->addrs, 0); list != NULL; list = g_slist_next(list)) {
+ temp_addr = g_new0(if_addr_t, 1);
+ if_addr = (if_addr_t *)list->data;
+ if (if_addr) {
+ temp_addr->ifat_type = if_addr->ifat_type;
+ if (temp_addr->ifat_type == IF_AT_IPv4) {
+ temp_addr->addr.ip4_addr = if_addr->addr.ip4_addr;
+ } else {
+ memcpy(temp_addr->addr.ip6_addr, if_addr->addr.ip6_addr, sizeof(if_addr->addr));
+ }
+ } else {
+ g_free(temp_addr);
+ temp_addr = NULL;
+ }
+ if (temp_addr) {
+ temp->addrs = g_slist_append(temp->addrs, temp_addr);
+ }
+ }
+ temp->loopback = if_info->loopback;
+ remote_interface_list = g_list_append(remote_interface_list, temp);
+}
+#endif
+#endif /* HAVE_LIBPCAP */
diff --git a/capture/capture_ifinfo.h b/capture/capture_ifinfo.h
new file mode 100644
index 00000000..97df2363
--- /dev/null
+++ b/capture/capture_ifinfo.h
@@ -0,0 +1,142 @@
+/** @file
+ *
+ * Definitions for routines to get information about capture interfaces
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef __CAPTURE_IFINFO_H__
+#define __CAPTURE_IFINFO_H__
+
+#include <glib.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+/*
+ * Explicitly set the interface_type enum values as these values are exposed
+ * in the preferences gui.interfaces_hidden_types string.
+ */
+typedef enum {
+ IF_WIRED = 0,
+ IF_AIRPCAP = 1,
+ IF_PIPE = 2,
+ IF_STDIN = 3,
+ IF_BLUETOOTH = 4,
+ IF_WIRELESS = 5,
+ IF_DIALUP = 6,
+ IF_USB = 7,
+ IF_EXTCAP = 8,
+ IF_VIRTUAL = 9
+} interface_type;
+
+/*
+ * The list of interfaces returned by "get_interface_list()" is
+ * a list of these structures.
+ */
+typedef struct {
+ char *name; /* e.g. "eth0" */
+ char *friendly_name; /* from OS, e.g. "Local Area Connection", or
+ NULL if not available */
+ char *vendor_description;
+ /* vendor description from pcap_findalldevs(),
+ e.g. "Realtek PCIe GBE Family Controller",
+ or NULL if not available */
+ GSList *addrs; /* containing address values of if_addr_t */
+ interface_type type; /* type of interface */
+ bool loopback; /* true if loopback, false otherwise */
+ char *extcap; /* extcap arguments, which present the data to call the extcap interface */
+} if_info_t;
+
+/*
+ * An address in the "addrs" list.
+ */
+typedef enum {
+ IF_AT_IPv4,
+ IF_AT_IPv6
+} if_address_type;
+
+typedef struct {
+ if_address_type ifat_type;
+ union {
+ uint32_t ip4_addr; /* 4 byte IP V4 address, or */
+ uint8_t ip6_addr[16];/* 16 byte IP V6 address */
+ } addr;
+} if_addr_t;
+
+/**
+ * Return the list of interfaces.
+ *
+ * Local interfaces are fetched by running dumpcap.
+ * The remote and extcap interfaces are appended to the list after that.
+ */
+extern GList *capture_interface_list(int *err, char **err_str, void (*update_cb)(void));
+
+/* Error values from "get_interface_list()/capture_interface_list()". */
+#define CANT_GET_INTERFACE_LIST 1 /* error getting list */
+#define DONT_HAVE_PCAP 2 /* couldn't load WinPcap/Npcap */
+
+void free_interface_list(GList *if_list);
+
+/**
+ * Get an if_info_t for a particular interface.
+ * (May require privilege, so should only be used by dumpcap.)
+ */
+extern if_info_t *if_info_get(const char *name);
+
+/**
+ * Free an if_info_t.
+ */
+void if_info_free(if_info_t *if_info);
+
+/*
+ * "get_if_capabilities()" and "capture_if_capabilities()" return a pointer
+ * to an allocated instance of this structure. "free_if_capabilities()"
+ * frees the returned instance.
+ */
+typedef struct {
+ bool can_set_rfmon; /* true if can be put into monitor mode */
+ GList *data_link_types; /* GList of data_link_info_t's */
+ GList *timestamp_types; /* GList of timestamp_info_t's */
+} if_capabilities_t;
+
+/*
+ * Information about data link types.
+ */
+typedef struct {
+ int dlt; /* e.g. DLT_EN10MB (which is 1) */
+ char *name; /* e.g. "EN10MB" or "DLT 1" */
+ char *description; /* descriptive name from wiretap e.g. "Ethernet", NULL if unknown */
+} data_link_info_t;
+
+/*
+ * Information about timestamp types.
+ */
+typedef struct {
+ char *name; /* e.g. "adapter_unsynced" */
+ char *description; /* description from libpcap e.g. "Adapter, not synced with system time" */
+} timestamp_info_t;
+
+/**
+ * Fetch the linktype list for the specified interface from a child process.
+ */
+extern if_capabilities_t *
+capture_get_if_capabilities(const char *devname, bool monitor_mode,
+ const char *auth_string,
+ char **err_primary_msg, char **err_secondary_msg,
+ void (*update_cb)(void));
+
+void free_if_capabilities(if_capabilities_t *caps);
+
+void add_interface_to_remote_list(if_info_t *if_info);
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* __CAPTURE_IFINFO_H__ */
diff --git a/capture/capture_session.h b/capture/capture_session.h
new file mode 100644
index 00000000..f7876723
--- /dev/null
+++ b/capture/capture_session.h
@@ -0,0 +1,145 @@
+/** @file
+ *
+ * State of a capture session
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef __CAPCHILD_CAPTURE_SESSION_H__
+#define __CAPCHILD_CAPTURE_SESSION_H__
+
+#ifndef _WIN32
+#include <sys/types.h>
+#include <stdint.h>
+#endif
+
+#include "capture_opts.h"
+
+#include <epan/fifo_string_cache.h>
+#include <wsutil/processes.h>
+
+#include "cfile.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+#ifdef HAVE_LIBPCAP
+/* Current state of capture engine. XXX - differentiate states */
+typedef enum {
+ CAPTURE_STOPPED, /**< stopped */
+ CAPTURE_PREPARING, /**< preparing, but still no response from capture child */
+ CAPTURE_RUNNING /**< capture child signalled ok, capture is running now */
+} capture_state;
+
+struct _info_data;
+
+/*
+ * State of a capture session.
+ */
+typedef struct _capture_session capture_session;
+
+/*
+ * Types of callbacks.
+ */
+
+/**
+ * Capture child told us we have a new (or the first) capture file.
+ */
+typedef bool (*new_file_fn)(capture_session *cap_session, char *new_file);
+
+/**
+ * Capture child told us we have new packets to read.
+ */
+typedef void (*new_packets_fn)(capture_session *cap_session, int to_read);
+
+/**
+ * Capture child told us how many dropped packets it counted.
+ */
+typedef void (*drops_fn)(capture_session *cap_session, uint32_t dropped,
+ const char *interface_name);
+
+/**
+ * Capture child told us that an error has occurred while starting
+ * the capture.
+ */
+typedef void (*error_fn)(capture_session *cap_session, char *error_msg,
+ char *secondary_error_msg);
+
+/**
+ * Capture child told us that an error has occurred while parsing a
+ * capture filter when starting/running the capture.
+ */
+typedef void (*cfilter_error_fn)(capture_session *cap_session, unsigned i,
+ const char *error_message);
+
+/**
+ * Capture child closed its side of the pipe, report any error and
+ * do the required cleanup.
+ */
+typedef void (*closed_fn)(capture_session *cap_session, char *msg);
+
+/*
+ * The structure for the session.
+ */
+struct _capture_session {
+ ws_process_id fork_child; /**< If not WS_INVALID_PID, in parent, process ID of child */
+ int fork_child_status; /**< Child exit status */
+ int pipe_input_id; /**< GLib input pipe source ID */
+#ifdef _WIN32
+ int signal_pipe_write_fd; /**< the pipe to signal the child */
+#endif
+ capture_state state; /**< current state of the capture engine */
+#ifndef _WIN32
+ uid_t owner; /**< owner of the cfile */
+ gid_t group; /**< group of the cfile */
+#endif
+ bool session_will_restart; /**< Set when session will restart */
+ uint32_t count; /**< Total number of frames captured */
+ uint32_t count_pending; /**< Number of frames captured but not yet read */
+ capture_options *capture_opts; /**< options for this capture */
+ capture_file *cf; /**< handle to cfile */
+ wtap_rec rec; /**< record we're reading packet metadata into */
+ Buffer buf; /**< Buffer we're reading packet data into */
+ struct wtap *wtap; /**< current wtap file */
+ struct _info_data *cap_data_info; /**< stats for this capture */
+
+ // If the user wants to ignore duplicate frames, we need these.
+ fifo_string_cache_t frame_dup_cache;
+ GChecksum *frame_cksum;
+
+ /*
+ * Routines supplied by our caller; we call them back to notify them
+ * of various events.
+ */
+ new_file_fn new_file;
+ new_packets_fn new_packets;
+ drops_fn drops;
+ error_fn error;
+ cfilter_error_fn cfilter_error;
+ closed_fn closed;
+};
+
+extern void
+capture_session_init(capture_session *cap_session, capture_file *cf,
+ new_file_fn new_file, new_packets_fn new_packets,
+ drops_fn drops, error_fn error,
+ cfilter_error_fn cfilter_error, closed_fn closed);
+
+void capture_process_finished(capture_session *cap_session);
+#else
+
+/* dummy is needed because clang throws the error: empty struct has size 0 in C, size 1 in C++ */
+typedef struct _capture_session {int dummy;} capture_session;
+
+#endif /* HAVE_LIBPCAP */
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* __CAPCHILD_CAPTURE_SESSION_H__ */
diff --git a/capture/capture_sync.c b/capture/capture_sync.c
new file mode 100644
index 00000000..1af14187
--- /dev/null
+++ b/capture/capture_sync.c
@@ -0,0 +1,2206 @@
+/* capture_sync.c
+ * Synchronisation between Wireshark capture parent and child instances
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+#define WS_LOG_DOMAIN LOG_DOMAIN_CAPTURE
+
+#include <wireshark.h>
+
+#ifdef HAVE_LIBPCAP
+
+#include <glib.h>
+#include <stdio.h>
+#include <stdlib.h>
+
+#include <signal.h>
+
+#include <wsutil/strtoi.h>
+#include <wsutil/ws_assert.h>
+
+#ifdef _WIN32
+#include <wsutil/unicode-utils.h>
+#include <wsutil/win32-utils.h>
+#include <wsutil/ws_pipe.h>
+#else
+#include <glib-unix.h>
+#endif
+
+#ifdef HAVE_SYS_WAIT_H
+# include <sys/wait.h>
+#endif
+
+#include "capture/capture-pcap-util.h"
+
+#ifndef _WIN32
+/*
+ * Define various POSIX macros (and, in the case of WCOREDUMP, non-POSIX
+ * macros) on UNIX systems that don't have them.
+ */
+#ifndef WIFEXITED
+# define WIFEXITED(status) (((status) & 0177) == 0)
+#endif
+#ifndef WIFSTOPPED
+# define WIFSTOPPED(status) (((status) & 0177) == 0177)
+#endif
+#ifndef WIFSIGNALED
+# define WIFSIGNALED(status) (!WIFSTOPPED(status) && !WIFEXITED(status))
+#endif
+#ifndef WEXITSTATUS
+# define WEXITSTATUS(status) ((status) >> 8)
+#endif
+#ifndef WTERMSIG
+# define WTERMSIG(status) ((status) & 0177)
+#endif
+#ifndef WCOREDUMP
+# define WCOREDUMP(status) ((status) & 0200)
+#endif
+#ifndef WSTOPSIG
+# define WSTOPSIG(status) ((status) >> 8)
+#endif
+#endif /* _WIN32 */
+
+#include <epan/packet.h>
+#include <epan/prefs.h>
+
+#include "file.h"
+
+#include "ui/capture.h"
+#include <capture/capture_sync.h>
+
+#include "sync_pipe.h"
+
+#ifdef _WIN32
+#include "capture/capture-wpcap.h"
+#endif
+
+#include "ui/ws_ui_util.h"
+
+#include <wsutil/filesystem.h>
+#include <wsutil/file_util.h>
+#include <wsutil/report_message.h>
+#include "extcap.h"
+
+#ifdef _WIN32
+#include <process.h> /* For spawning child process */
+#endif
+
+#include <wsutil/ws_pipe.h>
+
+#ifdef _WIN32
+static int create_dummy_signal_pipe(char **msg);
+static HANDLE dummy_signal_pipe; /* Dummy named pipe which lets the child check for a dropped connection */
+static char *dummy_control_id;
+#else
+static const char *sync_pipe_signame(int);
+#endif
+
+
+static gboolean sync_pipe_input_cb(GIOChannel *pipe_io, capture_session *cap_session);
+static int sync_pipe_wait_for_child(ws_process_id fork_child, char **msgp);
+static void pipe_convert_header(const unsigned char *header, int header_len, char *indicator, int *block_len);
+static ssize_t pipe_read_block(GIOChannel *pipe_io, char *indicator, int len, char *msg,
+ char **err_msg);
+
+static void (*fetch_dumpcap_pid)(ws_process_id) = NULL;
+
+static void free_argv(char** argv, int argc)
+{
+ int i;
+ for (i = 0; i < argc; i++)
+ g_free(argv[i]);
+ g_free(argv);
+}
+
+void
+capture_session_init(capture_session *cap_session, capture_file *cf,
+ new_file_fn new_file, new_packets_fn new_packets,
+ drops_fn drops, error_fn error,
+ cfilter_error_fn cfilter_error, closed_fn closed)
+{
+ cap_session->cf = cf;
+ cap_session->fork_child = WS_INVALID_PID; /* invalid process handle */
+ cap_session->pipe_input_id = 0;
+#ifdef _WIN32
+ cap_session->signal_pipe_write_fd = -1;
+#endif
+ cap_session->state = CAPTURE_STOPPED;
+#ifndef _WIN32
+ cap_session->owner = getuid();
+ cap_session->group = getgid();
+#endif
+ cap_session->count = 0;
+ cap_session->count_pending = 0;
+ cap_session->session_will_restart = false;
+
+ cap_session->new_file = new_file;
+ cap_session->new_packets = new_packets;
+ cap_session->drops = drops;
+ cap_session->error = error;
+ cap_session->cfilter_error = cfilter_error;
+ cap_session->closed = closed;
+ cap_session->frame_cksum = NULL;
+}
+
+void capture_process_finished(capture_session *cap_session)
+{
+ capture_options *capture_opts = cap_session->capture_opts;
+ interface_options *interface_opts;
+ GString *message;
+ unsigned i;
+
+ if (!extcap_session_stop(cap_session)) {
+ /* Atleast one extcap process did not fully finish yet, wait for it */
+ return;
+ }
+
+ if (cap_session->fork_child != WS_INVALID_PID) {
+ if (capture_opts->stop_after_extcaps) {
+ /* User has requested capture stop and all extcaps are gone now */
+ capture_opts->stop_after_extcaps = false;
+ sync_pipe_stop(cap_session);
+ }
+ /* Wait for child process to end, session is not closed yet */
+ return;
+ }
+
+ /* Construct message and close session */
+ message = g_string_new(capture_opts->closed_msg);
+ for (i = 0; i < capture_opts->ifaces->len; i++) {
+ interface_opts = &g_array_index(capture_opts->ifaces, interface_options, i);
+ if (interface_opts->if_type != IF_EXTCAP) {
+ continue;
+ }
+
+ if ((interface_opts->extcap_stderr != NULL) &&
+ (interface_opts->extcap_stderr->len > 0)) {
+ if (message->len > 0) {
+ g_string_append(message, "\n");
+ }
+ g_string_append(message, "Error from extcap pipe: ");
+ g_string_append(message, interface_opts->extcap_stderr->str);
+ }
+ }
+
+ cap_session->closed(cap_session, message->str);
+ g_string_free(message, true);
+ g_free(capture_opts->closed_msg);
+ capture_opts->closed_msg = NULL;
+ capture_opts->stop_after_extcaps = false;
+}
+
+/* Append an arg (realloc) to an argc/argv array */
+/* (add a string pointer to a NULL-terminated array of string pointers) */
+static char **
+sync_pipe_add_arg(char **args, int *argc, const char *arg)
+{
+ /* Grow the array; "*argc" currently contains the number of string
+ pointers, *not* counting the NULL pointer at the end, so we have
+ to add 2 in order to get the new size of the array, including the
+ new pointer and the terminating NULL pointer. */
+ args = (char **)g_realloc( (void *) args, (*argc + 2) * sizeof (char *));
+
+ /* Stuff the pointer into the penultimate element of the array, which
+ is the one at the index specified by "*argc". */
+ args[*argc] = g_strdup(arg);
+ /* Now bump the count. */
+ (*argc)++;
+
+ /* We overwrite the NULL pointer; put it back right after the
+ element we added. */
+ args[*argc] = NULL;
+
+ return args;
+}
+
+/* Initialize an argument list and add dumpcap to it. */
+static char **
+init_pipe_args(int *argc) {
+ char *exename;
+ char **argv;
+
+ /* Find the absolute path of the dumpcap executable. */
+ exename = get_executable_path("dumpcap");
+ if (exename == NULL) {
+ return NULL;
+ }
+
+ /* Allocate the string pointer array with enough space for the
+ terminating NULL pointer. */
+ *argc = 0;
+ argv = (char **)g_malloc(sizeof (char *));
+ *argv = NULL;
+
+ /* Make that the first argument in the argument list (argv[0]). */
+ argv = sync_pipe_add_arg(argv, argc, exename);
+
+ /* sync_pipe_add_arg strdupes exename, so we should free our copy */
+ g_free(exename);
+
+ return argv;
+}
+
+static gboolean
+pipe_io_cb(GIOChannel *pipe_io, GIOCondition condition _U_, void * user_data)
+{
+ capture_session *cap_session = (capture_session *)user_data;
+ if (!sync_pipe_input_cb(pipe_io, cap_session)) {
+ cap_session->pipe_input_id = 0;
+ return G_SOURCE_REMOVE;
+ }
+ return G_SOURCE_CONTINUE;
+}
+
+/*
+ * Open two pipes to dumpcap with the supplied arguments, one for its
+ * standard output and one for its standard error.
+ *
+ * On success, *msg is unchanged and 0 is returned; data_read_fd,
+ * message_read_fd, and fork_child point to the standard output pipe's
+ * file descriptor, the standard error pipe's file descriptor, and
+ * the child's PID/handle, respectively.
+ *
+ * On failure, *msg points to an error message for the failure, and -1 is
+ * returned, in which case *msg must be freed with g_free().
+ */
+/* XXX - assumes PIPE_BUF_SIZE > SP_MAX_MSG_LEN */
+#define ARGV_NUMBER_LEN 24
+#define PIPE_BUF_SIZE 5120
+static int
+#ifdef _WIN32
+sync_pipe_open_command(char* const argv[], int *data_read_fd,
+ GIOChannel **message_read_io, int *signal_write_fd,
+ ws_process_id *fork_child, GArray *ifaces,
+ char **msg, void(*update_cb)(void))
+#else
+sync_pipe_open_command(char* const argv[], int *data_read_fd,
+ GIOChannel **message_read_io, int *signal_write_fd _U_,
+ ws_process_id *fork_child, GArray *ifaces _U_,
+ char **msg, void(*update_cb)(void))
+#endif
+{
+ enum PIPES { PIPE_READ, PIPE_WRITE }; /* Constants 0 and 1 for PIPE_READ and PIPE_WRITE */
+ int message_read_fd = -1;
+#ifdef _WIN32
+ HANDLE sync_pipe[2]; /* pipe used to send messages from child to parent */
+ HANDLE data_pipe[2]; /* pipe used to send data from child to parent */
+ int signal_pipe_write_fd = -1;
+ HANDLE signal_pipe; /* named pipe used to send messages from parent to child (currently only stop) */
+ char control_id[ARGV_NUMBER_LEN];
+ char *signal_pipe_name;
+ size_t i_handles = 0;
+ HANDLE *handles;
+ GString *args = g_string_sized_new(200);
+ char *quoted_arg;
+ SECURITY_ATTRIBUTES sa;
+ STARTUPINFO si;
+ PROCESS_INFORMATION pi;
+ int i;
+ unsigned j;
+ interface_options *interface_opts;
+#else
+ int sync_pipe[2]; /* pipe used to send messages from child to parent */
+ int data_pipe[2]; /* pipe used to send data from child to parent */
+#endif
+ *fork_child = WS_INVALID_PID;
+ if (data_read_fd != NULL) {
+ *data_read_fd = -1;
+ }
+ *message_read_io = NULL;
+ ws_debug("sync_pipe_open_command");
+
+ if (!msg) {
+ /* We can't return anything */
+#ifdef _WIN32
+ g_string_free(args, true);
+#endif
+ return -1;
+ }
+
+#ifdef _WIN32
+ /* init SECURITY_ATTRIBUTES */
+ sa.nLength = sizeof(SECURITY_ATTRIBUTES);
+ sa.bInheritHandle = false;
+ sa.lpSecurityDescriptor = NULL;
+
+ /* Create a pipe for the child process to send us messages */
+ /* (increase this value if you have trouble while fast capture file switches) */
+ if (! CreatePipe(&sync_pipe[PIPE_READ], &sync_pipe[PIPE_WRITE], &sa, PIPE_BUF_SIZE)) {
+ /* Couldn't create the message pipe between parent and child. */
+ *msg = ws_strdup_printf("Couldn't create sync pipe: %s",
+ win32strerror(GetLastError()));
+ return -1;
+ }
+
+ /*
+ * Associate a C run-time file handle with the Windows HANDLE for the
+ * read side of the message pipe.
+ *
+ * (See http://www.flounder.com/handles.htm for information on various
+ * types of file handle in C/C++ on Windows.)
+ */
+ message_read_fd = _open_osfhandle( (intptr_t) sync_pipe[PIPE_READ], _O_BINARY);
+ if (message_read_fd == -1) {
+ *msg = ws_strdup_printf("Couldn't get C file handle for message read pipe: %s", g_strerror(errno));
+ CloseHandle(sync_pipe[PIPE_READ]);
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+ return -1;
+ }
+
+ if (data_read_fd != NULL) {
+ /* Create a pipe for the child process to send us data */
+ /* (increase this value if you have trouble while fast capture file switches) */
+ if (! CreatePipe(&data_pipe[PIPE_READ], &data_pipe[PIPE_WRITE], &sa, PIPE_BUF_SIZE)) {
+ /* Couldn't create the message pipe between parent and child. */
+ *msg = ws_strdup_printf("Couldn't create data pipe: %s",
+ win32strerror(GetLastError()));
+ ws_close(message_read_fd); /* Should close sync_pipe[PIPE_READ] */
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+ return -1;
+ }
+
+ /*
+ * Associate a C run-time file handle with the Windows HANDLE for the
+ * read side of the data pipe.
+ *
+ * (See http://www.flounder.com/handles.htm for information on various
+ * types of file handle in C/C++ on Windows.)
+ */
+ *data_read_fd = _open_osfhandle( (intptr_t) data_pipe[PIPE_READ], _O_BINARY);
+ if (*data_read_fd == -1) {
+ *msg = ws_strdup_printf("Couldn't get C file handle for data read pipe: %s", g_strerror(errno));
+ CloseHandle(data_pipe[PIPE_READ]);
+ CloseHandle(data_pipe[PIPE_WRITE]);
+ ws_close(message_read_fd); /* Should close sync_pipe[PIPE_READ] */
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+ return -1;
+ }
+ }
+
+ if (signal_write_fd != NULL) {
+ /* Create the signal pipe */
+ snprintf(control_id, ARGV_NUMBER_LEN, "%ld", GetCurrentProcessId());
+ signal_pipe_name = ws_strdup_printf(SIGNAL_PIPE_FORMAT, control_id);
+ signal_pipe = CreateNamedPipe(utf_8to16(signal_pipe_name),
+ PIPE_ACCESS_OUTBOUND, PIPE_TYPE_BYTE, 1, 65535, 65535, 0, NULL);
+ g_free(signal_pipe_name);
+
+ if (signal_pipe == INVALID_HANDLE_VALUE) {
+ /* Couldn't create the signal pipe between parent and child. */
+ *msg = ws_strdup_printf("Couldn't create signal pipe: %s",
+ win32strerror(GetLastError()));
+ ws_close(message_read_fd); /* Should close sync_pipe[PIPE_READ] */
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+ return -1;
+ }
+
+ /*
+ * Associate a C run-time file handle with the Windows HANDLE for the
+ * read side of the message pipe.
+ *
+ * (See http://www.flounder.com/handles.htm for information on various
+ * types of file handle in C/C++ on Windows.)
+ */
+ signal_pipe_write_fd = _open_osfhandle( (intptr_t) signal_pipe, _O_BINARY);
+ if (signal_pipe_write_fd == -1) {
+ /* Couldn't create the pipe between parent and child. */
+ *msg = ws_strdup_printf("Couldn't get C file handle for sync pipe: %s", g_strerror(errno));
+ ws_close(message_read_fd); /* Should close sync_pipe[PIPE_READ] */
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+ CloseHandle(signal_pipe);
+ return -1;
+ }
+ }
+
+ /* init STARTUPINFO & PROCESS_INFORMATION */
+ memset(&si, 0, sizeof(si));
+ si.cb = sizeof(si);
+ memset(&pi, 0, sizeof(pi));
+#ifdef DEBUG_CHILD
+ si.dwFlags = STARTF_USESHOWWINDOW;
+ si.wShowWindow = SW_SHOW;
+#else
+ si.dwFlags = STARTF_USESTDHANDLES|STARTF_USESHOWWINDOW;
+ si.wShowWindow = SW_HIDE; /* this hides the console window */
+
+ if (data_read_fd == NULL) {
+ si.hStdInput = GetStdHandle(STD_INPUT_HANDLE);
+ si.hStdOutput = GetStdHandle(STD_OUTPUT_HANDLE);
+ } else {
+ si.hStdInput = NULL; /* handle for named pipe*/
+ si.hStdOutput = data_pipe[PIPE_WRITE];
+ }
+ si.hStdError = sync_pipe[PIPE_WRITE];
+#endif
+
+ if (ifaces) {
+ for (j = 0; j < ifaces->len; j++) {
+ interface_opts = &g_array_index(ifaces, interface_options, j);
+ if (interface_opts->extcap_fifo != NULL) {
+ i_handles++;
+ }
+ }
+ }
+ handles = g_new(HANDLE, 3 + i_handles);
+ i_handles = 0;
+ if (si.hStdInput) {
+ handles[i_handles++] = si.hStdInput;
+ }
+ if (si.hStdOutput && (si.hStdOutput != si.hStdInput)) {
+ handles[i_handles++] = si.hStdOutput;
+ }
+ handles[i_handles++] = si.hStdError;
+ if (ifaces) {
+ for (j = 0; j < ifaces->len; j++) {
+ interface_opts = &g_array_index(ifaces, interface_options, j);
+ if (interface_opts->extcap_fifo != NULL) {
+ handles[i_handles++] = interface_opts->extcap_pipe_h;
+ }
+ }
+ }
+
+ /* convert args array into a single string */
+ /* XXX - could change sync_pipe_add_arg() instead */
+ /* there is a drawback here: the length is internally limited to 1024 bytes */
+ for(i=0; argv[i] != 0; i++) {
+ if(i != 0) g_string_append_c(args, ' '); /* don't prepend a space before the path!!! */
+ quoted_arg = protect_arg(argv[i]);
+ g_string_append(args, quoted_arg);
+ g_free(quoted_arg);
+ }
+
+ /* call dumpcap */
+ if(!win32_create_process(argv[0], args->str, NULL, NULL, i_handles, handles,
+ CREATE_NEW_CONSOLE, NULL, NULL, &si, &pi)) {
+ *msg = ws_strdup_printf("Couldn't run %s in child process: %s",
+ args->str, win32strerror(GetLastError()));
+ if (data_read_fd) {
+ ws_close(*data_read_fd); /* Should close data_pipe[PIPE_READ] */
+ CloseHandle(data_pipe[PIPE_WRITE]);
+ } else {
+ ws_close(signal_pipe_write_fd);
+ }
+ ws_close(message_read_fd); /* Should close sync_pipe[PIPE_READ] */
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+ g_string_free(args, true);
+ g_free(handles);
+ return -1;
+ }
+ *fork_child = pi.hProcess;
+ /* We may need to store this and close it later */
+ CloseHandle(pi.hThread);
+ g_string_free(args, true);
+ g_free(handles);
+
+ if (signal_write_fd != NULL) {
+ *signal_write_fd = signal_pipe_write_fd;
+ }
+#else /* _WIN32 */
+ /* Create a pipe for the child process to send us messages */
+ if (pipe(sync_pipe) < 0) {
+ /* Couldn't create the message pipe between parent and child. */
+ *msg = ws_strdup_printf("Couldn't create sync pipe: %s", g_strerror(errno));
+ return -1;
+ }
+
+ if (data_read_fd != NULL) {
+ /* Create a pipe for the child process to send us data */
+ if (pipe(data_pipe) < 0) {
+ /* Couldn't create the data pipe between parent and child. */
+ *msg = ws_strdup_printf("Couldn't create data pipe: %s", g_strerror(errno));
+ ws_close(sync_pipe[PIPE_READ]);
+ ws_close(sync_pipe[PIPE_WRITE]);
+ return -1;
+ }
+ }
+
+ if ((*fork_child = fork()) == 0) {
+ /*
+ * Child process - run dumpcap with the right arguments to make
+ * it just capture with the specified capture parameters
+ */
+ if (data_read_fd != NULL) {
+ dup2(data_pipe[PIPE_WRITE], 1);
+ ws_close(data_pipe[PIPE_READ]);
+ ws_close(data_pipe[PIPE_WRITE]);
+ }
+ dup2(sync_pipe[PIPE_WRITE], 2);
+ ws_close(sync_pipe[PIPE_READ]);
+ ws_close(sync_pipe[PIPE_WRITE]);
+ execv(argv[0], argv);
+ sync_pipe_write_int_msg(2, SP_EXEC_FAILED, errno);
+
+ /* Exit with "_exit()", so that we don't close the connection
+ to the X server (and cause stuff buffered up by our parent but
+ not yet sent to be sent, as that stuff should only be sent by
+ our parent). We've sent an error message to the parent, so
+ we exit with an exit status of 1 (any exit status other than
+ 0 or 1 will cause an additional message to report that exit
+ status, over and above the error message we sent to the parent). */
+ _exit(1);
+ }
+
+ if (fetch_dumpcap_pid && *fork_child > 0)
+ fetch_dumpcap_pid(*fork_child);
+
+ if (data_read_fd != NULL) {
+ *data_read_fd = data_pipe[PIPE_READ];
+ }
+ message_read_fd = sync_pipe[PIPE_READ];
+#endif
+
+ /* Parent process - read messages from the child process over the
+ sync pipe. */
+
+ /* Close the write sides of the pipes, so that only the child has them
+ open, and thus they completely close, and thus return to us
+ an EOF indication, if the child closes them (either deliberately
+ or by exiting abnormally). */
+#ifdef _WIN32
+ if (data_read_fd != NULL) {
+ CloseHandle(data_pipe[PIPE_WRITE]);
+ }
+ CloseHandle(sync_pipe[PIPE_WRITE]);
+#else
+ if (data_read_fd != NULL) {
+ ws_close(data_pipe[PIPE_WRITE]);
+ }
+ ws_close(sync_pipe[PIPE_WRITE]);
+#endif
+
+ if (*fork_child == WS_INVALID_PID) {
+ /* We couldn't even create the child process. */
+ *msg = ws_strdup_printf("Couldn't create child process: %s", g_strerror(errno));
+ if (data_read_fd != NULL) {
+ ws_close(*data_read_fd);
+ }
+#ifdef _WIN32
+ if (signal_write_fd != NULL) {
+ ws_close(signal_pipe_write_fd);
+ }
+#endif
+ ws_close(message_read_fd);
+ return -1;
+ }
+
+#ifdef _WIN32
+ *message_read_io = g_io_channel_win32_new_fd(message_read_fd);
+#else
+ *message_read_io = g_io_channel_unix_new(message_read_fd);
+#endif
+ g_io_channel_set_encoding(*message_read_io, NULL, NULL);
+ g_io_channel_set_buffered(*message_read_io, false);
+ g_io_channel_set_close_on_unref(*message_read_io, true);
+
+ /* we might wait for a moment till child is ready, so update screen now */
+ if (update_cb) update_cb();
+ return 0;
+}
+
+/* a new capture run: start a new dumpcap task and hand over parameters through command line */
+bool
+sync_pipe_start(capture_options *capture_opts, GPtrArray *capture_comments,
+ capture_session *cap_session, info_data_t* cap_data,
+ void (*update_cb)(void))
+{
+#ifdef _WIN32
+ size_t i_handles = 0;
+ char control_id[ARGV_NUMBER_LEN];
+#endif
+ GIOChannel *sync_pipe_read_io;
+ int argc;
+ char **argv;
+ int i;
+ unsigned j;
+ interface_options *interface_opts;
+
+ if (capture_opts->ifaces->len > 1)
+ capture_opts->use_pcapng = true;
+ ws_debug("sync_pipe_start");
+ capture_opts_log(LOG_DOMAIN_CAPTURE, LOG_LEVEL_DEBUG, capture_opts);
+
+ cap_session->fork_child = WS_INVALID_PID;
+ cap_session->capture_opts = capture_opts;
+
+ if (!extcap_init_interfaces(cap_session)) {
+ report_failure("Unable to init extcaps. (tmp fifo already exists?)");
+ return false;
+ }
+
+ argv = init_pipe_args(&argc);
+ if (!argv) {
+ /* We don't know where to find dumpcap. */
+ report_failure("We don't know where to find dumpcap.");
+ return false;
+ }
+
+ if (capture_opts->ifaces->len > 1)
+ argv = sync_pipe_add_arg(argv, &argc, "-t");
+
+ if (capture_opts->use_pcapng)
+ argv = sync_pipe_add_arg(argv, &argc, "-n");
+ else
+ argv = sync_pipe_add_arg(argv, &argc, "-P");
+
+ if (capture_comments != NULL) {
+ for (j = 0; j < capture_comments->len; j++) {
+ argv = sync_pipe_add_arg(argv, &argc, "--capture-comment");
+ argv = sync_pipe_add_arg(argv, &argc, (char*)g_ptr_array_index(capture_comments, j));
+ }
+ }
+
+ if (capture_opts->temp_dir) {
+ argv = sync_pipe_add_arg(argv, &argc, "--temp-dir");
+ argv = sync_pipe_add_arg(argv, &argc, capture_opts->temp_dir);
+ }
+
+ if (capture_opts->multi_files_on) {
+ if (capture_opts->has_autostop_filesize) {
+ char sfilesize[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-b");
+ snprintf(sfilesize, ARGV_NUMBER_LEN, "filesize:%u",capture_opts->autostop_filesize);
+ argv = sync_pipe_add_arg(argv, &argc, sfilesize);
+ }
+
+ if (capture_opts->has_file_duration) {
+ char sfile_duration[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-b");
+ snprintf(sfile_duration, ARGV_NUMBER_LEN, "duration:%f",capture_opts->file_duration);
+ argv = sync_pipe_add_arg(argv, &argc, sfile_duration);
+ }
+
+ if (capture_opts->has_file_interval) {
+ char sfile_interval[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-b");
+ snprintf(sfile_interval, ARGV_NUMBER_LEN, "interval:%d",capture_opts->file_interval);
+ argv = sync_pipe_add_arg(argv, &argc, sfile_interval);
+ }
+
+ if (capture_opts->has_file_packets) {
+ char sfile_packets[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-b");
+ snprintf(sfile_packets, ARGV_NUMBER_LEN, "packets:%d",capture_opts->file_packets);
+ argv = sync_pipe_add_arg(argv, &argc, sfile_packets);
+ }
+
+ if (capture_opts->has_ring_num_files) {
+ char sring_num_files[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-b");
+ snprintf(sring_num_files, ARGV_NUMBER_LEN, "files:%d",capture_opts->ring_num_files);
+ argv = sync_pipe_add_arg(argv, &argc, sring_num_files);
+ }
+
+ if (capture_opts->has_nametimenum) {
+ char nametimenum[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-b");
+ snprintf(nametimenum, ARGV_NUMBER_LEN, "nametimenum:2");
+ argv = sync_pipe_add_arg(argv, &argc, nametimenum);
+ }
+
+ if (capture_opts->has_autostop_files) {
+ char sautostop_files[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-a");
+ snprintf(sautostop_files, ARGV_NUMBER_LEN, "files:%d",capture_opts->autostop_files);
+ argv = sync_pipe_add_arg(argv, &argc, sautostop_files);
+ }
+ } else {
+ if (capture_opts->has_autostop_filesize) {
+ char sautostop_filesize[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-a");
+ snprintf(sautostop_filesize, ARGV_NUMBER_LEN, "filesize:%u",capture_opts->autostop_filesize);
+ argv = sync_pipe_add_arg(argv, &argc, sautostop_filesize);
+ }
+ }
+
+ if (capture_opts->has_autostop_packets) {
+ char scount[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-c");
+ snprintf(scount, ARGV_NUMBER_LEN, "%d",capture_opts->autostop_packets);
+ argv = sync_pipe_add_arg(argv, &argc, scount);
+ }
+
+ if (capture_opts->has_autostop_duration) {
+ char sautostop_duration[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-a");
+ snprintf(sautostop_duration, ARGV_NUMBER_LEN, "duration:%f",capture_opts->autostop_duration);
+ argv = sync_pipe_add_arg(argv, &argc, sautostop_duration);
+ }
+
+ if (capture_opts->has_autostop_written_packets) {
+ char scount[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-a");
+ snprintf(scount, ARGV_NUMBER_LEN, "packets:%d",capture_opts->autostop_written_packets);
+ argv = sync_pipe_add_arg(argv, &argc, scount);
+ }
+
+ if (capture_opts->group_read_access) {
+ argv = sync_pipe_add_arg(argv, &argc, "-g");
+ }
+
+ if (capture_opts->update_interval != DEFAULT_UPDATE_INTERVAL) {
+ char scount[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "--update-interval");
+ snprintf(scount, ARGV_NUMBER_LEN, "%d", capture_opts->update_interval);
+ argv = sync_pipe_add_arg(argv, &argc, scount);
+ }
+
+ for (j = 0; j < capture_opts->ifaces->len; j++) {
+ interface_opts = &g_array_index(capture_opts->ifaces, interface_options, j);
+
+ argv = sync_pipe_add_arg(argv, &argc, "-i");
+ if (interface_opts->extcap_fifo != NULL)
+ {
+#ifdef _WIN32
+ char *pipe = ws_strdup_printf("%s%" PRIuMAX, EXTCAP_PIPE_PREFIX, (uintmax_t)interface_opts->extcap_pipe_h);
+ argv = sync_pipe_add_arg(argv, &argc, pipe);
+ g_free(pipe);
+ i_handles++;
+#else
+ argv = sync_pipe_add_arg(argv, &argc, interface_opts->extcap_fifo);
+#endif
+ /* Add a name for the interface, to put into an IDB. */
+ argv = sync_pipe_add_arg(argv, &argc, "--ifname");
+ argv = sync_pipe_add_arg(argv, &argc, interface_opts->name);
+ if (interface_opts->descr != NULL)
+ {
+ /* Add a description for the interface, to put into an IDB. */
+ argv = sync_pipe_add_arg(argv, &argc, "--ifdescr");
+ argv = sync_pipe_add_arg(argv, &argc, interface_opts->descr);
+ }
+ }
+ else
+ argv = sync_pipe_add_arg(argv, &argc, interface_opts->name);
+
+ if (interface_opts->cfilter != NULL && strlen(interface_opts->cfilter) != 0) {
+ argv = sync_pipe_add_arg(argv, &argc, "-f");
+ argv = sync_pipe_add_arg(argv, &argc, interface_opts->cfilter);
+ }
+ if (interface_opts->has_snaplen) {
+ char ssnap[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-s");
+ snprintf(ssnap, ARGV_NUMBER_LEN, "%d", interface_opts->snaplen);
+ argv = sync_pipe_add_arg(argv, &argc, ssnap);
+ }
+
+ if (interface_opts->linktype != -1) {
+ const char *linktype = linktype_val_to_name(interface_opts->linktype);
+ if ( linktype != NULL )
+ {
+ argv = sync_pipe_add_arg(argv, &argc, "-y");
+ argv = sync_pipe_add_arg(argv, &argc, linktype);
+ }
+ }
+
+ if (!interface_opts->promisc_mode) {
+ argv = sync_pipe_add_arg(argv, &argc, "-p");
+ }
+
+#ifdef CAN_SET_CAPTURE_BUFFER_SIZE
+ if (interface_opts->buffer_size != DEFAULT_CAPTURE_BUFFER_SIZE) {
+ char buffer_size[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-B");
+ if(interface_opts->buffer_size == 0x00)
+ interface_opts->buffer_size = DEFAULT_CAPTURE_BUFFER_SIZE;
+ snprintf(buffer_size, ARGV_NUMBER_LEN, "%d", interface_opts->buffer_size);
+ argv = sync_pipe_add_arg(argv, &argc, buffer_size);
+ }
+#endif
+
+#ifdef HAVE_PCAP_CREATE
+ if (interface_opts->monitor_mode) {
+ argv = sync_pipe_add_arg(argv, &argc, "-I");
+ }
+#endif
+
+#ifdef HAVE_PCAP_REMOTE
+ if (interface_opts->datatx_udp)
+ argv = sync_pipe_add_arg(argv, &argc, "-u");
+
+ if (!interface_opts->nocap_rpcap)
+ argv = sync_pipe_add_arg(argv, &argc, "-r");
+
+ if (interface_opts->auth_type == CAPTURE_AUTH_PWD) {
+ char sauth[256];
+ argv = sync_pipe_add_arg(argv, &argc, "-A");
+ snprintf(sauth, sizeof(sauth), "%s:%s",
+ interface_opts->auth_username,
+ interface_opts->auth_password);
+ argv = sync_pipe_add_arg(argv, &argc, sauth);
+ }
+#endif
+
+#ifdef HAVE_PCAP_SETSAMPLING
+ if (interface_opts->sampling_method != CAPTURE_SAMP_NONE) {
+ char ssampling[ARGV_NUMBER_LEN];
+ argv = sync_pipe_add_arg(argv, &argc, "-m");
+ snprintf(ssampling, ARGV_NUMBER_LEN, "%s:%d",
+ interface_opts->sampling_method == CAPTURE_SAMP_BY_COUNT ? "count" :
+ interface_opts->sampling_method == CAPTURE_SAMP_BY_TIMER ? "timer" :
+ "undef",
+ interface_opts->sampling_param);
+ argv = sync_pipe_add_arg(argv, &argc, ssampling);
+ }
+#endif
+ if (interface_opts->timestamp_type) {
+ argv = sync_pipe_add_arg(argv, &argc, "--time-stamp-type");
+ argv = sync_pipe_add_arg(argv, &argc, interface_opts->timestamp_type);
+ }
+ }
+
+ /* dumpcap should be running in capture child mode (hidden feature) */
+#ifndef DEBUG_CHILD
+ argv = sync_pipe_add_arg(argv, &argc, "-Z");
+#ifdef _WIN32
+ snprintf(control_id, ARGV_NUMBER_LEN, "%ld", GetCurrentProcessId());
+ argv = sync_pipe_add_arg(argv, &argc, control_id);
+#else
+ argv = sync_pipe_add_arg(argv, &argc, SIGNAL_PIPE_CTRL_ID_NONE);
+#endif
+#endif
+
+ if (capture_opts->save_file) {
+ argv = sync_pipe_add_arg(argv, &argc, "-w");
+ argv = sync_pipe_add_arg(argv, &argc, capture_opts->save_file);
+ }
+ for (i = 0; i < argc; i++) {
+ ws_debug("argv[%d]: %s", i, argv[i]);
+ }
+ if (capture_opts->compress_type) {
+ argv = sync_pipe_add_arg(argv, &argc, "--compress-type");
+ argv = sync_pipe_add_arg(argv, &argc, capture_opts->compress_type);
+ }
+
+ int ret;
+ char* msg;
+#ifdef _WIN32
+ ret = sync_pipe_open_command(argv, NULL, &sync_pipe_read_io, &cap_session->signal_pipe_write_fd,
+ &cap_session->fork_child, capture_opts->ifaces, &msg, update_cb);
+#else
+ ret = sync_pipe_open_command(argv, NULL, &sync_pipe_read_io, NULL,
+ &cap_session->fork_child, NULL, &msg, update_cb);
+#endif
+
+ if (ret == -1) {
+ report_failure("%s", msg);
+ g_free(msg);
+ free_argv(argv, argc);
+ return false;
+ }
+
+ /* Parent process - read messages from the child process over the
+ sync pipe. */
+ free_argv(argv, argc);
+
+ cap_session->fork_child_status = 0;
+ cap_session->cap_data_info = cap_data;
+
+ /* We were able to set up to read the capture file;
+ arrange that our callback be called whenever it's possible
+ to read from the sync pipe, so that it's called when
+ the child process wants to tell us something. */
+
+ /* we have a running capture, now wait for the real capture filename */
+ if (cap_session->pipe_input_id) {
+ g_source_remove(cap_session->pipe_input_id);
+ cap_session->pipe_input_id = 0;
+ }
+ cap_session->pipe_input_id = g_io_add_watch(sync_pipe_read_io, G_IO_IN | G_IO_HUP, pipe_io_cb, cap_session);
+ /* Pipe will be closed when watch is removed */
+ g_io_channel_unref(sync_pipe_read_io);
+
+ return true;
+}
+
+/*
+ * Close the pipes we're using to read from dumpcap, and wait for it
+ * to exit. On success, *msgp is unchanged, and the exit status of
+ * dumpcap is returned. On failure (which includes "dumpcap exited
+ * due to being killed by a signal or an exception"), *msgp points
+ * to an error message for the failure, and -1 is returned. In the
+ * latter case, *msgp must be freed with g_free().
+ */
+static int
+sync_pipe_close_command(int *data_read_fd, GIOChannel *message_read_io,
+ ws_process_id *fork_child, char **msgp)
+{
+ ws_close(*data_read_fd);
+ if (message_read_io != NULL)
+ g_io_channel_unref(message_read_io);
+
+#ifdef _WIN32
+ /* XXX - Should we signal the child somehow? */
+ sync_pipe_kill(*fork_child);
+#endif
+
+ return sync_pipe_wait_for_child(*fork_child, msgp);
+}
+
+/*
+ * Run dumpcap with the supplied arguments.
+ *
+ * On success, *data points to a buffer containing the dumpcap output,
+ * *primary_msg and *secondary_message are NULL, and 0 is returned; *data
+ * must be freed with g_free().
+ *
+ * On failure, *data is NULL, *primary_msg points to an error message,
+ * *secondary_msg either points to an additional error message or is
+ * NULL, and -1 is returned; *primary_msg, and *secondary_msg if not NULL,
+ * must be freed with g_free().
+ */
+/* XXX - assumes PIPE_BUF_SIZE > SP_MAX_MSG_LEN */
+#define PIPE_BUF_SIZE 5120
+static int
+sync_pipe_run_command_actual(char* const argv[], char **data, char **primary_msg,
+ char **secondary_msg, void(*update_cb)(void))
+{
+ char *msg;
+ int data_pipe_read_fd, ret;
+ GIOChannel *sync_pipe_read_io;
+ ws_process_id fork_child;
+ char *wait_msg;
+ char buffer[PIPE_BUF_SIZE+1] = {0};
+ ssize_t nread;
+ char indicator;
+ int32_t exec_errno = 0;
+ int primary_msg_len;
+ char *primary_msg_text;
+ int secondary_msg_len;
+ char *secondary_msg_text;
+ char *combined_msg;
+ GString *data_buf = NULL;
+ ssize_t count;
+
+ ret = sync_pipe_open_command(argv, &data_pipe_read_fd, &sync_pipe_read_io, NULL,
+ &fork_child, NULL, &msg, update_cb);
+ if (ret == -1) {
+ *primary_msg = msg;
+ *secondary_msg = NULL;
+ *data = NULL;
+ return -1;
+ }
+
+ /*
+ * We were able to set up to read dumpcap's output. Do so.
+ *
+ * First, wait for an SP_ERROR_MSG message or SP_SUCCESS message.
+ */
+ nread = pipe_read_block(sync_pipe_read_io, &indicator, SP_MAX_MSG_LEN,
+ buffer, primary_msg);
+ if(nread <= 0) {
+ /* We got a read error from the sync pipe, or we got no data at
+ all from the sync pipe, so we're not going to be getting any
+ data or error message from the child process. Pick up its
+ exit status, and complain.
+
+ We don't have to worry about killing the child, if the sync pipe
+ returned an error. Usually this error is caused as the child killed
+ itself while going down. Even in the rare cases that this isn't the
+ case, the child will get an error when writing to the broken pipe
+ the next time, cleaning itself up then. */
+ ret = sync_pipe_wait_for_child(fork_child, &wait_msg);
+ if(nread == 0) {
+ /* We got an EOF from the sync pipe. That means that it exited
+ before giving us any data to read. If ret is -1, we report
+ that as a bad exit (e.g., exiting due to a signal); otherwise,
+ we report it as a premature exit. */
+ if (ret == -1)
+ *primary_msg = wait_msg;
+ else
+ *primary_msg = g_strdup("Child dumpcap closed sync pipe prematurely");
+ } else {
+ /* We got an error from the sync pipe. If ret is -1, report
+ both the sync pipe I/O error and the wait error. */
+ if (ret == -1) {
+ combined_msg = ws_strdup_printf("%s\n\n%s", *primary_msg, wait_msg);
+ g_free(*primary_msg);
+ g_free(wait_msg);
+ *primary_msg = combined_msg;
+ }
+ }
+ *secondary_msg = NULL;
+ *data = NULL;
+
+ return -1;
+ }
+
+ /* we got a valid message block from the child, process it */
+ switch(indicator) {
+
+ case SP_EXEC_FAILED:
+ /*
+ * Exec of dumpcap failed. Get the errno for the failure.
+ */
+ if (!ws_strtoi32(buffer, NULL, &exec_errno)) {
+ ws_warning("Invalid errno: %s", buffer);
+ }
+
+ /*
+ * Pick up the child status.
+ */
+ ret = sync_pipe_close_command(&data_pipe_read_fd, sync_pipe_read_io,
+ &fork_child, &msg);
+ if (ret == -1) {
+ /*
+ * Child process failed unexpectedly, or wait failed; msg is the
+ * error message.
+ */
+ *primary_msg = msg;
+ *secondary_msg = NULL;
+ } else {
+ /*
+ * Child process failed, but returned the expected exit status.
+ * Return the messages it gave us, and indicate failure.
+ */
+ *primary_msg = ws_strdup_printf("Couldn't run dumpcap in child process: %s",
+ g_strerror(exec_errno));
+ *secondary_msg = NULL;
+ ret = -1;
+ }
+ *data = NULL;
+ break;
+
+ case SP_ERROR_MSG:
+ /*
+ * Error from dumpcap; there will be a primary message and a
+ * secondary message.
+ */
+
+ /* convert primary message */
+ pipe_convert_header((unsigned char*)buffer, 4, &indicator, &primary_msg_len);
+ primary_msg_text = buffer+4;
+ /* convert secondary message */
+ pipe_convert_header((unsigned char*)primary_msg_text + primary_msg_len, 4, &indicator,
+ &secondary_msg_len);
+ secondary_msg_text = primary_msg_text + primary_msg_len + 4;
+ /* the capture child will close the sync_pipe, nothing to do */
+
+ /*
+ * Pick up the child status.
+ */
+ ret = sync_pipe_close_command(&data_pipe_read_fd, sync_pipe_read_io,
+ &fork_child, &msg);
+ if (ret == -1) {
+ /*
+ * Child process failed unexpectedly, or wait failed; msg is the
+ * error message.
+ */
+ *primary_msg = msg;
+ *secondary_msg = NULL;
+ } else {
+ /*
+ * Child process failed, but returned the expected exit status.
+ * Return the messages it gave us, and indicate failure.
+ */
+ *primary_msg = g_strdup(primary_msg_text);
+ *secondary_msg = g_strdup(secondary_msg_text);
+ ret = -1;
+ }
+ *data = NULL;
+ break;
+
+ case SP_SUCCESS:
+ /* read the output from the command */
+ data_buf = g_string_new("");
+ while ((count = ws_read(data_pipe_read_fd, buffer, PIPE_BUF_SIZE)) > 0) {
+ buffer[count] = '\0';
+ g_string_append(data_buf, buffer);
+ }
+
+ /*
+ * Pick up the child status.
+ */
+ ret = sync_pipe_close_command(&data_pipe_read_fd, sync_pipe_read_io,
+ &fork_child, &msg);
+ if (ret == -1) {
+ /*
+ * Child process failed unexpectedly, or wait failed; msg is the
+ * error message.
+ */
+ *primary_msg = msg;
+ *secondary_msg = NULL;
+ g_string_free(data_buf, true);
+ *data = NULL;
+ } else {
+ /*
+ * Child process succeeded.
+ */
+ *primary_msg = NULL;
+ *secondary_msg = NULL;
+ *data = g_string_free(data_buf, false);
+ }
+ break;
+
+ default:
+ /*
+ * Pick up the child status.
+ */
+ ret = sync_pipe_close_command(&data_pipe_read_fd, sync_pipe_read_io,
+ &fork_child, &msg);
+ if (ret == -1) {
+ /*
+ * Child process failed unexpectedly, or wait failed; msg is the
+ * error message.
+ */
+ *primary_msg = msg;
+ *secondary_msg = NULL;
+ } else {
+ /*
+ * Child process returned an unknown status.
+ */
+ *primary_msg = ws_strdup_printf("dumpcap process gave an unexpected message type: 0x%02x",
+ indicator);
+ *secondary_msg = NULL;
+ ret = -1;
+ }
+ *data = NULL;
+ break;
+ }
+ return ret;
+}
+
+/* centralised logging and timing for sync_pipe_run_command_actual(),
+* redirects to sync_pipe_run_command_actual()
+*/
+static int
+sync_pipe_run_command(char* const argv[], char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void))
+{
+ int ret, i;
+ int64_t start_time;
+ double elapsed;
+ int logging_enabled;
+
+ /* check if logging is actually enabled, otherwise don't expend the CPU generating logging */
+ logging_enabled = ws_log_msg_is_active(WS_LOG_DOMAIN, LOG_LEVEL_INFO);
+ if (logging_enabled) {
+ start_time = g_get_monotonic_time();
+ ws_debug("sync_pipe_run_command() starts");
+ for (i=0; argv[i] != 0; i++) {
+ ws_noisy(" argv[%d]: %s", i, argv[i]);
+ }
+ }
+ /* do the actual sync pipe run command */
+ ret = sync_pipe_run_command_actual(argv, data, primary_msg, secondary_msg, update_cb);
+
+ if (logging_enabled) {
+ elapsed = (g_get_monotonic_time() - start_time) / 1e6;
+
+ ws_debug("sync_pipe_run_command() ends, taking %.3fs, result=%d", elapsed, ret);
+
+ }
+ return ret;
+}
+
+
+int
+sync_interface_set_80211_chan(const char *iface, const char *freq, const char *type,
+ const char *center_freq1, const char *center_freq2,
+ char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void))
+{
+ int argc, ret;
+ char **argv;
+ char *opt;
+
+ argv = init_pipe_args(&argc);
+
+ if (!argv) {
+ *primary_msg = g_strdup("We don't know where to find dumpcap.");
+ *secondary_msg = NULL;
+ *data = NULL;
+ return -1;
+ }
+
+ argv = sync_pipe_add_arg(argv, &argc, "-i");
+ argv = sync_pipe_add_arg(argv, &argc, iface);
+
+ if (center_freq2)
+ opt = ws_strdup_printf("%s,%s,%s,%s", freq, type, center_freq1, center_freq2);
+ else if (center_freq1)
+ opt = ws_strdup_printf("%s,%s,%s", freq, type, center_freq1);
+ else if (type)
+ opt = ws_strdup_printf("%s,%s", freq, type);
+ else
+ opt = g_strdup(freq);
+
+ if (!opt) {
+ *primary_msg = g_strdup("Out of mem.");
+ *secondary_msg = NULL;
+ *data = NULL;
+ free_argv(argv, argc);
+ return -1;
+ }
+
+ argv = sync_pipe_add_arg(argv, &argc, "-k");
+ argv = sync_pipe_add_arg(argv, &argc, opt);
+
+#ifndef DEBUG_CHILD
+ /* Run dumpcap in capture child mode */
+ argv = sync_pipe_add_arg(argv, &argc, "-Z");
+ argv = sync_pipe_add_arg(argv, &argc, SIGNAL_PIPE_CTRL_ID_NONE);
+#endif
+
+ ret = sync_pipe_run_command(argv, data, primary_msg, secondary_msg, update_cb);
+ g_free(opt);
+ free_argv(argv, argc);
+ return ret;
+}
+
+/*
+ * Get the list of interfaces using dumpcap.
+ *
+ * On success, *data points to a buffer containing the dumpcap output,
+ * *primary_msg and *secondary_msg are NULL, and 0 is returned. *data
+ * must be freed with g_free().
+ *
+ * On failure, *data is NULL, *primary_msg points to an error message,
+ * *secondary_msg either points to an additional error message or is
+ * NULL, and -1 is returned; *primary_msg, and *secondary_msg if not NULL,
+ * must be freed with g_free().
+ */
+int
+sync_interface_list_open(char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void))
+{
+ int argc;
+ char **argv;
+ int ret;
+
+ ws_debug("sync_interface_list_open");
+
+ argv = init_pipe_args(&argc);
+
+ if (!argv) {
+ *primary_msg = g_strdup("We don't know where to find dumpcap..");
+ *secondary_msg = NULL;
+ *data = NULL;
+ return -1;
+ }
+
+ /* Ask for the interface list */
+ argv = sync_pipe_add_arg(argv, &argc, "-D");
+
+#ifndef DEBUG_CHILD
+ /* Run dumpcap in capture child mode */
+ argv = sync_pipe_add_arg(argv, &argc, "-Z");
+ argv = sync_pipe_add_arg(argv, &argc, SIGNAL_PIPE_CTRL_ID_NONE);
+#endif
+ ret = sync_pipe_run_command(argv, data, primary_msg, secondary_msg, update_cb);
+ free_argv(argv, argc);
+ return ret;
+}
+
+/*
+ * Get the capabilities of an interface using dumpcap.
+ *
+ * On success, *data points to a buffer containing the dumpcap output,
+ * *primary_msg and *secondary_msg are NULL, and 0 is returned. *data
+ * must be freed with g_free().
+ *
+ * On failure, *data is NULL, *primary_msg points to an error message,
+ * *secondary_msg either points to an additional error message or is
+ * NULL, and -1 is returned; *primary_msg, and *secondary_msg if not NULL,
+ * must be freed with g_free().
+ */
+int
+sync_if_capabilities_open(const char *ifname, bool monitor_mode, const char* auth,
+ char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void))
+{
+ int argc;
+ char **argv;
+ int ret;
+
+ ws_debug("sync_if_capabilities_open");
+
+ argv = init_pipe_args(&argc);
+
+ if (!argv) {
+ *primary_msg = g_strdup("We don't know where to find dumpcap.");
+ *secondary_msg = NULL;
+ *data = NULL;
+ return -1;
+ }
+
+ /* Ask for the interface capabilities */
+ argv = sync_pipe_add_arg(argv, &argc, "-i");
+ argv = sync_pipe_add_arg(argv, &argc, ifname);
+ argv = sync_pipe_add_arg(argv, &argc, "-L");
+ argv = sync_pipe_add_arg(argv, &argc, "--list-time-stamp-types");
+ if (monitor_mode)
+ argv = sync_pipe_add_arg(argv, &argc, "-I");
+ if (auth) {
+ argv = sync_pipe_add_arg(argv, &argc, "-A");
+ argv = sync_pipe_add_arg(argv, &argc, auth);
+ }
+
+#ifndef DEBUG_CHILD
+ /* Run dumpcap in capture child mode */
+ argv = sync_pipe_add_arg(argv, &argc, "-Z");
+ argv = sync_pipe_add_arg(argv, &argc, SIGNAL_PIPE_CTRL_ID_NONE);
+#endif
+ ret = sync_pipe_run_command(argv, data, primary_msg, secondary_msg, update_cb);
+ free_argv(argv, argc);
+ return ret;
+}
+
+/*
+ * Start getting interface statistics using dumpcap. On success, read_fd
+ * contains the file descriptor for the pipe's stdout, *msg is unchanged,
+ * and zero is returned. On failure, *msg will point to an error message
+ * that must be g_free()d, and -1 will be returned.
+ */
+int
+sync_interface_stats_open(int *data_read_fd, ws_process_id *fork_child, char **msg, void (*update_cb)(void))
+{
+ int argc;
+ char **argv;
+ int ret;
+ GIOChannel *message_read_io;
+ char *wait_msg;
+ char buffer[PIPE_BUF_SIZE+1] = {0};
+ ssize_t nread;
+ char indicator;
+ int32_t exec_errno = 0;
+ int primary_msg_len;
+ char *primary_msg_text;
+ int secondary_msg_len;
+ /*char *secondary_msg_text;*/
+ char *combined_msg;
+
+ ws_debug("sync_interface_stats_open");
+
+ argv = init_pipe_args(&argc);
+
+ if (!argv) {
+ *msg = g_strdup("We don't know where to find dumpcap.");
+ return -1;
+ }
+
+ /* Ask for the interface statistics */
+ argv = sync_pipe_add_arg(argv, &argc, "-S");
+
+#ifndef DEBUG_CHILD
+ argv = sync_pipe_add_arg(argv, &argc, "-Z");
+#ifdef _WIN32
+ ret = create_dummy_signal_pipe(msg);
+ if (ret == -1) {
+ return -1;
+ }
+ argv = sync_pipe_add_arg(argv, &argc, dummy_control_id);
+#else
+ argv = sync_pipe_add_arg(argv, &argc, SIGNAL_PIPE_CTRL_ID_NONE);
+#endif
+#endif
+ ret = sync_pipe_open_command(argv, data_read_fd, &message_read_io, NULL,
+ fork_child, NULL, msg, update_cb);
+ free_argv(argv, argc);
+ if (ret == -1) {
+ return -1;
+ }
+
+ /*
+ * We were able to set up to read dumpcap's output. Do so.
+ *
+ * First, wait for an SP_ERROR_MSG message or SP_SUCCESS message.
+ */
+ nread = pipe_read_block(message_read_io, &indicator, SP_MAX_MSG_LEN,
+ buffer, msg);
+ if(nread <= 0) {
+ /* We got a read error from the sync pipe, or we got no data at
+ all from the sync pipe, so we're not going to be getting any
+ data or error message from the child process. Pick up its
+ exit status, and complain.
+
+ We don't have to worry about killing the child, if the sync pipe
+ returned an error. Usually this error is caused as the child killed
+ itself while going down. Even in the rare cases that this isn't the
+ case, the child will get an error when writing to the broken pipe
+ the next time, cleaning itself up then. */
+ ret = sync_pipe_wait_for_child(*fork_child, &wait_msg);
+ g_io_channel_unref(message_read_io);
+ ws_close(*data_read_fd);
+ if(nread == 0) {
+ /* We got an EOF from the sync pipe. That means that it exited
+ before giving us any data to read. If ret is -1, we report
+ that as a bad exit (e.g., exiting due to a signal); otherwise,
+ we report it as a premature exit. */
+ if (ret == -1)
+ *msg = wait_msg;
+ else
+ *msg = g_strdup("Child dumpcap closed sync pipe prematurely");
+ } else {
+ /* We got an error from the sync pipe. If ret is -1, report
+ both the sync pipe I/O error and the wait error. */
+ if (ret == -1) {
+ combined_msg = ws_strdup_printf("%s\n\n%s", *msg, wait_msg);
+ g_free(*msg);
+ g_free(wait_msg);
+ *msg = combined_msg;
+ }
+ }
+ return -1;
+ }
+
+ /* we got a valid message block from the child, process it */
+ switch(indicator) {
+
+ case SP_EXEC_FAILED:
+ /*
+ * Exec of dumpcap failed. Get the errno for the failure.
+ */
+ if (!ws_strtoi32(buffer, NULL, &exec_errno)) {
+ ws_warning("Invalid errno: %s", buffer);
+ }
+ *msg = ws_strdup_printf("Couldn't run dumpcap in child process: %s",
+ g_strerror(exec_errno));
+
+ /*
+ * Pick up the child status.
+ */
+ sync_pipe_close_command(data_read_fd, message_read_io,
+ fork_child, msg);
+ ret = -1;
+ break;
+
+ case SP_ERROR_MSG:
+ /*
+ * Error from dumpcap; there will be a primary message and a
+ * secondary message.
+ */
+
+ /* convert primary message */
+ pipe_convert_header((unsigned char*)buffer, 4, &indicator, &primary_msg_len);
+ primary_msg_text = buffer+4;
+ /* convert secondary message */
+ pipe_convert_header((unsigned char*)primary_msg_text + primary_msg_len, 4, &indicator,
+ &secondary_msg_len);
+ /*secondary_msg_text = primary_msg_text + primary_msg_len + 4;*/
+ /* the capture child will close the sync_pipe, nothing to do */
+
+ /*
+ * Pick up the child status.
+ */
+ ret = sync_pipe_close_command(data_read_fd, message_read_io,
+ fork_child, msg);
+ if (ret == -1) {
+ /*
+ * Child process failed unexpectedly, or wait failed; msg is the
+ * error message.
+ */
+ } else {
+ /*
+ * Child process failed, but returned the expected exit status.
+ * Return the messages it gave us, and indicate failure.
+ */
+ *msg = g_strdup(primary_msg_text);
+ ret = -1;
+ }
+ break;
+
+ case SP_SUCCESS:
+ /* Close the message pipe. */
+ g_io_channel_unref(message_read_io);
+ break;
+
+ default:
+ /*
+ * Pick up the child status.
+ */
+ ret = sync_pipe_close_command(data_read_fd, message_read_io,
+ fork_child, msg);
+ if (ret == -1) {
+ /*
+ * Child process failed unexpectedly, or wait failed; msg is the
+ * error message.
+ */
+ } else {
+ /*
+ * Child process returned an unknown status.
+ */
+ *msg = ws_strdup_printf("dumpcap process gave an unexpected message type: 0x%02x",
+ indicator);
+ ret = -1;
+ }
+ break;
+ }
+ return ret;
+}
+
+/* Close down the stats process */
+int
+sync_interface_stats_close(int *read_fd, ws_process_id *fork_child, char **msg)
+{
+#ifdef _WIN32
+ CloseHandle(dummy_signal_pipe);
+ dummy_signal_pipe = NULL;
+#else
+ /*
+ * Don't bother waiting for the child. sync_pipe_close_command
+ * does this for us on Windows.
+ */
+ sync_pipe_kill(*fork_child);
+#endif
+ return sync_pipe_close_command(read_fd, NULL, fork_child, msg);
+}
+
+/* read a number of bytes from a pipe */
+/* (blocks until enough bytes read or an error occurs) */
+static ssize_t
+pipe_read_bytes(GIOChannel *pipe_io, char *bytes, size_t required, char **msg)
+{
+ GError *err = NULL;
+ size_t newly;
+ size_t offset = 0;
+
+ while(required) {
+ g_io_channel_read_chars(pipe_io, &bytes[offset], required, &newly, &err);
+ if (err != NULL) {
+ ws_debug("read from pipe %p: error(%u): %s", pipe_io, err->code, err->message);
+ *msg = ws_strdup_printf("Error reading from sync pipe: %s", err->message);
+ g_clear_error(&err);
+ return -1;
+ }
+ if (newly == 0) {
+ /* EOF */
+ ws_debug("read from pipe %p: EOF (capture closed?)", pipe_io);
+ *msg = 0;
+ return offset;
+ }
+
+ required -= newly;
+ offset += newly;
+ }
+
+ *msg = NULL;
+ return offset;
+}
+
+/*
+ * Read a line from a pipe; similar to fgets, but doesn't block.
+ *
+ * XXX - just stops reading if there's nothing to be read right now;
+ * that could conceivably mean that you don't get a complete line.
+ */
+int
+sync_pipe_gets_nonblock(int pipe_fd, char *bytes, int max) {
+ ssize_t newly;
+ int offset = -1;
+
+ while(offset < max - 1) {
+ offset++;
+ if (! ws_pipe_data_available(pipe_fd))
+ break;
+ newly = ws_read(pipe_fd, &bytes[offset], 1);
+ if (newly == 0) {
+ /* EOF - not necessarily an error */
+ break;
+ } else if (newly == -1) {
+ /* error */
+ ws_debug("read from pipe %d: error(%u): %s", pipe_fd, errno, g_strerror(errno));
+ return -1;
+ } else if (bytes[offset] == '\n') {
+ break;
+ }
+ }
+
+ if (offset >= 0)
+ bytes[offset] = '\0';
+
+ return offset;
+}
+
+
+/* convert header values (indicator and 3-byte length) */
+static void
+pipe_convert_header(const unsigned char *header, int header_len _U_, char *indicator, int *block_len) {
+
+ ws_assert(header_len == 4);
+
+ /* convert header values */
+ *indicator = header[0];
+ *block_len = (header[1]&0xFF)<<16 | (header[2]&0xFF)<<8 | (header[3]&0xFF);
+}
+
+/* read a message from the sending pipe in the standard format
+ (1-byte message indicator, 3-byte message length (excluding length
+ and indicator field), and the rest is the message) */
+static ssize_t
+pipe_read_block(GIOChannel *pipe_io, char *indicator, int len, char *msg,
+ char **err_msg)
+{
+ int required;
+ ssize_t newly;
+ char header[4];
+
+ /* read header (indicator and 3-byte length) */
+ newly = pipe_read_bytes(pipe_io, header, 4, err_msg);
+ if(newly != 4) {
+ if (newly == 0) {
+ /*
+ * Immediate EOF; if the capture child exits normally, this
+ * is an "I'm done" indication, so don't report it as an
+ * error.
+ */
+ ws_debug("read %p got an EOF", pipe_io);
+ return 0;
+ }
+ ws_debug("read %p failed to read header: %lu", pipe_io, (long)newly);
+ if (newly != -1) {
+ /*
+ * Short read, but not an immediate EOF.
+ */
+ *err_msg = ws_strdup_printf("Premature EOF reading from sync pipe: got only %ld bytes",
+ (long)newly);
+ }
+ return -1;
+ }
+
+ /* convert header values */
+ pipe_convert_header((unsigned char*)header, 4, indicator, &required);
+
+ /* only indicator with no value? */
+ if(required == 0) {
+ ws_debug("read %p indicator: %c empty value", pipe_io, *indicator);
+ return 4;
+ }
+
+ /* does the data fit into the given buffer? */
+ if(required > len) {
+ size_t bytes_read;
+ GError *err = NULL;
+ ws_debug("read %p length error, required %d > len %d, header: 0x%02x 0x%02x 0x%02x 0x%02x",
+ pipe_io, required, len,
+ header[0], header[1], header[2], header[3]);
+
+ /* we have a problem here, try to read some more bytes from the pipe to debug where the problem really is */
+ memcpy(msg, header, sizeof(header));
+ g_io_channel_read_chars(pipe_io, &msg[sizeof(header)], len-sizeof(header), &bytes_read, &err);
+ if (err != NULL) { /* error */
+ ws_debug("read from pipe %p: error(%u): %s", pipe_io, err->code, err->message);
+ g_clear_error(&err);
+ }
+ *err_msg = ws_strdup_printf("Unknown message from dumpcap reading header, try to show it as a string: %s",
+ msg);
+ return -1;
+ }
+ len = required;
+
+ /* read the actual block data */
+ newly = pipe_read_bytes(pipe_io, msg, required, err_msg);
+ if(newly != required) {
+ if (newly != -1) {
+ *err_msg = ws_strdup_printf("Unknown message from dumpcap reading data, try to show it as a string: %s",
+ msg);
+ }
+ return -1;
+ }
+
+ /* XXX If message is "2part", the msg probably won't be sent to debug log correctly */
+ ws_debug("read %p ok indicator: %c len: %u msg: %s", pipe_io, *indicator, len, msg);
+ *err_msg = NULL;
+ return newly + 4;
+}
+
+
+/* There's stuff to read from the sync pipe, meaning the child has sent
+ us a message, or the sync pipe has closed, meaning the child has
+ closed it (perhaps because it exited). */
+static gboolean
+sync_pipe_input_cb(GIOChannel *pipe_io, capture_session *cap_session)
+{
+ int ret;
+ char buffer[SP_MAX_MSG_LEN+1] = {0};
+ ssize_t nread;
+ char indicator;
+ int32_t exec_errno = 0;
+ int primary_len;
+ char *primary_msg;
+ int secondary_len;
+ char *secondary_msg;
+ char *wait_msg, *combined_msg;
+ uint32_t npackets = 0;
+
+ nread = pipe_read_block(pipe_io, &indicator, SP_MAX_MSG_LEN, buffer,
+ &primary_msg);
+ if(nread <= 0) {
+ /* We got a read error, or a bad message, or an EOF, from the sync pipe.
+
+ If we got a read error or a bad message, nread is -1 and
+ primary_msg is set to point to an error message. We don't
+ have to worry about killing the child; usually this error
+ is caused as the child killed itself while going down.
+ Even in the rare cases that this isn't the case, the child
+ will get an error when writing to the broken pipe the next time,
+ cleaning itself up then.
+
+ If we got an EOF, nread is 0 and primary_msg isn't set. This
+ is an indication that the capture is finished. */
+ ret = sync_pipe_wait_for_child(cap_session->fork_child, &wait_msg);
+ if(nread == 0) {
+ /* We got an EOF from the sync pipe. That means that the capture
+ child exited, and not in the middle of a message; we treat
+ that as an indication that it's done, and only report an
+ error if ret is -1, in which case wait_msg is the error
+ message. */
+ if (ret == -1)
+ primary_msg = wait_msg;
+ } else {
+ /* We got an error from the sync pipe. If ret is -1, report
+ both the sync pipe I/O error and the wait error. */
+ if (ret == -1) {
+ combined_msg = ws_strdup_printf("%s\n\n%s", primary_msg, wait_msg);
+ g_free(primary_msg);
+ g_free(wait_msg);
+ primary_msg = combined_msg;
+ }
+ }
+
+ /* No more child process. */
+ cap_session->fork_child = WS_INVALID_PID;
+ cap_session->fork_child_status = ret;
+
+#ifdef _WIN32
+ ws_close(cap_session->signal_pipe_write_fd);
+#endif
+ cap_session->capture_opts->closed_msg = primary_msg;
+ if (extcap_session_stop(cap_session)) {
+ capture_process_finished(cap_session);
+ } else {
+ extcap_request_stop(cap_session);
+ }
+ return false;
+ }
+
+ /* we got a valid message block from the child, process it */
+ switch(indicator) {
+ case SP_FILE:
+ if(!cap_session->new_file(cap_session, buffer)) {
+ ws_debug("file failed, closing capture");
+
+ /* We weren't able to open the new capture file; user has been
+ alerted. The sync pipe will close after we return false. */
+
+ /* The child has sent us a filename which we couldn't open.
+
+ This could mean that the child is creating and deleting files
+ (ring buffer mode) faster than we can handle it.
+
+ That should only be the case for very fast file switches;
+ We can't do much more than telling the child to stop.
+ (This is the "emergency brake" if the user e.g. wants to
+ switch files every second).
+
+ This can also happen if the user specified "-", meaning
+ "standard output", as the capture file. */
+ sync_pipe_stop(cap_session);
+ cap_session->closed(cap_session, NULL);
+ return false;
+ }
+ break;
+ case SP_PACKET_COUNT:
+ if (!ws_strtou32(buffer, NULL, &npackets)) {
+ ws_warning("Invalid packets number: %s", buffer);
+ }
+ ws_debug("new packets %u", npackets);
+ cap_session->count += npackets;
+ cap_session->new_packets(cap_session, npackets);
+ break;
+ case SP_EXEC_FAILED:
+ /*
+ * Exec of dumpcap failed. Get the errno for the failure.
+ */
+ if (!ws_strtoi32(buffer, NULL, &exec_errno)) {
+ ws_warning("Invalid errno: %s", buffer);
+ }
+ primary_msg = ws_strdup_printf("Couldn't run dumpcap in child process: %s",
+ g_strerror(exec_errno));
+ cap_session->error(cap_session, primary_msg, NULL);
+ /* the capture child will close the sync_pipe, nothing to do for now */
+ /* (an error message doesn't mean we have to stop capturing) */
+ break;
+ case SP_ERROR_MSG:
+ /* convert primary message */
+ pipe_convert_header((unsigned char*)buffer, 4, &indicator, &primary_len);
+ primary_msg = buffer+4;
+ /* convert secondary message */
+ pipe_convert_header((unsigned char*)primary_msg + primary_len, 4, &indicator, &secondary_len);
+ secondary_msg = primary_msg + primary_len + 4;
+ /* message output */
+ cap_session->error(cap_session, primary_msg, secondary_msg);
+ /* the capture child will close the sync_pipe, nothing to do for now */
+ /* (an error message doesn't mean we have to stop capturing) */
+ break;
+ case SP_BAD_FILTER: {
+ const char *message=NULL;
+ uint32_t indx = 0;
+ const char* end;
+
+ if (ws_strtou32(buffer, &end, &indx) && end[0] == ':') {
+ message = end + 1;
+ }
+
+ cap_session->cfilter_error(cap_session, indx, message);
+ /* the capture child will close the sync_pipe, nothing to do for now */
+ break;
+ }
+ case SP_DROPS: {
+ const char *name = NULL;
+ const char* end;
+ uint32_t num = 0;
+
+ if (ws_strtou32(buffer, &end, &num) && end[0] == ':') {
+ name = end + 1;
+ }
+
+ cap_session->drops(cap_session, num, name);
+ break;
+ }
+ default:
+ if (g_ascii_isprint(indicator))
+ ws_warning("Unknown indicator '%c'", indicator);
+ else
+ ws_warning("Unknown indicator '\\x%02x", indicator);
+ break;
+ }
+
+ return true;
+}
+
+
+
+/*
+ * dumpcap is exiting; wait for it to exit. On success, *msgp is
+ * unchanged, and the exit status of dumpcap is returned. On
+ * failure (which includes "dumpcap exited due to being killed by
+ * a signal or an exception"), *msgp points to an error message
+ * for the failure, and -1 is returned. In the latter case, *msgp
+ * must be freed with g_free().
+ */
+static int
+sync_pipe_wait_for_child(ws_process_id fork_child, char **msgp)
+{
+ int fork_child_status;
+#ifndef _WIN32
+ int retry_waitpid = 3;
+#endif
+ int ret = -1;
+ int64_t start_time;
+ double elapsed;
+
+ start_time = g_get_monotonic_time();
+
+ ws_debug("wait till child closed");
+ ws_assert(fork_child != WS_INVALID_PID);
+
+ *msgp = NULL; /* assume no error */
+#ifdef _WIN32
+ if (_cwait(&fork_child_status, (intptr_t) fork_child, _WAIT_CHILD) == -1) {
+ *msgp = ws_strdup_printf("Error from cwait(): %s", g_strerror(errno));
+ ret = -1;
+ } else {
+ /*
+ * The child exited; return its exit status. Do not treat this as
+ * an error.
+ */
+ ret = fork_child_status;
+ if ((fork_child_status & 0xC0000000) == ERROR_SEVERITY_ERROR) {
+ /* Probably an exception code */
+ *msgp = ws_strdup_printf("Child dumpcap process died: %s",
+ win32strexception(fork_child_status));
+ ret = -1;
+ }
+ }
+#else
+ while (--retry_waitpid >= 0) {
+ if (waitpid(fork_child, &fork_child_status, 0) != -1) {
+ /* waitpid() succeeded */
+ if (WIFEXITED(fork_child_status)) {
+ /*
+ * The child exited; return its exit status. Do not treat this as
+ * an error.
+ */
+ ret = WEXITSTATUS(fork_child_status);
+ } else if (WIFSTOPPED(fork_child_status)) {
+ /* It stopped, rather than exiting. "Should not happen." */
+ *msgp = ws_strdup_printf("Child dumpcap process stopped: %s",
+ sync_pipe_signame(WSTOPSIG(fork_child_status)));
+ ret = -1;
+ } else if (WIFSIGNALED(fork_child_status)) {
+ /* It died with a signal. */
+ *msgp = ws_strdup_printf("Child dumpcap process died: %s%s",
+ sync_pipe_signame(WTERMSIG(fork_child_status)),
+ WCOREDUMP(fork_child_status) ? " - core dumped" : "");
+ ret = -1;
+ } else {
+ /* What? It had to either have exited, or stopped, or died with
+ a signal; what happened here? */
+ *msgp = ws_strdup_printf("Bad status from waitpid(): %#o",
+ fork_child_status);
+ ret = -1;
+ }
+ } else {
+ /* waitpid() failed */
+ if (errno == EINTR) {
+ /*
+ * Signal interrupted waitpid().
+ *
+ * If it's SIGALRM, we just want to keep waiting, in case
+ * there's some timer using it (e.g., in a GUI toolkit).
+ *
+ * If you ^C TShark (or Wireshark), that should deliver
+ * SIGINT to dumpcap as well. dumpcap catches SIGINT,
+ * and should clean up and exit, so we should eventually
+ * see that and clean up and terminate.
+ *
+ * If we're sent a SIGTERM, we should (and do) catch it,
+ * and TShark, at least, calls sync_pipe_stop(). which
+ * kills dumpcap, so we should eventually see that and
+ * clean up and terminate.
+ */
+ ws_warning("waitpid returned EINTR. retrying.");
+ continue;
+ } else if (errno == ECHILD) {
+ /*
+ * The process identified by fork_child either doesn't
+ * exist any more or isn't our child process (anymore?).
+ *
+ * echld might have already reaped the child.
+ */
+ ret = fetch_dumpcap_pid ? 0 : -1;
+ } else {
+ /* Unknown error. */
+ *msgp = ws_strdup_printf("Error from waitpid(): %s", g_strerror(errno));
+ ret = -1;
+ }
+ }
+ break;
+ }
+#endif
+
+ elapsed = (g_get_monotonic_time() - start_time) / 1e6;
+ ws_debug("capture child closed after %.3fs", elapsed);
+ return ret;
+}
+
+
+#ifndef _WIN32
+/* convert signal to corresponding name */
+static const char *
+sync_pipe_signame(int sig)
+{
+ const char *sigmsg;
+ static char sigmsg_buf[6+1+3+1];
+
+ switch (sig) {
+
+ case SIGHUP:
+ sigmsg = "Hangup";
+ break;
+
+ case SIGINT:
+ sigmsg = "Interrupted";
+ break;
+
+ case SIGQUIT:
+ sigmsg = "Quit";
+ break;
+
+ case SIGILL:
+ sigmsg = "Illegal instruction";
+ break;
+
+ case SIGTRAP:
+ sigmsg = "Trace trap";
+ break;
+
+ case SIGABRT:
+ sigmsg = "Abort";
+ break;
+
+ case SIGFPE:
+ sigmsg = "Arithmetic exception";
+ break;
+
+ case SIGKILL:
+ sigmsg = "Killed";
+ break;
+
+ case SIGBUS:
+ sigmsg = "Bus error";
+ break;
+
+ case SIGSEGV:
+ sigmsg = "Segmentation violation";
+ break;
+
+ /* http://metalab.unc.edu/pub/Linux/docs/HOWTO/GCC-HOWTO
+ Linux is POSIX compliant. These are not POSIX-defined signals ---
+ ISO/IEC 9945-1:1990 (IEEE Std 1003.1-1990), paragraph B.3.3.1.1 sez:
+
+ ``The signals SIGBUS, SIGEMT, SIGIOT, SIGTRAP, and SIGSYS
+ were omitted from POSIX.1 because their behavior is
+ implementation dependent and could not be adequately catego-
+ rized. Conforming implementations may deliver these sig-
+ nals, but must document the circumstances under which they
+ are delivered and note any restrictions concerning their
+ delivery.''
+
+ So we only check for SIGSYS on those systems that happen to
+ implement them (a system can be POSIX-compliant and implement
+ them, it's just that POSIX doesn't *require* a POSIX-compliant
+ system to implement them).
+ */
+
+#ifdef SIGSYS
+ case SIGSYS:
+ sigmsg = "Bad system call";
+ break;
+#endif
+
+ case SIGPIPE:
+ sigmsg = "Broken pipe";
+ break;
+
+ case SIGALRM:
+ sigmsg = "Alarm clock";
+ break;
+
+ case SIGTERM:
+ sigmsg = "Terminated";
+ break;
+
+ default:
+ /* Returning a static buffer is ok in the context we use it here */
+ snprintf(sigmsg_buf, sizeof sigmsg_buf, "Signal %d", sig);
+ sigmsg = sigmsg_buf;
+ break;
+ }
+ return sigmsg;
+}
+#endif
+
+
+#ifdef _WIN32
+
+static int create_dummy_signal_pipe(char **msg) {
+ char *dummy_signal_pipe_name;
+
+ if (dummy_signal_pipe != NULL) return 0;
+
+ if (!dummy_control_id) {
+ dummy_control_id = ws_strdup_printf("%ld.dummy", GetCurrentProcessId());
+ }
+
+ /* Create the signal pipe */
+ dummy_signal_pipe_name = ws_strdup_printf(SIGNAL_PIPE_FORMAT, dummy_control_id);
+ dummy_signal_pipe = CreateNamedPipe(utf_8to16(dummy_signal_pipe_name),
+ PIPE_ACCESS_OUTBOUND, PIPE_TYPE_BYTE, 1, 65535, 65535, 0, NULL);
+ g_free(dummy_signal_pipe_name);
+ if (dummy_signal_pipe == INVALID_HANDLE_VALUE) {
+ *msg = ws_strdup_printf("Couldn't create signal pipe: %s",
+ win32strerror(GetLastError()));
+ return -1;
+ }
+ return 0;
+}
+
+/* tell the child through the signal pipe that we want to quit the capture */
+static void
+signal_pipe_capquit_to_child(capture_session *cap_session)
+{
+ const char quit_msg[] = "QUIT";
+ int ret;
+
+ ws_debug("signal_pipe_capquit_to_child");
+
+ /* it doesn't matter *what* we send here, the first byte will stop the capture */
+ /* simply sending a "QUIT" string */
+ /*sync_pipe_write_string_msg(cap_session->signal_pipe_write_fd, SP_QUIT, quit_msg);*/
+ ret = ws_write(cap_session->signal_pipe_write_fd, quit_msg, sizeof quit_msg);
+ if(ret == -1) {
+ ws_warning("%d header: error %s", cap_session->signal_pipe_write_fd, win32strerror(GetLastError()));
+ }
+}
+#endif
+
+
+/* user wants to stop the capture run */
+void
+sync_pipe_stop(capture_session *cap_session)
+{
+ if (cap_session->fork_child != WS_INVALID_PID) {
+#ifndef _WIN32
+ /* send the SIGINT signal to close the capture child gracefully. */
+ int sts = kill(cap_session->fork_child, SIGINT);
+ if (sts != 0) {
+ ws_warning("Sending SIGINT to child failed: %s\n", g_strerror(errno));
+ }
+#else
+#define STOP_SLEEP_TIME 500 /* ms */
+ DWORD status;
+
+ /* First, use the special signal pipe to try to close the capture child
+ * gracefully.
+ */
+ signal_pipe_capquit_to_child(cap_session);
+
+ /* Next, wait for the process to exit on its own */
+ status = WaitForSingleObject((HANDLE) cap_session->fork_child, STOP_SLEEP_TIME);
+
+ /* Force the issue. */
+ if (status != WAIT_OBJECT_0) {
+ ws_warning("sync_pipe_stop: forcing child to exit");
+ sync_pipe_kill(cap_session->fork_child);
+ }
+#endif
+ }
+}
+
+
+/* Wireshark has to exit, force the capture child to close */
+void
+sync_pipe_kill(ws_process_id fork_child)
+{
+ if (fork_child != WS_INVALID_PID) {
+#ifndef _WIN32
+ int sts = kill(fork_child, SIGTERM); /* SIGTERM so it can clean up if necessary */
+ if (sts != 0) {
+ ws_warning("Sending SIGTERM to child failed: %s\n", g_strerror(errno));
+ }
+#else
+ /* Remark: This is not the preferred method of closing a process!
+ * the clean way would be getting the process id of the child process,
+ * then getting window handle hWnd of that process (using EnumChildWindows),
+ * and then do a SendMessage(hWnd, WM_CLOSE, 0, 0)
+ *
+ * Unfortunately, I don't know how to get the process id from the
+ * handle. OpenProcess will get an handle (not a window handle)
+ * from the process ID; it will not get a window handle from the
+ * process ID. (How could it? A process can have more than one
+ * window. For that matter, a process might have *no* windows,
+ * as a process running dumpcap, the normal child process program,
+ * probably does.)
+ *
+ * Hint: GenerateConsoleCtrlEvent() will only work if both processes are
+ * running in the same console; that's not necessarily the case for
+ * us, as we might not be running in a console.
+ * And this also will require to have the process id.
+ */
+ TerminateProcess((HANDLE) (fork_child), 0);
+
+#endif
+ }
+}
+
+void capture_sync_set_fetch_dumpcap_pid_cb(void(*cb)(ws_process_id pid)) {
+ fetch_dumpcap_pid = cb;
+}
+
+#endif /* HAVE_LIBPCAP */
diff --git a/capture/capture_sync.h b/capture/capture_sync.h
new file mode 100644
index 00000000..7b3020cc
--- /dev/null
+++ b/capture/capture_sync.h
@@ -0,0 +1,116 @@
+/* capture_sync.h
+ * Synchronisation between Wireshark capture parent and child instances
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+
+/** @file
+ *
+ * Sync mode capture (internal interface).
+ *
+ * Will start a new Wireshark child instance which will do the actual capture
+ * work.
+ */
+
+#ifndef __CAPTURE_SYNC_H__
+#define __CAPTURE_SYNC_H__
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+struct _info_data;
+
+/**
+ * Start a new capture session.
+ * Create a capture child which is doing the real capture work.
+ * The various capture_input_... functions will be called, if something had
+ * happened.
+ *
+ * Most of the parameters are passed through the global capture_opts.
+ *
+ * @param capture_opts the options
+ * @param capture_comments if not NULL, a GPtrArray * to a set of comments
+ * to put in the capture file's Section Header Block if it's a pcapng file
+ * @param cap_session a handle for the capture session
+ * @param cap_data a struct with capture info data
+ * @param update_cb update screen
+ * @return true if a capture could be started, false if not
+ */
+extern bool
+sync_pipe_start(capture_options *capture_opts, GPtrArray *capture_comments,
+ capture_session *cap_session, struct _info_data* cap_data,
+ void(*update_cb)(void));
+
+/** User wants to stop capturing, gracefully close the capture child */
+extern void
+sync_pipe_stop(capture_session *cap_session);
+
+/** User wants to stop the program, just kill the child as soon as possible */
+extern void
+sync_pipe_kill(ws_process_id fork_child);
+
+/**
+ * Set wireless channel using dumpcap
+ * On success, *data points to a buffer containing the dumpcap output,
+ * *primary_msg and *secondary_msg are NULL, and 0 is returned. *data
+ * must be freed with g_free().
+ *
+ * On failure, *data is NULL, *primary_msg points to an error message,
+ * *secondary_msg either points to an additional error message or is
+ * NULL, and -1 or errno value is returned; *primary_msg, and
+ * *secondary_msg if not NULL must be freed with g_free().
+ *
+ * @param iface (monitor) network interface name
+ * @param freq channel control frequency string (in MHz)
+ * @param type channel type string (or NULL if not used)
+ * @param center_freq1 VHT channel center frequency (or NULL if not used)
+ * @param center_freq2 VHT channel center frequency 2 (or NULL if not used)
+ * @param data On success, *data points to a buffer containing the dumpcap output, On failure *data is NULL
+ * @param primary_msg On success NULL, On failure points to an error message
+ * @param secondary_msg On success NULL, On failure either points to an additional error message or is NULL
+ * @param update_cb update callback
+ * @return 0 on success
+ */
+extern int
+sync_interface_set_80211_chan(const char *iface, const char *freq, const char *type,
+ const char *center_freq1, const char *center_freq2,
+ char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void));
+
+/** Get an interface list using dumpcap */
+extern int
+sync_interface_list_open(char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void));
+
+/** Get interface capabilities using dumpcap */
+extern int
+sync_if_capabilities_open(const char *ifname, bool monitor_mode, const char* auth,
+ char **data, char **primary_msg,
+ char **secondary_msg, void (*update_cb)(void));
+
+/** Start getting interface statistics using dumpcap. */
+extern int
+sync_interface_stats_open(int *read_fd, ws_process_id *fork_child, char **msg, void (*update_cb)(void));
+
+/** Stop gathering statistics. */
+extern int
+sync_interface_stats_close(int *read_fd, ws_process_id *fork_child, char **msg);
+
+/** Read a line from a pipe, similar to fgets. Non-blocking. */
+extern int
+sync_pipe_gets_nonblock(int pipe_fd, char *bytes, int max);
+
+/* set a callback to be called after fork with the pid of the forked child */
+extern void capture_sync_set_fetch_dumpcap_pid_cb(void(*cb)(ws_process_id pid));
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* __CAPTURE_SYNC_H__ */
diff --git a/capture/capture_win_ifnames.c b/capture/capture_win_ifnames.c
new file mode 100644
index 00000000..25de291b
--- /dev/null
+++ b/capture/capture_win_ifnames.c
@@ -0,0 +1,269 @@
+/* capture_win_ifnames.c
+ * Routines supporting the use of Windows friendly interface names within Wireshark
+ * Copyright 2011-2012, Mike Garratt <wireshark@evn.co.nz>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+
+#ifdef _WIN32
+
+#include <wireshark.h>
+
+#include <winsock2.h>
+#include <windows.h>
+#include <iphlpapi.h>
+#include <stdio.h>
+#include <stdlib.h>
+
+#include <ntddndis.h>
+
+#ifndef NDIS_IF_MAX_STRING_SIZE
+#define NDIS_IF_MAX_STRING_SIZE IF_MAX_STRING_SIZE /* =256 in <ifdef.h> */
+#endif
+
+#ifndef NETIO_STATUS
+#define NETIO_STATUS DWORD
+#endif
+
+#include "capture/capture_ifinfo.h"
+#include "capture/capture_win_ifnames.h"
+
+#include <wsutil/file_util.h>
+
+static int gethexdigit(const char *p)
+{
+ if(*p >= '0' && *p <= '9'){
+ return *p - '0';
+ }else if(*p >= 'A' && *p <= 'F'){
+ return *p - 'A' + 0xA;
+ }else if(*p >= 'a' && *p <= 'f'){
+ return *p - 'a' + 0xa;
+ }else{
+ return -1; /* Not a hex digit */
+ }
+}
+
+static bool get8hexdigits(const char *p, DWORD *d)
+{
+ int digit;
+ DWORD val;
+ int i;
+
+ val = 0;
+ for(i = 0; i < 8; i++){
+ digit = gethexdigit(p++);
+ if(digit == -1){
+ return false; /* Not a hex digit */
+ }
+ val = (val << 4) | digit;
+ }
+ *d = val;
+ return true;
+}
+
+static bool get4hexdigits(const char *p, WORD *w)
+{
+ int digit;
+ WORD val;
+ int i;
+
+ val = 0;
+ for(i = 0; i < 4; i++){
+ digit = gethexdigit(p++);
+ if(digit == -1){
+ return false; /* Not a hex digit */
+ }
+ val = (val << 4) | digit;
+ }
+ *w = val;
+ return true;
+}
+
+/*
+ * If a string is a GUID in {}, fill in a GUID structure with the GUID
+ * value and return true; otherwise, if the string is not a valid GUID
+ * in {}, return false.
+ */
+bool
+parse_as_guid(const char *guid_text, GUID *guid)
+{
+ int i;
+ int digit1, digit2;
+
+ if(*guid_text != '{'){
+ return false; /* Nope, not enclosed in {} */
+ }
+ guid_text++;
+ /* There must be 8 hex digits; if so, they go into guid->Data1 */
+ if(!get8hexdigits(guid_text, &guid->Data1)){
+ return false; /* nope, not 8 hex digits */
+ }
+ guid_text += 8;
+ /* Now there must be a hyphen */
+ if(*guid_text != '-'){
+ return false; /* Nope */
+ }
+ guid_text++;
+ /* There must be 4 hex digits; if so, they go into guid->Data2 */
+ if(!get4hexdigits(guid_text, &guid->Data2)){
+ return false; /* nope, not 4 hex digits */
+ }
+ guid_text += 4;
+ /* Now there must be a hyphen */
+ if(*guid_text != '-'){
+ return false; /* Nope */
+ }
+ guid_text++;
+ /* There must be 4 hex digits; if so, they go into guid->Data3 */
+ if(!get4hexdigits(guid_text, &guid->Data3)){
+ return false; /* nope, not 4 hex digits */
+ }
+ guid_text += 4;
+ /* Now there must be a hyphen */
+ if(*guid_text != '-'){
+ return false; /* Nope */
+ }
+ guid_text++;
+ /*
+ * There must be 4 hex digits; if so, they go into the first 2 bytes
+ * of guid->Data4.
+ */
+ for(i = 0; i < 2; i++){
+ digit1 = gethexdigit(guid_text);
+ if(digit1 == -1){
+ return false; /* Not a hex digit */
+ }
+ guid_text++;
+ digit2 = gethexdigit(guid_text);
+ if(digit2 == -1){
+ return false; /* Not a hex digit */
+ }
+ guid_text++;
+ guid->Data4[i] = (digit1 << 4)|(digit2);
+ }
+ /* Now there must be a hyphen */
+ if(*guid_text != '-'){
+ return false; /* Nope */
+ }
+ guid_text++;
+ /*
+ * There must be 12 hex digits; if so,t hey go into the next 6 bytes
+ * of guid->Data4.
+ */
+ for(i = 0; i < 6; i++){
+ digit1 = gethexdigit(guid_text);
+ if(digit1 == -1){
+ return false; /* Not a hex digit */
+ }
+ guid_text++;
+ digit2 = gethexdigit(guid_text);
+ if(digit2 == -1){
+ return false; /* Not a hex digit */
+ }
+ guid_text++;
+ guid->Data4[i+2] = (digit1 << 4)|(digit2);
+ }
+ /* Now there must be a closing } */
+ if(*guid_text != '}'){
+ return false; /* Nope */
+ }
+ guid_text++;
+ /* And that must be the end of the string */
+ if(*guid_text != '\0'){
+ return false; /* Nope */
+ }
+ return true;
+}
+
+/**********************************************************************************/
+/* Get the friendly name for the given GUID */
+char *
+get_interface_friendly_name_from_device_guid(__in GUID *guid)
+{
+ HRESULT hr;
+
+ /* Need to convert an Interface GUID to the interface friendly name (e.g. "Local Area Connection")
+ * The functions required to do this all reside within iphlpapi.dll
+ */
+
+ NET_LUID InterfaceLuid;
+ hr = ConvertInterfaceGuidToLuid(guid, &InterfaceLuid);
+ if(hr == NO_ERROR) {
+ /* guid->luid success */
+ WCHAR wName[NDIS_IF_MAX_STRING_SIZE + 1];
+ hr = ConvertInterfaceLuidToAlias(&InterfaceLuid, wName, NDIS_IF_MAX_STRING_SIZE+1);
+ if(hr == NO_ERROR) {
+ /* luid->friendly name success */
+
+ /* Get the required buffer size, and then convert the string
+ * from UTF-16 to UTF-8. */
+ int size;
+ char *name;
+ size = WideCharToMultiByte(CP_UTF8, 0, wName, -1, NULL, 0, NULL, NULL);
+ if(size != 0) {
+ name = (char *) g_malloc(size);
+ if (name != NULL) {
+ size = WideCharToMultiByte(CP_UTF8, 0, wName, -1, name, size, NULL, NULL);
+ if(size != 0) {
+ return name;
+ }
+ /* Failed, clean up the allocation */
+ g_free(name);
+ }
+ }
+ }
+ }
+
+ /* Failed to get a name */
+ return NULL;
+}
+
+/*
+ * Given an interface name, try to extract the GUID from it and parse it.
+ * If that fails, return NULL; if that succeeds, attempt to get the
+ * friendly name for the interface in question. If that fails, return
+ * NULL, otherwise return the friendly name, allocated with g_malloc()
+ * (so that it must be freed with g_free()).
+ */
+char *
+get_windows_interface_friendly_name(const char *interface_devicename)
+{
+ const char* guid_text;
+ GUID guid;
+
+ /* Extract the guid text from the interface device name */
+ if(strncmp("\\Device\\NPF_", interface_devicename, 12)==0){
+ guid_text=interface_devicename+12; /* skip over the '\Device\NPF_' prefix, assume the rest is the guid text */
+ }else{
+ guid_text=interface_devicename;
+ }
+
+ if (!parse_as_guid(guid_text, &guid)){
+ return NULL; /* not a GUID, so no friendly name */
+ }
+
+ /* guid okay, get the interface friendly name associated with the guid */
+ return get_interface_friendly_name_from_device_guid(&guid);
+}
+
+/**************************************************************************************/
+#endif
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 4
+ * tab-width: 8
+ * indent-tabs-mode: nil
+ * End:
+ *
+ * vi: set shiftwidth=4 tabstop=8 expandtab:
+ * :indentSize=4:tabSize=8:noTabs=true:
+ */
diff --git a/capture/capture_win_ifnames.h b/capture/capture_win_ifnames.h
new file mode 100644
index 00000000..9161bcf5
--- /dev/null
+++ b/capture/capture_win_ifnames.h
@@ -0,0 +1,35 @@
+/** @file
+ *
+ * Routines supporting the use of Windows friendly interface names within Wireshark
+ * Copyright 2011-2012, Mike Garratt <wireshark@evn.co.nz>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef CAPTURE_WIN_IFNAMES_H
+#define CAPTURE_WIN_IFNAMES_H
+
+/*
+ * If a string is a GUID in {}, fill in a GUID structure with the GUID
+ * value and return true; otherwise, if the string is not a valid GUID
+ * in {}, return false.
+ */
+extern bool parse_as_guid(const char *guid_text, GUID *guid);
+
+/* Get the friendly name for the given GUID */
+extern char *get_interface_friendly_name_from_device_guid(GUID *guid);
+
+/*
+ * Given an interface name, try to extract the GUID from it and parse it.
+ * If that fails, return NULL; if that succeeds, attempt to get the
+ * friendly name for the interface in question. If that fails, return
+ * NULL, otherwise return the friendly name, allocated with g_malloc()
+ * (so that it must be freed with g_free()).
+ */
+extern char *get_windows_interface_friendly_name(const char *interface_devicename);
+
+#endif
diff --git a/capture/iface_monitor.c b/capture/iface_monitor.c
new file mode 100644
index 00000000..f742a664
--- /dev/null
+++ b/capture/iface_monitor.c
@@ -0,0 +1,396 @@
+/* iface_monitor.c
+ * interface monitor by Pontus Fuchs <pontus.fuchs@gmail.com>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#include "config.h"
+#include "iface_monitor.h"
+
+#ifdef HAVE_LIBPCAP
+
+#if defined(HAVE_LIBNL)
+
+/*
+ * Linux with libnl.
+ */
+
+#include <stdio.h>
+#include <string.h>
+#include <errno.h>
+
+DIAG_OFF_PEDANTIC
+#include <netlink/msg.h>
+DIAG_ON_PEDANTIC
+#include <netlink/attr.h>
+DIAG_OFF_PEDANTIC
+#include <netlink/route/link.h>
+DIAG_ON_PEDANTIC
+
+#ifndef IFF_UP
+/*
+ * Apparently, some versions of libnl drag in headers that define IFF_UP
+ * and others don't. Include <net/if.h> iff IFF_UP isn't already defined,
+ * so that if <linux/if.h> has been included by some or all of the
+ * netlink headers, we don't include <net/if.h> and get a bunch of
+ * complaints about various structures being redefined.
+ */
+#include <net/if.h>
+#endif
+
+/* libnl 1.x compatibility code */
+#ifdef HAVE_LIBNL1
+#define nl_sock nl_handle
+#define nl_socket_disable_seq_check nl_disable_sequence_check
+
+static inline struct nl_handle *nl_socket_alloc(void)
+{
+ return nl_handle_alloc();
+}
+
+static inline void nl_socket_free(struct nl_sock *h)
+{
+ nl_handle_destroy(h);
+}
+#endif /* HAVE_LIBNL1 */
+
+static struct nl_sock *iface_mon_sock;
+
+static void
+iface_mon_handler2(struct nl_object *obj, void *arg)
+{
+ struct rtnl_link *filter;
+ struct rtnl_link *link_obj;
+ int flags, up;
+ char *ifname;
+ iface_mon_cb cb = (iface_mon_cb)arg;
+
+ filter = rtnl_link_alloc();
+ if (!filter) {
+ fprintf(stderr, "error allocating filter\n");
+ return;
+ }
+
+ if (nl_object_match_filter (obj, OBJ_CAST (filter)) == 0) {
+ rtnl_link_put(filter);
+ return;
+ }
+
+ link_obj = (struct rtnl_link *) obj;
+ flags = rtnl_link_get_flags (link_obj);
+ ifname = rtnl_link_get_name(link_obj);
+
+ /*
+ * You can't bind a PF_PACKET socket to an interface that's not
+ * up, so an interface going down is an "interface should be
+ * removed" indication.
+ *
+ * XXX - what indication, if any, do we get if the interface
+ * *completely goes away*?
+ *
+ * XXX - can we get events if an interface's link-layer or
+ * network addresses change?
+ */
+ up = (flags & IFF_UP) ? 1 : 0;
+
+#ifdef HAVE_LIBNL1
+ cb(ifname, 0, up);
+#else
+ int msg_type = nl_object_get_msgtype(obj);
+
+ switch (msg_type) {
+ case RTM_NEWLINK:
+ cb(ifname, 1, up);
+ break;
+ case RTM_DELLINK:
+ cb(ifname, 0, 0);
+ break;
+ default:
+ /* Ignore other events */
+ break;
+ }
+#endif
+
+ rtnl_link_put(filter);
+
+ return;
+}
+
+static int
+iface_mon_handler(struct nl_msg *msg, void *arg)
+{
+ nl_msg_parse (msg, &iface_mon_handler2, arg);
+ return 0;
+}
+
+void
+iface_mon_event(void)
+{
+ nl_recvmsgs_default(iface_mon_sock);
+}
+
+int
+iface_mon_get_sock(void)
+{
+ return nl_socket_get_fd(iface_mon_sock);
+}
+
+int
+iface_mon_start(iface_mon_cb cb)
+{
+ int err;
+
+ iface_mon_sock = nl_socket_alloc();
+ if (!iface_mon_sock) {
+ fprintf(stderr, "Failed to allocate netlink socket.\n");
+ return -ENOMEM;
+ }
+
+ nl_socket_disable_seq_check(iface_mon_sock);
+
+ nl_socket_modify_cb(iface_mon_sock, NL_CB_VALID, NL_CB_CUSTOM, iface_mon_handler, (void *)cb);
+
+ if (nl_connect(iface_mon_sock, NETLINK_ROUTE)) {
+ fprintf(stderr, "Failed to connect to generic netlink.\n");
+ err = -ENOLINK;
+ goto out_handle_destroy;
+ }
+
+ nl_socket_add_membership(iface_mon_sock, RTNLGRP_LINK);
+
+ return 0;
+
+out_handle_destroy:
+ nl_socket_free(iface_mon_sock);
+ return err;
+}
+
+void
+iface_mon_stop(void)
+{
+ if(iface_mon_sock)
+ nl_socket_free(iface_mon_sock);
+ iface_mon_sock = NULL;
+}
+
+#elif defined(__APPLE__)
+
+/*
+ * macOS.
+ */
+
+#include <stddef.h>
+#include <stdio.h>
+#include <errno.h>
+#include <unistd.h>
+
+#include <sys/socket.h>
+#include <sys/ioctl.h>
+#include <sys/types.h>
+#include <net/if.h>
+#include <sys/kern_event.h>
+
+#include <glib.h>
+
+static int s;
+static iface_mon_cb callback;
+
+int
+iface_mon_start(iface_mon_cb cb)
+{
+ int ret;
+ struct kev_request key;
+
+ /* Create a socket of type PF_SYSTEM to listen for events. */
+ s = socket(PF_SYSTEM, SOCK_RAW, SYSPROTO_EVENT);
+ if (s == -1)
+ return -errno;
+
+ /*
+ * Ask for DLIL messages.
+ *
+ * XXX - also ask for KEV_INET_SUBCLASS and KEV_INET6_SUBCLASS,
+ * to detect new or changed network addresses, so those can be
+ * updated as well? Can we specify multiple filters on a socket,
+ * or must we specify KEV_ANY_SUBCLASS and filter the events after
+ * receiving them?
+ */
+ key.vendor_code = KEV_VENDOR_APPLE;
+ key.kev_class = KEV_NETWORK_CLASS;
+ key.kev_subclass = KEV_DL_SUBCLASS;
+ if (ioctl(s, SIOCSKEVFILT, &key) == -1) {
+ ret = -errno;
+ close(s);
+ return ret;
+ }
+
+ callback = cb;
+ return 0;
+}
+
+void
+iface_mon_stop(void)
+{
+ close(s);
+}
+
+int
+iface_mon_get_sock(void)
+{
+ return s;
+}
+
+/*
+ * Size of buffer for kernel network event.
+ */
+#define NET_EVENT_DATA_SIZE (KEV_MSG_HEADER_SIZE + sizeof (struct net_event_data))
+
+void
+iface_mon_event(void)
+{
+ char msg[NET_EVENT_DATA_SIZE];
+ ssize_t received;
+ struct kern_event_msg *kem;
+ struct net_event_data *evd;
+ size_t evd_len;
+ char ifr_name[IFNAMSIZ];
+
+ received = recv(s, msg, sizeof msg, 0);
+ if (received < 0) {
+ /* Error - ignore. */
+ return;
+ }
+ if ((size_t)received < sizeof msg) {
+ /* Short read - ignore. */
+ return;
+ }
+ kem = (struct kern_event_msg *)msg;
+ evd_len = kem->total_size - KEV_MSG_HEADER_SIZE;
+ if (evd_len != sizeof (struct net_event_data)) {
+ /* Length of the message is bogus. */
+ return;
+ }
+ evd = (struct net_event_data *)&kem->event_data[0];
+ snprintf(ifr_name, IFNAMSIZ, "%s%u", evd->if_name, evd->if_unit);
+
+ /*
+ * Check type of event.
+ *
+ * Note: if we also ask for KEV_INET_SUBCLASS, we will get
+ * events with keys
+ *
+ * KEV_INET_NEW_ADDR
+ * KEV_INET_CHANGED_ADDR
+ * KEV_INET_CHANGED_ADDR
+ * KEV_INET_SIFDSTADDR
+ * KEV_INET_SIFBRDADDR
+ * KEV_INET_SIFNETMASK
+ *
+ * reflecting network address changes, with the data being a
+ * struct kev_in_data rather than struct net_event_data, and
+ * if we also ask for KEV_INET6_SUBCLASS, we will get events
+ * with keys
+ *
+ * KEV_INET6_NEW_LL_ADDR
+ * KEV_INET6_NEW_USER_ADDR
+ * KEV_INET6_NEW_RTADV_ADDR
+ * KEV_INET6_ADDR_DELETED
+ *
+ * with the data being a struct kev_in6_data.
+ */
+ switch (kem->event_code) {
+
+ case KEV_DL_IF_ATTACHED:
+ /*
+ * A new interface has arrived.
+ *
+ * XXX - what we really want is "a new BPFable interface
+ * has arrived", but that's not available. While we're
+ * asking for additional help from BPF, it'd also be
+ * nice if we could ask it for a list of all interfaces
+ * that have had bpfattach()/bpf_attach() done on them,
+ * so we don't have to try to open the device in order
+ * to see whether we should show it as something on
+ * which we can capture.
+ */
+ callback(ifr_name, 1, 1);
+ break;
+
+ case KEV_DL_IF_DETACHED:
+ /*
+ * An existing interface has been removed.
+ *
+ * XXX - use KEV_DL_IF_DETACHING instead, as that's
+ * called shortly after bpfdetach() is called, and
+ * bpfdetach() makes an interface no longer BPFable,
+ * and that's what we *really* care about.
+ */
+ callback(ifr_name, 0, 0);
+ break;
+
+ default:
+ /*
+ * Is there any reason to care about:
+ *
+ * KEV_DL_LINK_ON
+ * KEV_DL_LINK_OFF
+ * KEV_DL_SIFFLAGS
+ * KEV_DL_LINK_ADDRESS_CHANGED
+ * KEV_DL_IFCAP_CHANGED
+ *
+ * or any of the other events? On Snow Leopard and, I think,
+ * earlier releases, you can't attach a BPF device to an
+ * interface that's not up, so KEV_DL_SIFFLAGS might be
+ * worth listening to so that we only say "here's a new
+ * interface" when it goes up; on Lion (and possibly Mountain
+ * Lion), an interface doesn't have to be up in order to
+ * have a BPF device attached to it.
+ */
+ break;
+ }
+}
+
+#else /* don't have something we support */
+
+int
+iface_mon_start(iface_mon_cb cb _U_)
+{
+ return -1;
+}
+
+void
+iface_mon_stop(void)
+{
+}
+
+int
+iface_mon_get_sock(void)
+{
+ return -1;
+}
+
+void
+iface_mon_event(void)
+{
+}
+
+#endif /* HAVE_LIBNL */
+
+#endif /* HAVE_LIBPCAP */
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 4
+ * tab-width: 8
+ * indent-tabs-mode: nil
+ * End:
+ *
+ * vi: set shiftwidth=4 tabstop=8 expandtab:
+ * :indentSize=4:tabSize=8:noTabs=true:
+ */
diff --git a/capture/iface_monitor.h b/capture/iface_monitor.h
new file mode 100644
index 00000000..61a7ea1b
--- /dev/null
+++ b/capture/iface_monitor.h
@@ -0,0 +1,74 @@
+/** @file
+ *
+ * interface monitor by Pontus Fuchs <pontus.fuchs@gmail.com>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+#ifndef IFACE_MONITOR_H
+#define IFACE_MONITOR_H
+
+#include <wireshark.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+#ifdef HAVE_LIBPCAP
+
+/*
+ * Callback for interface changes.
+ *
+ * iface is a pointer to the name of the interface.
+ *
+ * up is 1 if the interface is up, 0 if it's down.
+ *
+ * XXX - we really want "gone", not "down", where "gone" may include
+ * "down" if the OS requires an interface to be up in order to start
+ * a capture on it (as is the case in Linux and in macOS prior to
+ * Lion), but should also include *gone*, as in "there is no longer
+ * an interface with this name, so it's neither down nor up".
+ *
+ * We also may want other events, such as address changes, so what
+ * we might want is "add", "remove", and "modify" as the events.
+ */
+typedef void (*iface_mon_cb)(const char *iface, int added, int up);
+
+/*
+ * Start watching for interface changes.
+ */
+int
+iface_mon_start(iface_mon_cb cb);
+
+/*
+ * Stop watching for interface changes.
+ */
+void
+iface_mon_stop(void);
+
+/*
+ * Get the socket on which interface changes are delivered, so that
+ * we can add it to the event loop.
+ *
+ * XXX - what if it's not a socket or other file descriptor?
+ */
+int
+iface_mon_get_sock(void);
+
+/*
+ * Call this if something is readable from the interface change socket.
+ * It will call the callback as appropriate.
+ */
+void
+iface_mon_event(void);
+
+#endif /* HAVE_LIBPCAP */
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* IFACE_MONITOR_H */
diff --git a/capture/ws80211_utils.c b/capture/ws80211_utils.c
new file mode 100644
index 00000000..2211f80c
--- /dev/null
+++ b/capture/ws80211_utils.c
@@ -0,0 +1,1265 @@
+/*
+ * ws80211 utilities
+ * Copyright 2012, Pontus Fuchs <pontus.fuchs@gmail.com>
+
+Parts of this file was copied from iw:
+
+Copyright (c) 2007, 2008 Johannes Berg
+Copyright (c) 2007 Andy Lutomirski
+Copyright (c) 2007 Mike Kershaw
+Copyright (c) 2008-2009 Luis R. Rodriguez
+
+SPDX-License-Identifier: ISC
+*/
+
+#include "config.h"
+#include "ws80211_utils.h"
+
+#include <stdio.h>
+
+#include <glib.h>
+#include <glib/gstdio.h>
+
+#if defined(HAVE_LIBNL) && defined(HAVE_NL80211)
+#include <string.h>
+#include <errno.h>
+#include <unistd.h>
+
+#include <net/if.h>
+#include <sys/ioctl.h>
+
+#include <netlink/genl/genl.h>
+#include <netlink/genl/family.h>
+#include <netlink/genl/ctrl.h>
+#include <netlink/msg.h>
+#include <netlink/attr.h>
+
+#include <linux/nl80211.h>
+
+#ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP
+static int ws80211_get_protocol_features(int* features);
+#endif /* HAVE_NL80211_SPLIT_WIPHY_DUMP */
+
+/* libnl 1.x compatibility code */
+#ifdef HAVE_LIBNL1
+#define nl_sock nl_handle
+static inline struct nl_handle *nl_socket_alloc(void)
+{
+ return nl_handle_alloc();
+}
+
+static inline void nl_socket_free(struct nl_sock *h)
+{
+ nl_handle_destroy(h);
+}
+#endif /* HAVE_LIBNL1 */
+
+struct nl80211_state {
+ struct nl_sock *nl_sock;
+ int nl80211_id;
+ int have_split_wiphy;
+};
+
+static struct nl80211_state nl_state;
+
+int ws80211_init(void)
+{
+ int err;
+#ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP
+ int features = 0;
+#endif /* HAVE_NL80211_SPLIT_WIPHY_DUMP */
+
+ struct nl80211_state *state = &nl_state;
+
+ state->nl_sock = nl_socket_alloc();
+ if (!state->nl_sock) {
+ fprintf(stderr, "Failed to allocate netlink socket.\n");
+ return -ENOMEM;
+ }
+
+ if (genl_connect(state->nl_sock)) {
+ fprintf(stderr, "Failed to connect to generic netlink.\n");
+ err = -ENOLINK;
+ goto out_handle_destroy;
+ }
+
+ state->nl80211_id = genl_ctrl_resolve(state->nl_sock, "nl80211");
+ if (state->nl80211_id < 0) {
+ fprintf(stderr, "nl80211 not found.\n");
+ err = -ENOENT;
+ goto out_handle_destroy;
+ }
+#ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP
+ ws80211_get_protocol_features(&features);
+ if (features & NL80211_PROTOCOL_FEATURE_SPLIT_WIPHY_DUMP)
+ state->have_split_wiphy = true;
+#endif /* HAVE_NL80211_SPLIT_WIPHY_DUMP */
+
+ return WS80211_INIT_OK;
+
+ out_handle_destroy:
+ nl_socket_free(state->nl_sock);
+ state->nl_sock = 0;
+ return err;
+}
+
+static int error_handler(struct sockaddr_nl *nla _U_, struct nlmsgerr *err,
+ void *arg)
+{
+ int *ret = (int *)arg;
+ *ret = err->error;
+ return NL_STOP;
+}
+
+static int finish_handler(struct nl_msg *msg _U_, void *arg)
+{
+ int *ret = (int *)arg;
+ *ret = 0;
+ return NL_SKIP;
+}
+
+static int ack_handler(struct nl_msg *msg _U_, void *arg)
+{
+ int *ret = (int *)arg;
+ *ret = 0;
+ return NL_STOP;
+}
+
+static int nl80211_do_cmd(struct nl_msg *msg, struct nl_cb *cb)
+{
+ /*
+ * XXX - Coverity doesn't understand how libnl works, so it
+ * doesn't know that nl_recvmsgs() calls the callback, and
+ * that the callback has had a pointer to err registered
+ * with it, and therefore that nl_recvmsgs() can change
+ * err as a side-effect, so it thinks this can loop
+ * infinitely.
+ *
+ * The proper way to address this is to help Coverity to
+ * understand the behaviour of nl_recvmsgs(), in that it
+ * does call the callback, setting err. This help would be
+ * provided through a so called 'model' of this function.
+ * We declare err to be volatile to work around it.
+ *
+ * XXX - that workaround provokes a compiler complaint that
+ * casting a pointer to it to "void *" discards the
+ * volatile qualifier. Perhaps we should just re-close
+ * Coverity CID 997052 as "false positive".
+ */
+ volatile int err;
+
+ if (!nl_state.nl_sock)
+ return -ENOLINK;
+
+ err = nl_send_auto_complete(nl_state.nl_sock, msg);
+ if (err < 0)
+ goto out;
+
+ err = 1;
+
+ nl_cb_err(cb, NL_CB_CUSTOM, error_handler, (void *)&err);
+ nl_cb_set(cb, NL_CB_FINISH, NL_CB_CUSTOM, finish_handler, (void *)&err);
+ nl_cb_set(cb, NL_CB_ACK, NL_CB_CUSTOM, ack_handler, (void *)&err);
+
+ while (err > 0)
+ nl_recvmsgs(nl_state.nl_sock, cb);
+ out:
+ nl_cb_put(cb);
+
+ return err;
+}
+
+struct nliface_cookie
+{
+ char *ifname;
+ GArray *interfaces;
+};
+
+static struct ws80211_interface *
+ get_interface_by_name(GArray *interfaces,
+ char* ifname)
+{
+ unsigned int i;
+ struct ws80211_interface *iface;
+
+ for (i = 0; i < interfaces->len; i++) {
+ iface = g_array_index(interfaces, struct ws80211_interface *, i);
+ if (!strcmp(iface->ifname, ifname))
+ return iface;
+ }
+ return NULL;
+}
+
+#ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP
+static int get_features_handler(struct nl_msg *msg, void *arg)
+{
+ int *feat = (int*) arg;
+ struct nlattr *tb_msg[NL80211_ATTR_MAX + 1];
+ struct genlmsghdr *gnlh = (struct genlmsghdr *)nlmsg_data(nlmsg_hdr(msg));
+
+ nla_parse(tb_msg, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
+ genlmsg_attrlen(gnlh, 0), NULL);
+
+ if (tb_msg[NL80211_ATTR_PROTOCOL_FEATURES])
+ *feat = nla_get_u32(tb_msg[NL80211_ATTR_PROTOCOL_FEATURES]);
+
+ return NL_SKIP;
+}
+
+static int ws80211_get_protocol_features(int* features)
+{
+ struct nl_msg *msg;
+ struct nl_cb *cb;
+ int ret;
+
+ msg = nlmsg_alloc();
+ if (!msg) {
+ fprintf(stderr, "failed to allocate netlink message\n");
+ return 2;
+ }
+
+ cb = nl_cb_alloc(NL_CB_DEFAULT);
+
+ genlmsg_put(msg, 0, 0, nl_state.nl80211_id, 0, 0,
+ NL80211_CMD_GET_PROTOCOL_FEATURES, 0);
+
+ nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, get_features_handler, features);
+
+ ret = nl80211_do_cmd(msg, cb);
+ nlmsg_free(msg);
+ return ret;
+}
+#endif /* HAVE_NL80211_SPLIT_WIPHY_DUMP */
+
+#ifdef NL80211_BAND_ATTR_HT_CAPA
+static void parse_band_ht_capa(struct ws80211_interface *iface,
+ struct nlattr *tb)
+{
+ bool ht40;
+
+ if (!tb) return;
+
+ iface->channel_types |= 1 << WS80211_CHAN_HT20;
+ ht40 = !!(nla_get_u16(tb) & 0x02);
+ if (ht40) {
+ iface->channel_types |= 1 << WS80211_CHAN_HT40MINUS;
+ iface->channel_types |= 1 << WS80211_CHAN_HT40PLUS;
+ }
+}
+#endif /* NL80211_BAND_ATTR_HT_CAPA */
+
+#ifdef HAVE_NL80211_VHT_CAPABILITY
+static void parse_band_vht_capa(struct ws80211_interface *iface,
+ struct nlattr *tb)
+{
+ uint32_t chan_capa;
+ if (!tb) return;
+
+ chan_capa = (nla_get_u32(tb) >> 2) & 3;
+ if (chan_capa == 1) {
+ iface->channel_types |= 1 << WS80211_CHAN_VHT160;
+ }
+ if (chan_capa == 2) {
+ iface->channel_types |= 1 << WS80211_CHAN_VHT160;
+ iface->channel_types |= 1 << WS80211_CHAN_VHT80P80;
+ }
+ iface->channel_types |= 1 << WS80211_CHAN_VHT80;
+}
+#endif /* HAVE_NL80211_VHT_CAPABILITY */
+
+static void parse_supported_iftypes(struct ws80211_interface *iface,
+ struct nlattr *tb)
+{
+ struct nlattr *nl_mode;
+ int rem_mode;
+
+ if (!tb) return;
+
+ nla_for_each_nested(nl_mode, tb, rem_mode) {
+ if (nla_type(nl_mode) == NL80211_IFTYPE_MONITOR)
+ iface->cap_monitor = 1;
+ }
+}
+
+static void parse_band_freqs(struct ws80211_interface *iface,
+ struct nlattr *tb)
+{
+ struct nlattr *nl_freq;
+ struct nlattr *tb_freq[NL80211_FREQUENCY_ATTR_MAX + 1];
+ static struct nla_policy freq_policy[NL80211_FREQUENCY_ATTR_MAX + 1] = {
+ {NLA_UNSPEC, 0, 0}, /* __NL80211_FREQUENCY_ATTR_INVALID */
+ {NLA_U32, 0, 0}, /* NL80211_FREQUENCY_ATTR_FREQ */
+ {NLA_FLAG, 0, 0}, /* NL80211_FREQUENCY_ATTR_DISABLED */
+ {NLA_FLAG, 0, 0}, /* NL80211_FREQUENCY_ATTR_PASSIVE_SCAN */
+ {NLA_FLAG, 0, 0}, /* NL80211_FREQUENCY_ATTR_NO_IBSS */
+ {NLA_FLAG, 0, 0}, /* NL80211_FREQUENCY_ATTR_RADAR */
+ {NLA_U32, 0, 0} /* NL80211_FREQUENCY_ATTR_MAX_TX_POWER */
+ };
+ int rem_freq;
+
+ if (!tb) return;
+
+ nla_for_each_nested(nl_freq, tb, rem_freq) {
+ uint32_t freq;
+ nla_parse(tb_freq, NL80211_FREQUENCY_ATTR_MAX,
+ (struct nlattr *)nla_data(nl_freq),
+ nla_len(nl_freq), freq_policy);
+ if (!tb_freq[NL80211_FREQUENCY_ATTR_FREQ])
+ continue;
+ if (tb_freq[NL80211_FREQUENCY_ATTR_DISABLED])
+ continue;
+
+ freq = nla_get_u32(tb_freq[NL80211_FREQUENCY_ATTR_FREQ]);
+ g_array_append_val(iface->frequencies, freq);
+ }
+}
+
+static void parse_wiphy_bands(struct ws80211_interface *iface,
+ struct nlattr *tb)
+{
+ struct nlattr *nl_band;
+ struct nlattr *tb_band[NL80211_BAND_ATTR_MAX + 1];
+ int rem_band;
+
+ if (!tb) return;
+
+ nla_for_each_nested(nl_band, tb, rem_band) {
+ nla_parse(tb_band, NL80211_BAND_ATTR_MAX,
+ (struct nlattr *)nla_data(nl_band),
+ nla_len(nl_band), NULL);
+
+#ifdef NL80211_BAND_ATTR_HT_CAPA
+ parse_band_ht_capa(iface, tb_band[NL80211_BAND_ATTR_HT_CAPA]);
+#endif /* NL80211_BAND_ATTR_HT_CAPA */
+#ifdef HAVE_NL80211_VHT_CAPABILITY
+ parse_band_vht_capa(iface, tb_band[NL80211_BAND_ATTR_VHT_CAPA]);
+#endif /* HAVE_NL80211_VHT_CAPABILITY */
+ parse_band_freqs(iface, tb_band[NL80211_BAND_ATTR_FREQS]);
+ }
+}
+
+static void parse_supported_commands(struct ws80211_interface *iface,
+ struct nlattr *tb)
+{
+ /* Can frequency be set? Only newer versions of cfg80211 supports this */
+#ifdef HAVE_NL80211_CMD_SET_CHANNEL
+ int cmd;
+ struct nlattr *nl_cmd;
+
+ if (!tb) return;
+
+ nla_for_each_nested(nl_cmd, tb, cmd) {
+ if(nla_get_u32(nl_cmd) == NL80211_CMD_SET_CHANNEL)
+ iface->can_set_freq = true;
+ }
+#else
+ iface->can_set_freq = true;
+#endif
+}
+
+static int get_phys_handler(struct nl_msg *msg, void *arg)
+{
+ struct nlattr *tb_msg[NL80211_ATTR_MAX + 1];
+ struct genlmsghdr *gnlh = (struct genlmsghdr *)nlmsg_data(nlmsg_hdr(msg));
+
+ struct nliface_cookie *cookie = (struct nliface_cookie *)arg;
+
+ struct ws80211_interface *iface;
+ char* ifname;
+ int added = 0;
+
+ nla_parse(tb_msg, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
+ genlmsg_attrlen(gnlh, 0), NULL);
+
+ if (!tb_msg[NL80211_ATTR_WIPHY_NAME])
+ return NL_SKIP;
+
+ ifname = ws_strdup_printf("%s.mon", nla_get_string(tb_msg[NL80211_ATTR_WIPHY_NAME]));
+ iface = get_interface_by_name(cookie->interfaces, ifname);
+
+ if (!iface) {
+ iface = (struct ws80211_interface *)g_malloc0(sizeof(*iface));
+ if (!iface) {
+ g_free(ifname);
+ return NL_SKIP;
+ }
+ added = 1;
+ iface->ifname = ifname;
+ iface->frequencies = g_array_new(false, false, sizeof(uint32_t));
+ iface->channel_types = 1 << WS80211_CHAN_NO_HT;
+ } else {
+ g_free(ifname);
+ }
+
+ parse_supported_iftypes(iface, tb_msg[NL80211_ATTR_SUPPORTED_IFTYPES]);
+ parse_wiphy_bands(iface, tb_msg[NL80211_ATTR_WIPHY_BANDS]);
+ parse_supported_commands(iface, tb_msg[NL80211_ATTR_SUPPORTED_COMMANDS]);
+
+ if (added)
+ g_array_append_val(cookie->interfaces, iface);
+
+ return NL_SKIP;
+}
+
+static int ws80211_get_phys(GArray *interfaces)
+{
+ struct nliface_cookie cookie;
+ struct nl_msg *msg;
+ struct nl_cb *cb;
+ int ret;
+ msg = nlmsg_alloc();
+ if (!msg) {
+ fprintf(stderr, "failed to allocate netlink message\n");
+ return 2;
+ }
+
+ cb = nl_cb_alloc(NL_CB_DEFAULT);
+
+ cookie.interfaces = interfaces;
+
+ genlmsg_put(msg, 0, 0, nl_state.nl80211_id, 0,
+ NLM_F_DUMP, NL80211_CMD_GET_WIPHY, 0);
+
+#ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP
+ if (nl_state.have_split_wiphy) {
+ NLA_PUT_FLAG(msg, NL80211_ATTR_SPLIT_WIPHY_DUMP);
+ }
+#endif /* #ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP */
+ nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, get_phys_handler, &cookie);
+
+ ret = nl80211_do_cmd(msg, cb);
+ nlmsg_free(msg);
+ return ret;
+
+#ifdef HAVE_NL80211_SPLIT_WIPHY_DUMP
+nla_put_failure:
+ nlmsg_free(msg);
+ fprintf(stderr, "building message failed\n");
+ return -1;
+#endif /* HAVE_NL80211_SPLIT_WIPHY_DUMP */
+}
+
+static int get_freq_wext(const char *ifname)
+{
+ int fd;
+ int ret = -1;
+ /* Ugly hack to avoid including wireless.h */
+ struct {
+ char name1[IFNAMSIZ];
+ __s32 m;
+ __s16 e;
+ __u8 i;
+ __u8 flags;
+ } wrq;
+
+ fd = socket(AF_INET, SOCK_DGRAM, 0);
+ if (fd == -1)
+ return -1;
+
+ (void) g_strlcpy(wrq.name1, ifname, IFNAMSIZ);
+ /* SIOCGIWFREQ */
+ if (ioctl(fd, 0x8B05, &wrq) == 0) {
+ if (wrq.e == 6)
+ ret = wrq.m;
+ }
+ close(fd);
+ return ret;
+}
+
+struct __iface_info
+{
+ struct ws80211_iface_info *pub;
+ int type;
+ int phyidx;
+};
+
+static int get_iface_info_handler(struct nl_msg *msg, void *arg)
+{
+ struct genlmsghdr *gnlh = (struct genlmsghdr *)nlmsg_data(nlmsg_hdr(msg));
+ struct nlattr *tb_msg[NL80211_ATTR_MAX + 1];
+ struct __iface_info *iface_info = (struct __iface_info *)arg;
+
+ nla_parse(tb_msg, NL80211_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
+ genlmsg_attrlen(gnlh, 0), NULL);
+
+ if (tb_msg[NL80211_ATTR_IFTYPE]) {
+ iface_info->type = nla_get_u32(tb_msg[NL80211_ATTR_IFTYPE]);
+ }
+ if (tb_msg[NL80211_ATTR_WIPHY]) {
+ iface_info->phyidx = nla_get_u32(tb_msg[NL80211_ATTR_WIPHY]);
+ }
+
+ if (tb_msg[NL80211_ATTR_WIPHY_FREQ]) {
+ bool found_ch_width = false;
+ iface_info->pub->current_freq = nla_get_u32(tb_msg[NL80211_ATTR_WIPHY_FREQ]);
+ iface_info->pub->current_chan_type = WS80211_CHAN_NO_HT;
+#ifdef HAVE_NL80211_VHT_CAPABILITY
+ if (tb_msg[NL80211_ATTR_CHANNEL_WIDTH]) {
+ switch (nla_get_u32(tb_msg[NL80211_ATTR_CHANNEL_WIDTH])) {
+ case NL80211_CHAN_WIDTH_80:
+ iface_info->pub->current_chan_type = WS80211_CHAN_VHT80;
+ found_ch_width = true;
+ break;
+ case NL80211_CHAN_WIDTH_80P80:
+ iface_info->pub->current_chan_type = WS80211_CHAN_VHT80P80;
+ found_ch_width = true;
+ break;
+ case NL80211_CHAN_WIDTH_160:
+ iface_info->pub->current_chan_type = WS80211_CHAN_VHT160;
+ found_ch_width = true;
+ break;
+ }
+ }
+ if (tb_msg[NL80211_ATTR_CENTER_FREQ1]) {
+ iface_info->pub->current_center_freq1 =
+ nla_get_u32(tb_msg[NL80211_ATTR_CENTER_FREQ1]);
+ }
+ if (tb_msg[NL80211_ATTR_CENTER_FREQ2]) {
+ iface_info->pub->current_center_freq2 =
+ nla_get_u32(tb_msg[NL80211_ATTR_CENTER_FREQ2]);
+ }
+#endif
+ if (!found_ch_width && tb_msg[NL80211_ATTR_WIPHY_CHANNEL_TYPE]) {
+ switch (nla_get_u32(tb_msg[NL80211_ATTR_WIPHY_CHANNEL_TYPE])) {
+
+ case NL80211_CHAN_NO_HT:
+ iface_info->pub->current_chan_type = WS80211_CHAN_NO_HT;
+ break;
+
+ case NL80211_CHAN_HT20:
+ iface_info->pub->current_chan_type = WS80211_CHAN_HT20;
+ break;
+
+ case NL80211_CHAN_HT40MINUS:
+ iface_info->pub->current_chan_type = WS80211_CHAN_HT40MINUS;
+ break;
+
+ case NL80211_CHAN_HT40PLUS:
+ iface_info->pub->current_chan_type = WS80211_CHAN_HT40PLUS;
+ break;
+ }
+ }
+
+ }
+ return NL_SKIP;
+}
+
+
+static int __ws80211_get_iface_info(const char *name, struct __iface_info *iface_info)
+{
+ int devidx;
+ struct nl_msg *msg;
+ struct nl_cb *cb;
+ msg = nlmsg_alloc();
+ if (!msg) {
+ fprintf(stderr, "failed to allocate netlink message\n");
+ return 2;
+ }
+
+ cb = nl_cb_alloc(NL_CB_DEFAULT);
+
+ devidx = if_nametoindex(name);
+
+ genlmsg_put(msg, 0, 0, nl_state.nl80211_id, 0,
+ 0, NL80211_CMD_GET_INTERFACE, 0);
+ NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, devidx);
+
+ nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, get_iface_info_handler, iface_info);
+
+ if (nl80211_do_cmd(msg, cb)) {
+ nlmsg_free(msg);
+ return -1;
+ }
+
+ /* Old kernels can't get the current freq via netlink. Try WEXT too :( */
+ if (iface_info->pub->current_freq == -1)
+ iface_info->pub->current_freq = get_freq_wext(name);
+ nlmsg_free(msg);
+ return 0;
+
+nla_put_failure:
+ nlmsg_free(msg);
+ fprintf(stderr, "building message failed\n");
+ return -1;
+}
+
+int ws80211_get_iface_info(const char *name, struct ws80211_iface_info *iface_info)
+{
+ struct __iface_info __iface_info;
+
+ memset(iface_info, 0, sizeof(*iface_info));
+ __iface_info.pub = iface_info;
+ __iface_info.type = -1;
+ __iface_info.phyidx= -1;
+ __iface_info.pub->current_freq = -1;
+ __iface_info.pub->current_chan_type = WS80211_CHAN_NO_HT;
+
+ return __ws80211_get_iface_info(name, &__iface_info);
+}
+
+static int ws80211_keep_only_monitor(GArray *interfaces)
+{
+ unsigned int j;
+ struct ws80211_interface *iface;
+restart:
+ for (j = 0; j < interfaces->len; j++) {
+ iface = g_array_index(interfaces, struct ws80211_interface *, j);
+ if (!iface->cap_monitor) {
+ g_array_remove_index(interfaces, j);
+ g_array_free(iface->frequencies, true);
+ g_free(iface->ifname);
+ g_free(iface);
+ goto restart;
+ }
+ }
+ return 0;
+}
+
+static int ws80211_populate_devices(GArray *interfaces)
+{
+ FILE *fh;
+ char line[200];
+ char *t;
+ char *t2;
+ char *ret;
+ int i;
+ unsigned int j;
+
+ struct ws80211_iface_info pub = {-1, WS80211_CHAN_NO_HT, -1, -1, WS80211_FCS_ALL};
+ struct __iface_info iface_info;
+ struct ws80211_interface *iface;
+
+ /* Get a list of phy's that can handle monitor mode */
+ ws80211_get_phys(interfaces);
+ ws80211_keep_only_monitor(interfaces);
+
+ fh = g_fopen("/proc/net/dev", "r");
+ if(!fh) {
+ fprintf(stderr, "Cannot open /proc/net/dev");
+ return -ENOENT;
+ }
+
+ /* Skip the first two lines */
+ for (i = 0; i < 2; i++) {
+ ret = fgets(line, sizeof(line), fh);
+ if (ret == NULL) {
+ fprintf(stderr, "Error parsing /proc/net/dev");
+ fclose(fh);
+ return -1;
+ }
+ }
+
+ /* Update names of user created monitor interfaces */
+ while(fgets(line, sizeof(line), fh)) {
+ t = index(line, ':');
+ if (!t)
+ continue;
+ *t = 0;
+ t = line;
+ while (*t == ' ')
+ t++;
+ memset(&iface_info, 0, sizeof(iface_info));
+ iface_info.pub = &pub;
+ __ws80211_get_iface_info(t, &iface_info);
+
+ if (iface_info.type == NL80211_IFTYPE_MONITOR) {
+ for (j = 0; j < interfaces->len; j++) {
+ iface = g_array_index(interfaces, struct ws80211_interface *, j);
+ t2 = ws_strdup_printf("phy%d.mon", iface_info.phyidx);
+ if (t2) {
+ if (!strcmp(t2, iface->ifname)) {
+ g_free(iface->ifname);
+ iface->ifname = g_strdup(t);
+ }
+ g_free(t2);
+ }
+ }
+ }
+ }
+ fclose(fh);
+ return 0;
+}
+
+static int ws80211_iface_up(const char *ifname)
+{
+ int sock;
+ struct ifreq ifreq;
+
+ sock = socket(AF_PACKET, SOCK_RAW, 0);
+ if (sock == -1)
+ return -1;
+
+ (void) g_strlcpy(ifreq.ifr_name, ifname, sizeof(ifreq.ifr_name));
+
+ if (ioctl(sock, SIOCGIFFLAGS, &ifreq))
+ goto out_err;
+
+ ifreq.ifr_flags |= IFF_UP;
+
+ if (ioctl(sock, SIOCSIFFLAGS, &ifreq))
+ goto out_err;
+
+ close(sock);
+ return 0;
+
+out_err:
+ close(sock);
+ return -1;
+}
+
+/* Needed for NLA_PUT_STRING, which passes strlen as an int */
+DIAG_OFF_CLANG(shorten-64-to-32)
+static int ws80211_create_on_demand_interface(const char *name)
+{
+ int devidx, phyidx, err;
+ struct nl_msg *msg;
+ struct nl_cb *cb;
+
+ devidx = if_nametoindex(name);
+ if (devidx)
+ return ws80211_iface_up(name);
+
+ if (sscanf(name, "phy%d.mon", &phyidx) != 1)
+ return -EINVAL;
+
+ cb = nl_cb_alloc(NL_CB_DEFAULT);
+ msg = nlmsg_alloc();
+ if (!msg) {
+ fprintf(stderr, "failed to allocate netlink message\n");
+ return 2;
+ }
+
+ genlmsg_put(msg, 0, 0, nl_state.nl80211_id, 0,
+ 0, NL80211_CMD_NEW_INTERFACE, 0);
+ NLA_PUT_U32(msg, NL80211_ATTR_WIPHY, phyidx);
+
+ NLA_PUT_STRING(msg, NL80211_ATTR_IFNAME, name);
+ NLA_PUT_U32(msg, NL80211_ATTR_IFTYPE, NL80211_IFTYPE_MONITOR);
+
+ err = nl80211_do_cmd(msg, cb);
+ nlmsg_free(msg);
+ if (err)
+ return err;
+ return ws80211_iface_up(name);
+
+nla_put_failure:
+ nlmsg_free(msg);
+ fprintf(stderr, "building message failed\n");
+ return 2;
+}
+DIAG_ON_CLANG(shorten-64-to-32)
+
+int ws80211_set_freq(const char *name, uint32_t freq, int chan_type, uint32_t _U_ center_freq, uint32_t _U_ center_freq2)
+{
+ int devidx, err;
+ struct nl_msg *msg;
+ struct nl_cb *cb;
+
+ err = ws80211_create_on_demand_interface(name);
+ if (err)
+ return err;
+
+ msg = nlmsg_alloc();
+ if (!msg) {
+ fprintf(stderr, "failed to allocate netlink message\n");
+ return 2;
+ }
+
+ cb = nl_cb_alloc(NL_CB_DEFAULT);
+
+ devidx = if_nametoindex(name);
+
+#ifdef HAVE_NL80211_CMD_SET_CHANNEL
+ genlmsg_put(msg, 0, 0, nl_state.nl80211_id, 0,
+ 0, NL80211_CMD_SET_CHANNEL, 0);
+#else
+ genlmsg_put(msg, 0, 0, nl_state.nl80211_id, 0,
+ 0, NL80211_CMD_SET_WIPHY, 0);
+#endif
+
+ NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, devidx);
+ NLA_PUT_U32(msg, NL80211_ATTR_WIPHY_FREQ, freq);
+
+ switch (chan_type) {
+
+#ifdef NL80211_BAND_ATTR_HT_CAPA
+ case WS80211_CHAN_NO_HT:
+ NLA_PUT_U32(msg, NL80211_ATTR_WIPHY_CHANNEL_TYPE, NL80211_CHAN_NO_HT);
+ break;
+
+ case WS80211_CHAN_HT20:
+ NLA_PUT_U32(msg, NL80211_ATTR_WIPHY_CHANNEL_TYPE, NL80211_CHAN_HT20);
+ break;
+
+ case WS80211_CHAN_HT40MINUS:
+ NLA_PUT_U32(msg, NL80211_ATTR_WIPHY_CHANNEL_TYPE, NL80211_CHAN_HT40MINUS);
+ break;
+
+ case WS80211_CHAN_HT40PLUS:
+ NLA_PUT_U32(msg, NL80211_ATTR_WIPHY_CHANNEL_TYPE, NL80211_CHAN_HT40PLUS);
+ break;
+#endif
+#ifdef HAVE_NL80211_VHT_CAPABILITY
+ case WS80211_CHAN_VHT80:
+ NLA_PUT_U32(msg, NL80211_ATTR_CHANNEL_WIDTH, NL80211_CHAN_WIDTH_80);
+ NLA_PUT_U32(msg, NL80211_ATTR_CENTER_FREQ1, center_freq);
+ break;
+
+ case WS80211_CHAN_VHT80P80:
+ NLA_PUT_U32(msg, NL80211_ATTR_CHANNEL_WIDTH, NL80211_CHAN_WIDTH_80P80);
+ NLA_PUT_U32(msg, NL80211_ATTR_CENTER_FREQ1, center_freq);
+ NLA_PUT_U32(msg, NL80211_ATTR_CENTER_FREQ2, center_freq2);
+ break;
+
+ case WS80211_CHAN_VHT160:
+ NLA_PUT_U32(msg, NL80211_ATTR_CHANNEL_WIDTH, NL80211_CHAN_WIDTH_160);
+ NLA_PUT_U32(msg, NL80211_ATTR_CENTER_FREQ1, center_freq);
+ break;
+#endif
+ default:
+ break;
+ }
+ err = nl80211_do_cmd(msg, cb);
+ nlmsg_free(msg);
+ return err;
+
+nla_put_failure:
+ nlmsg_free(msg);
+ fprintf(stderr, "building message failed\n");
+ return 2;
+
+}
+
+GArray* ws80211_find_interfaces(void)
+{
+ GArray *interfaces;
+
+ if (!nl_state.nl_sock)
+ return NULL;
+
+ interfaces = g_array_new(false, false, sizeof(struct ws80211_interface *));
+ if (!interfaces)
+ return NULL;
+
+ if (ws80211_populate_devices(interfaces)) {
+ ws80211_free_interfaces(interfaces);
+ return NULL;
+ }
+ return interfaces;
+}
+
+int
+ws80211_str_to_chan_type(const char *s)
+{
+ int ret = -1;
+ if (!s)
+ return -1;
+
+ if (!strcmp(s, CHAN_NO_HT))
+ ret = WS80211_CHAN_NO_HT;
+ if (!strcmp(s, CHAN_HT20))
+ ret = WS80211_CHAN_HT20;
+ if (!strcmp(s, CHAN_HT40MINUS))
+ ret = WS80211_CHAN_HT40MINUS;
+ if (!strcmp(s, CHAN_HT40PLUS))
+ ret = WS80211_CHAN_HT40PLUS;
+ if (!strcmp(s, CHAN_VHT80))
+ ret = WS80211_CHAN_VHT80;
+ if (!strcmp(s, CHAN_VHT80P80))
+ ret = WS80211_CHAN_VHT80P80;
+ if (!strcmp(s, CHAN_VHT160))
+ ret = WS80211_CHAN_VHT160;
+
+ return ret;
+}
+
+const char
+*ws80211_chan_type_to_str(int type)
+{
+ switch (type) {
+ case WS80211_CHAN_NO_HT:
+ return CHAN_NO_HT;
+ case WS80211_CHAN_HT20:
+ return CHAN_HT20;
+ case WS80211_CHAN_HT40MINUS:
+ return CHAN_HT40MINUS;
+ case WS80211_CHAN_HT40PLUS:
+ return CHAN_HT40PLUS;
+ case WS80211_CHAN_VHT80:
+ return CHAN_VHT80;
+ case WS80211_CHAN_VHT80P80:
+ return CHAN_VHT80P80;
+ case WS80211_CHAN_VHT160:
+ return CHAN_VHT160;
+ }
+ return NULL;
+}
+
+bool ws80211_has_fcs_filter(void)
+{
+ return false;
+}
+
+int ws80211_set_fcs_validation(const char *name _U_, enum ws80211_fcs_validation fcs_validation _U_)
+{
+ return -1;
+}
+
+const char *network_manager_path = "/usr/sbin/NetworkManager"; /* Is this correct? */
+const char *ws80211_get_helper_path(void) {
+ if (g_file_test(network_manager_path, G_FILE_TEST_IS_EXECUTABLE)) {
+ return network_manager_path;
+ }
+ return NULL;
+}
+
+#elif defined(HAVE_AIRPCAP)
+
+#include <wsutil/unicode-utils.h>
+
+#include "airpcap.h"
+#include "airpcap_loader.h"
+
+int ws80211_init(void)
+{
+ if (airpcap_get_dll_state() == AIRPCAP_DLL_OK) {
+ return WS80211_INIT_OK;
+ }
+ return WS80211_INIT_NOT_SUPPORTED;
+}
+
+static const char *airpcap_dev_prefix_ = "\\\\.\\";
+
+GArray* ws80211_find_interfaces(void)
+{
+ GArray *interfaces;
+ GList *airpcap_if_list, *cur_if;
+ int err;
+ char *err_str = NULL;
+
+ interfaces = g_array_new(false, false, sizeof(struct ws80211_interface *));
+ if (!interfaces)
+ return NULL;
+
+ airpcap_if_list = get_airpcap_interface_list(&err, &err_str);
+
+ if (airpcap_if_list == NULL || g_list_length(airpcap_if_list) == 0){
+ g_free(err_str);
+ g_array_free(interfaces, true);
+ return NULL;
+ }
+
+ for (cur_if = airpcap_if_list; cur_if; cur_if = g_list_next(cur_if)) {
+ struct ws80211_interface *iface;
+ airpcap_if_info_t *airpcap_if_info = (airpcap_if_info_t *) cur_if->data;
+ char *ifname;
+ uint32_t chan;
+ uint32_t i;
+
+ if (!airpcap_if_info) continue;
+ ifname = airpcap_if_info->name;
+ if (strlen(ifname) > 4 && g_str_has_prefix(ifname, airpcap_dev_prefix_)) ifname += 4;
+
+ iface = (struct ws80211_interface *)g_malloc0(sizeof(*iface));
+ iface->ifname = g_strdup(ifname);
+ iface->can_set_freq = true;
+ iface->frequencies = g_array_new(false, false, sizeof(uint32_t));
+
+ iface->channel_types = 1 << WS80211_CHAN_NO_HT;
+ /*
+ * AirPcap stores per-channel capabilities. We should probably
+ * do the same. */
+ for (i = 0; i < airpcap_if_info->numSupportedChannels; i++) {
+ if (airpcap_if_info->pSupportedChannels[i].Flags & FLAG_CAN_BE_HIGH) {
+ iface->channel_types |= 1 << WS80211_CHAN_HT40MINUS;
+ iface->channel_types |= 1 << WS80211_CHAN_HT40PLUS;
+ break;
+ }
+ }
+
+ iface->cap_monitor = 1;
+
+ for (chan = 0; chan < airpcap_if_info->numSupportedChannels; chan++) {
+ g_array_append_val(iface->frequencies, airpcap_if_info->pSupportedChannels[chan].Frequency);
+ }
+
+ g_array_append_val(interfaces, iface);
+ }
+
+ return interfaces;
+}
+
+int ws80211_get_iface_info(const char *name, struct ws80211_iface_info *iface_info)
+{
+ GList *airpcap_if_list;
+ int err;
+ char *err_str = NULL;
+ airpcap_if_info_t *airpcap_if_info;
+
+ if (!iface_info) return -1;
+
+ airpcap_if_list = get_airpcap_interface_list(&err, &err_str);
+
+ if (airpcap_if_list == NULL || g_list_length(airpcap_if_list) == 0){
+ g_free(err_str);
+ return -1;
+ }
+
+ airpcap_if_info = get_airpcap_if_from_name(airpcap_if_list, name);
+
+ if (!airpcap_if_info) {
+ free_airpcap_interface_list(airpcap_if_list);
+ return -1;
+ }
+
+ memset(iface_info, 0, sizeof(*iface_info));
+ iface_info->current_freq = airpcap_if_info->channelInfo.Frequency;
+ switch (airpcap_if_info->channelInfo.ExtChannel) {
+ case 0:
+ iface_info->current_chan_type = WS80211_CHAN_NO_HT;
+ break;
+ case -1:
+ iface_info->current_chan_type = WS80211_CHAN_HT40MINUS;
+ break;
+ case 1:
+ iface_info->current_chan_type = WS80211_CHAN_HT40PLUS;
+ break;
+ default:
+ return -1;
+ }
+
+ switch (airpcap_if_info->CrcValidationOn) {
+ case AIRPCAP_VT_ACCEPT_CORRECT_FRAMES:
+ iface_info->current_fcs_validation = WS80211_FCS_VALID;
+ break;
+ case AIRPCAP_VT_ACCEPT_CORRUPT_FRAMES:
+ iface_info->current_fcs_validation = WS80211_FCS_INVALID;
+ break;
+ default:
+ iface_info->current_fcs_validation = WS80211_FCS_ALL;
+ break;
+ }
+
+ return 0;
+}
+
+int ws80211_set_freq(const char *name, uint32_t freq, int chan_type, uint32_t _U_ center_freq, uint32_t _U_ center_freq2)
+{
+ GList *airpcap_if_list;
+ int err;
+ char *err_str = NULL;
+ airpcap_if_info_t *airpcap_if_info;
+ char err_buf[AIRPCAP_ERRBUF_SIZE];
+ PAirpcapHandle adapter;
+ int ret_val = -1;
+
+ airpcap_if_list = get_airpcap_interface_list(&err, &err_str);
+
+ if (airpcap_if_list == NULL || g_list_length(airpcap_if_list) == 0){
+ g_free(err_str);
+ return ret_val;
+ }
+
+ airpcap_if_info = get_airpcap_if_from_name(airpcap_if_list, name);
+
+ if (!airpcap_if_info) {
+ free_airpcap_interface_list(airpcap_if_list);
+ return ret_val;
+ }
+
+ adapter = airpcap_if_open(airpcap_if_info->name, err_buf);
+ if (adapter) {
+ airpcap_if_info->channelInfo.Frequency = freq;
+ switch (chan_type) {
+ case WS80211_CHAN_HT40MINUS:
+ airpcap_if_info->channelInfo.ExtChannel = -1;
+ break;
+ case WS80211_CHAN_HT40PLUS:
+ airpcap_if_info->channelInfo.ExtChannel = 1;
+ break;
+ default:
+ airpcap_if_info->channelInfo.ExtChannel = 0;
+ break;
+ }
+
+ if (airpcap_if_set_device_channel_ex(adapter, airpcap_if_info->channelInfo)) {
+ ret_val = 0;
+ }
+ airpcap_if_close(adapter);
+ }
+
+ free_airpcap_interface_list(airpcap_if_list);
+ return ret_val;
+}
+
+int ws80211_str_to_chan_type(const char *s _U_)
+{
+ return -1;
+}
+
+const char *ws80211_chan_type_to_str(int type _U_)
+{
+ return NULL;
+}
+
+bool ws80211_has_fcs_filter(void)
+{
+ return true;
+}
+
+int ws80211_set_fcs_validation(const char *name, enum ws80211_fcs_validation fcs_validation)
+{
+ GList *airpcap_if_list;
+ int err;
+ char *err_str = NULL;
+ airpcap_if_info_t *airpcap_if_info;
+ char err_buf[AIRPCAP_ERRBUF_SIZE];
+ PAirpcapHandle adapter;
+ int ret_val = -1;
+
+ airpcap_if_list = get_airpcap_interface_list(&err, &err_str);
+
+ if (airpcap_if_list == NULL || g_list_length(airpcap_if_list) == 0){
+ g_free(err_str);
+ return ret_val;
+ }
+
+ airpcap_if_info = get_airpcap_if_from_name(airpcap_if_list, name);
+
+ if (!airpcap_if_info) {
+ free_airpcap_interface_list(airpcap_if_list);
+ return ret_val;
+ }
+
+ adapter = airpcap_if_open(airpcap_if_info->name, err_buf);
+ if (adapter) {
+ AirpcapValidationType val_type = AIRPCAP_VT_ACCEPT_EVERYTHING;
+ switch (fcs_validation) {
+ case WS80211_FCS_VALID:
+ val_type = AIRPCAP_VT_ACCEPT_CORRECT_FRAMES;
+ break;
+ case WS80211_FCS_INVALID:
+ val_type = AIRPCAP_VT_ACCEPT_CORRUPT_FRAMES;
+ break;
+ default:
+ break;
+ }
+
+ if (airpcap_if_set_fcs_validation(adapter, val_type)) {
+ /* Appears to be necessary for this to take effect. */
+ airpcap_if_store_cur_config_as_adapter_default(adapter);
+ ret_val = 0;
+ }
+ airpcap_if_close(adapter);
+ }
+
+ free_airpcap_interface_list(airpcap_if_list);
+ return ret_val;
+}
+
+static char *airpcap_conf_path = NULL;
+const char *ws80211_get_helper_path(void)
+{
+ HKEY h_key = NULL;
+
+ if (!airpcap_conf_path && RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T("SOFTWARE\\AirPcap"), 0, KEY_QUERY_VALUE|KEY_WOW64_32KEY, &h_key) == ERROR_SUCCESS) {
+ DWORD reg_ret;
+ TCHAR airpcap_dir_utf16[MAX_PATH];
+ DWORD ad_size = sizeof(airpcap_dir_utf16)/sizeof(TCHAR);
+
+ reg_ret = RegQueryValueEx(h_key, NULL, NULL, NULL,
+ (LPBYTE) &airpcap_dir_utf16, &ad_size);
+
+ if (reg_ret == ERROR_SUCCESS) {
+ airpcap_dir_utf16[ad_size-1] = L'\0';
+ g_free(airpcap_conf_path);
+ airpcap_conf_path = ws_strdup_printf("%s\\AirpcapConf.exe", utf_16to8(airpcap_dir_utf16));
+
+ if (!g_file_test(airpcap_conf_path, G_FILE_TEST_IS_EXECUTABLE)) {
+ g_free(airpcap_conf_path);
+ airpcap_conf_path = NULL;
+ }
+ }
+ }
+
+ return airpcap_conf_path;
+}
+
+#else /* Everyone else. */
+int ws80211_init(void)
+{
+ return WS80211_INIT_NOT_SUPPORTED;
+}
+
+GArray* ws80211_find_interfaces(void)
+{
+ return NULL;
+}
+
+int ws80211_get_iface_info(const char *name _U_, struct ws80211_iface_info *iface_info _U_)
+{
+ return -1;
+}
+
+int ws80211_set_freq(const char *name _U_, uint32_t freq _U_, int _U_ chan_type, uint32_t _U_ center_freq, uint32_t _U_ center_freq2)
+{
+ return -1;
+}
+
+int ws80211_str_to_chan_type(const char *s _U_)
+{
+ return -1;
+}
+
+const char *ws80211_chan_type_to_str(int type _U_)
+{
+ return NULL;
+}
+
+bool ws80211_has_fcs_filter(void)
+{
+ return false;
+}
+
+int ws80211_set_fcs_validation(const char *name _U_, enum ws80211_fcs_validation fcs_validation _U_)
+{
+ return -1;
+}
+
+const char *ws80211_get_helper_path(void) {
+ return NULL;
+}
+
+#endif /* HAVE_LIBNL && HAVE_NL80211 */
+
+/* Common to everyone */
+
+void ws80211_free_interfaces(GArray *interfaces)
+{
+ struct ws80211_interface *iface;
+
+ if (!interfaces)
+ return;
+
+ while (interfaces->len) {
+ iface = g_array_index(interfaces, struct ws80211_interface *, 0);
+ g_array_remove_index(interfaces, 0);
+ g_array_free(iface->frequencies, true);
+ g_free(iface->ifname);
+ g_free(iface);
+ }
+ g_array_free(interfaces, true);
+}
+
+/*
+ * Editor modelines - https://www.wireshark.org/tools/modelines.html
+ *
+ * Local variables:
+ * c-basic-offset: 8
+ * tab-width: 8
+ * indent-tabs-mode: t
+ * End:
+ *
+ * vi: set shiftwidth=8 tabstop=8 noexpandtab:
+ * :indentSize=8:tabSize=8:noTabs=false:
+ */
diff --git a/capture/ws80211_utils.h b/capture/ws80211_utils.h
new file mode 100644
index 00000000..a354e51e
--- /dev/null
+++ b/capture/ws80211_utils.h
@@ -0,0 +1,135 @@
+/** @file
+ *
+ * Copyright 2012, Pontus Fuchs <pontus.fuchs@gmail.com>
+ *
+ * Wireshark - Network traffic analyzer
+ * By Gerald Combs <gerald@wireshark.org>
+ * Copyright 1998 Gerald Combs
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+
+#ifndef __WS80211_UTILS_H__
+#define __WS80211_UTILS_H__
+
+#include <wireshark.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif /* __cplusplus */
+
+enum ws80211_channel_type {
+ WS80211_CHAN_NO_HT,
+ WS80211_CHAN_HT20,
+ WS80211_CHAN_HT40MINUS,
+ WS80211_CHAN_HT40PLUS,
+ WS80211_CHAN_VHT80,
+ WS80211_CHAN_VHT80P80,
+ WS80211_CHAN_VHT160
+};
+
+#define CHAN_NO_HT "NOHT"
+#define CHAN_HT20 "HT20"
+#define CHAN_HT40MINUS "HT40-"
+#define CHAN_HT40PLUS "HT40+"
+#define CHAN_VHT80 "VHT80"
+#define CHAN_VHT80P80 "VHT80+80"
+#define CHAN_VHT160 "VHT160"
+
+/* XXX This doesn't match AirpcapValidationType. Should it? */
+enum ws80211_fcs_validation {
+ WS80211_FCS_ALL,
+ WS80211_FCS_VALID,
+ WS80211_FCS_INVALID
+};
+
+struct ws80211_interface
+{
+ char *ifname;
+ bool can_set_freq;
+ bool can_check_fcs;
+ GArray *frequencies; /* Array of uint32_t? */
+ int channel_types; /* Union for all bands */
+ int cap_monitor;
+};
+
+struct ws80211_iface_info {
+ int current_freq;
+ enum ws80211_channel_type current_chan_type;
+ int current_center_freq1;
+ int current_center_freq2;
+ enum ws80211_fcs_validation current_fcs_validation;
+};
+
+/** Initialize the 802.11 environment.
+ * On Linux this initializes an nl80211_state struct.
+ * On Windows this checks the AirPcap status. It does *not* load the
+ * AirPcap DLL. That happens when the program starts.
+ *
+ * @return WS80211_INIT_OK on success, WS80211_INIT_NOT_SUPPORTED if the
+ * 802.11 environment isn't supported, or the negative of an errno value
+ * on failure.
+ */
+#define WS80211_INIT_OK 0
+#define WS80211_INIT_NOT_SUPPORTED 1
+
+int ws80211_init(void);
+
+/** Build a list of 802.11 interfaces.
+ *
+ * @return A GArray of pointers to struct ws80211_interface on success, NULL on failure.
+ */
+/* XXX Should we make this an array of structs instead of an array of struct pointers?
+ * It'd save a bit of mallocing and freeing. */
+GArray* ws80211_find_interfaces(void);
+
+int ws80211_get_iface_info(const char *name, struct ws80211_iface_info *iface_info);
+
+/** Free an interface list.
+ *
+ * @param interfaces A list of interfaces created with ws80211_find_interfaces().
+ */
+void ws80211_free_interfaces(GArray *interfaces);
+
+/** Set the frequency and channel width for an interface.
+ *
+ * @param name The interface name.
+ * @param freq The frequency in MHz.
+ * @param chan_type The HT channel type (no, 20Mhz, 40Mhz...).
+ * @param center_freq The center frequency in MHz (if 80MHz, 80+80MHz or 160MHz).
+ * @param center_freq2 The 2nd center frequency in MHz (if 80+80MHz).
+ * @return Zero on success, nonzero on failure.
+ */
+int ws80211_set_freq(const char *name, uint32_t freq, int chan_type, uint32_t _U_ center_freq, uint32_t _U_ center_freq2);
+
+int ws80211_str_to_chan_type(const char *s);
+const char *ws80211_chan_type_to_str(int type);
+
+/** Check to see if we have FCS filtering.
+ *
+ * @return true if FCS filtering is supported on this platform.
+ */
+bool ws80211_has_fcs_filter(void);
+
+/** Set the FCS validation behavior for an interface.
+ *
+ * @param name The interface name.
+ * @param fcs_validation The desired validation behavior.
+ * @return Zero on success, nonzero on failure.
+ */
+int ws80211_set_fcs_validation(const char *name, enum ws80211_fcs_validation fcs_validation);
+
+
+/** Get the path to a helper application.
+ * Return the path to a separate 802.11 helper application, e.g.
+ * the AirPcap control panel or the GNOME Network Manager.
+ *
+ * @return The path to the helper on success, NULL on failure.
+ */
+const char *ws80211_get_helper_path(void);
+
+#ifdef __cplusplus
+}
+#endif /* __cplusplus */
+
+#endif /* __WS80211_UTILS_H__ */