A drop-in tracing library / DLL for most normal Winsock calls.
It sits between your program and the Winsock library (ws2_32.dll
).
It works best for MSVC since the stack-walking code requires the program's
PDB symbol-file to be present. And unfortunately MinGW/CygWin doesn't produce
PDB-symbols (GNU-debugger instead relies on the archaic BFD library).
Example output from c:\> ahost msdn.com
showing all the addresses of msdn.com
(ahost
is part of the DNS library C-ares) :
[] (screenshot_ahost-msdn-com-win10.png?raw=true)
-
Colourised trace of the Winsock calls with function parameters and return values. The colours are configurable.
-
Runtime caller information: Using Microsoft's dbghelp (or psapi) APIs together with the programs PDB-file, the filename, line-number of the calling function-name is shown. In the above example,
WSAStartup()
is called fromahost.c
, line 67. Which should be 59 bytes into themain()
function. This should be here. -
Precise Timestamps: All trace-lines starts with a precise timestamp obtained from
QueryPerformanceCounter()
.
The timestamp is controlled bytrace_time
in thewsock_trace
config-file. -
Extension functions: Winsock has several Microsoft-specific extension functions (like
AcceptEx
andConnectEx
).
Winsock-Trace is able to trace these too. -
IP-Country information from MaxMind. (using the CSV files
geoip
andgeoip6
are always enabled).
Thanks to the Tor-project for a simplified version of these MaxMind GeoIP-databases.
Also many thanks to MaxMind and IP2Location [3] for their data-bases. -
IP-Location information from IP2Location. (this is contolled by
USE_IP2LOCATION = 1
in the makefiles).
The aboveMountain View/California
is Google's well-known location.
Many thanks to IP2Location [3] for their data-bases. -
Domain Name System-based Blackhole List (DNSBL) support: with the help of DROP-files from the Spamhaus project, it can detect IPv4 / IPv6-addresses uses by spammers and cyber-criminals. The more potent Spamhaus BGPCC is on the to-do list.
-
LuaJIT script support: very preliminary at the moment. The idea is that
.lua
scripts could change the behaviour of Wsock-trace at runtime without rebuilding it.
To be able to get more precise Geo-IP information for addresses (city and
region), Wsock-trace can use the IP2Location library.
Do this:
-
Sign-up for an account and download the free IP2Location LITE databases here.
-
Put the
IP2LOCATION-LITE-DBx.BIN
file (or similar, see [*] below) into your%HOME%
or%APPDATA%
directory. -
Clone this repository along with its submodules:
git clone --recursive -j8 https://github.com/gvanem/wsock-trace.git
If you have already cloned this repository, you can initialize and update the submodules like so:
git submodule update --init --recursive
-
Edit the respective makefile to say
USE_IP2LOCATION = 1
-
Then do the specific installation for your compiler (see below).
[*]: to enable locations for both IPv4 and IPv6 addresses, download and use a
file named like IP2LOCATION-LITE-DBx.IPV6.BIN
.
These files contains both IPv4 and IPv6 records.
Enter the src
sub-directory and do a nmake -f Makefile.vc6.
This produces a wsock_trace.lib
that you'll need to use to
link your project(s) with. This lib would then trace the normal
Winsock calls. Example screen-shot above or details in
Running samples below.
Link with wsock_trace.lib
instead of the system's ws32_2.lib
. Thus
most normal Winsock calls are traced on entry and exit. Remember to
compile using -Zi
to produce debug-symbols. And remember to use -debug
when linking your program. See src/Makefile.vc6
for an example.
It is not adviced to use option /Oy
(enable frame pointer omission)
since that will make it difficult for StackWalk64()
to figure out the
filename and line of the calling function.
Enter the src
sub-directory and do a make -f Makefile.MinGW or
make -f Makefile.Cygwin.
Link with libwsock_trace.a
instead of the system's libws32_2.a
(i.e.
-lws32_2
). So copy this library to a directory in $(LIBRARY_PATH)
and
use -lwsock_trace
to link. The Makefile.MinGW
already does the copying
to $(MINGW32)/lib
.
The trace-level and other settings are controlled by a config-file
wsock_trace
. This file is searched along these places until found:
- The file pointed to by
%WSOCK_TRACE
. - The current directory.
- The
%HOME
directory. - Then finally the
%APPDATA
directory.
wsock_trace
is read in init.c
at startup. Read it's contents; the comments therein should be self-explanatory.
If wsock_trace
is not found in one of the above directories, the default
trace_level
is set to 1.
There is no install.bat
file for Wsock-Trace. So you should copy the
following files (here at GitHub) to your %HOME
or %APPDATA
directory:
wsock_trace
geoip
geoip6
drop.txt
edrop.txt
dropv6.txt
These environment variables are on the form:
<drive>:\Documents and Settings\<User Name>\ProgramData
. (Win-XP)<drive>:\Users\<User Name>\AppData\Roaming
. (Win-Vista+)
Since it's a confusing subject what a program's configuration directory should be,
it's best to define a %HOME%
to point to the excact place for such config-files.
Example output from src/test.exe
(built with MSVC):
* ws_trace/test.c(45) (main+50): WSAStartup (2.2) --> No error.
* ws_trace/test.c(24) (do_wsock_tests+125): gethostbyaddr (127.0.0.1, 4, AF_INET) --> 0x003C8780.
* ws_trace/test.c(27) (do_wsock_tests+150): gethostbyaddr (0.0.0.0, 4, AF_INET) --> 0x003C8780.
* ws_trace/test.c(29) (do_wsock_tests+164): gethostbyaddr (::1, 16, AF_INET6) --> 0x003C8780.
* ws_trace/test.c(31) (do_wsock_tests+175): gethostbyname (localhost) --> 0x003C8780.
* ws_trace/test.c(31) (do_wsock_tests+187): socket (AF_INET, SOCK_STREAM, 0) --> 1724.
* ws_trace/test.c(33) (do_wsock_tests+196): WSAGetLastError() --> No error.
* ws_trace/test.c(36) (do_wsock_tests+343): select (n=0-1724, rd, NULL, NULL, {tv=1.000001s}) --> No error.
* ws_trace/test.c(37) (do_wsock_tests+358): FD_ISSET (1724, fd) --> 0.
* ws_trace/test.c(47) (main+61): WSACleanup() --> No error.
^ ^ ^ ^
| | | |___ The traced Winsock function and the result.
| | |
| | |____ The calling function with displacement (i.e. offset from
| | (nearest public symbol).
| |_____ Line number in src-file.
|
|____ Source-file relative of the application.
Here is a more realistic and useful example with wsock_trace.lib
linked to
Nmap [1]:
c:\> nmap -sT -P0 -p23,80 10.0.0.1
* mswin32/winfix.cc(134) (win_pre_init+68): WSAStartup (2.2) --> No error.
Starting Nmap 6.02 ( http://nmap.org ) at 2012-07-24 12:48 CET
* g:/vc_2010/sdk/include/wspiapi.h(1011) (WspiapiGetAddrInfo+79): WSASetLastError (0).
* g:/vc_2010/sdk/include/wspiapi.h(1011) (WspiapiGetAddrInfo+79): WSASetLastError (0).
* g:/vc_2010/sdk/include/wspiapi.h(1011) (WspiapiGetAddrInfo+79): WSASetLastError (0).
* g:/vc_2010/sdk/include/wspiapi.h(1011) (WspiapiGetAddrInfo+79): WSASetLastError (0).
* g:/vc_2010/sdk/include/wspiapi.h(1011) (WspiapiGetAddrInfo+79): WSASetLastError (0).
* g:/vc_2010/sdk/include/wspiapi.h(1011) (WspiapiGetAddrInfo+79): WSASetLastError (0).
* scan_engine.cc(3002) (sendConnectScanProbe+266): socket (AF_INET, SOCK_STREAM, 6) --> 1780.
* nbase/nbase_misc.c(261) (unblock_socket+65): ioctlsocket (1780, FIONBIO, 1) --> No error.
* scan_engine.cc(2964) (init_socket+82): setsockopt (1780, SOL_SOCKET, SO_LINGER, 1, 4) --> No error.
* libnetutil/netutil.cc(856) (set_ttl+34): setsockopt (1780, IPPROTO_IP, IP_TTL, ULONG_MAX, 4) --> WSAEINVAL: Invalid arguments (10022).
* scan_engine.cc(3021) (sendConnectScanProbe+599): connect (1780, 10.0.0.1:23) --> WSAEWOULDBLOCK: Call would block (10035).
* nbase/nbase_misc.c(133) (socket_errno+12): WSAGetLastError() --> WSAEWOULDBLOCK: Call would block (10035).
* scan_engine.cc(922) (ConnectScanInfo::watchSD+82): FD_ISSET (1780, fd) --> 0.
* scan_engine.cc(3002) (sendConnectScanProbe+266): socket (AF_INET, SOCK_STREAM, 6) --> 1720.
* nbase/nbase_misc.c(261) (unblock_socket+65): ioctlsocket (1720, FIONBIO, 1) --> No error.
* scan_engine.cc(2964) (init_socket+82): setsockopt (1720, SOL_SOCKET, SO_LINGER, 1, 4) --> No error.
* libnetutil/netutil.cc(856) (set_ttl+34): setsockopt (1720, IPPROTO_IP, IP_TTL, ULONG_MAX, 4) --> WSAEINVAL: Invalid arguments (10022).
* scan_engine.cc(3021) (sendConnectScanProbe+599): connect (1720, 10.0.0.1:80) --> WSAEWOULDBLOCK: Call would block (10035).
* nbase/nbase_misc.c(133) (socket_errno+12): WSAGetLastError() --> WSAEWOULDBLOCK: Call would block (10035).
* scan_engine.cc(922) (ConnectScanInfo::watchSD+82): FD_ISSET (1720, fd) --> 0.
* scan_engine.cc(3964) (do_one_select_round+473): select (n=0-1780, rd, wr, ex, {tv=0.985000s}) --> 3.
* nbase/nbase_misc.c(133) (socket_errno+12): WSAGetLastError() --> WSAEWOULDBLOCK: Call would block (10035).
* scan_engine.cc(4015) (do_one_select_round+1894): FD_ISSET (1720, fd) --> 0.
* scan_engine.cc(4015) (do_one_select_round+1917): FD_ISSET (1720, fd) --> 1.
* scan_engine.cc(4019) (do_one_select_round+2012): getsockopt (1720, SOL_SOCKET, SO_ERROR, 0, 4) --> No error.
* scan_engine.cc(938) (ConnectScanInfo::clearSD+82): FD_ISSET (1720, fd) --> 1.
* scan_engine.cc(789) (ConnectProbe::~ConnectProbe+37): closesocket (1720) --> No error.
* scan_engine.cc(4015) (do_one_select_round+1894): FD_ISSET (1780, fd) --> 1.
* scan_engine.cc(4019) (do_one_select_round+2012): getsockopt (1780, SOL_SOCKET, SO_ERROR, 0, 4) --> No error.
* scan_engine.cc(938) (ConnectScanInfo::clearSD+82): FD_ISSET (1780, fd) --> 1.
* scan_engine.cc(789) (ConnectProbe::~ConnectProbe+37): closesocket (1780) --> No error.
Nmap scan report for router (10.0.0.1)
Host is up (0.0019s latency).
PORT STATE SERVICE
23/tcp open telnet
80/tcp open http
Nmap done: 1 IP address (1 host up) scanned in 7.61 seconds
* mswin32/winfix.cc(290) (win_cleanup+12): WSACleanup() --> No error.
Notes:
- Nmap uses wrong arguments to
setsockopt()
; a TTL of ULONG_MAX. - Nmap also calls
WSAStartup()
before the startup message. - Last but not least, notice how
wsock_trace
handles (demangles) C++ symbols just fine thanks todbghelp.dll
andUnDecorateSymbolName()
. I.e. the destructorConnectProbe::~ConnectProbe
above is callingclosesocket()
at offset 37. (you can turn off C++ demangling bycpp_demangle = 0
in the config-file).
And another example from C-ares's' adig.c:
c:\> adig -t PTR 89.42.216.144
* adig.c(216) (main+105): WSAStartup (2.2) --> No error.
* ares_process.c(1065) (open_udp_socket+248): socket (AF_INET, SOCK_DGRAM, 0) --> 1604.
* ares_process.c(857) (setsocknonblock+61): ioctlsocket (1604, FIONBIO, 1) --> No error.
* ares_process.c(1077) (open_udp_socket+345): connect (1604, 8.8.8.8:53) --> No error.
* ares_process.c(791) (ares__send_query+484): send (1604, 0x00034BDA, 31, 0) --> 31 bytes tx.
* adig.c(397) (main+1780): select (n=0-1604, rd, wr, NULL, {tv=3.109000s}) --> 1.
* ares_process.c(456) (read_udp_packets+146): FD_ISSET (1604, fd) --> 1.
* ares_process.c(485) (read_udp_packets+413): recvfrom (1604, 0x0013F894, 513, 0, 8.8.8.8:53) --> 106 bytes rx.
Domain name not found
id: 58187
flags: qr rd ra
opcode: QUERY
rcode: NXDOMAIN
Questions:
89.42.216.144 . PTR
Answers:
NS records:
. 1413 SOA a.root-servers.net.
nstld.verisign-grs.com.
( 2012072400 1800 900 604800 86400 )
Additional records:
* ares__close_sockets.c(63) (ares__close_sockets+408): closesocket (1604) --> No error.
* adig.c(411) (main+1894): WSACleanup() --> No error.
By default, the tracing of htons()
,htonl()
, ntohs()
and ntohl()
are
excluded from the trace.
You can edit the %HOME/wsock_trace
file and exclude whatever calls you like.
A more eleborated example from 2 OpenVPN programs (a client and a server linked to
wsock_trace.lib
) running a simple test (in OpenVPN's root-dir):
cd sample
start /pos=200,50,1000,800 ..\openvpn.exe --config sample-config-files/loopback-server
start /pos=800,150,1000,1000 ..\openvpn.exe --config sample-config-files/loopback-client
A Larger version.
The names of the import libraries and the names of the 32-bit .DLLs are:
- For MSVC:
wsock_trace.lib
andwsock_trace.dll
. - For MinGW:
libwsock_trace.a
andwsock_trace_mw.dll
. - For CygWin32:
libwsock_trace.a
andwsock_trace_cyg.dll
.
And the 64-bit equivalents:
- For MSVC:
wsock_trace_x64.lib
andwsock_trace_x64.dll
. - For MinGW:
libwsock_trace_x64.a
andwsock_trace_mw_x64.dll
. - For CygWin64:
libwsock_trace_x64.a
andwsock_trace_cyg_x64.dll
.
These DLLs off-course needs to be in current directory or on %PATH
. The reason
I've chosen to make it a DLL and not a static-lib is that applications
using wsock_trace.lib
needs not to be re-linked when I do change the inner
workings of the wsock_trace
source code (I've done that a lot lately).
As long as the ABI is stable (e.g. not adding functions to the wsock_trace.def
file), the application using wsock_trace.dll
should work the same. Only the
trace should change.
Note that some virus scanners may find the behaviour of programs linked to
wsock_trace.lib
suspicious.
-
Get the decoding of calling function, file-name and lines in the MinGW/CygWin ports working.
-
Lua-script integration; use a
*.lua
file to exclude/include processes and/or functions to trace. -
Injecting
wsock_trace.dll
into a remote process. Ref: ** http://www.viksoe.dk/code/wepmetering.htm ** -
Optionally load Wireshark's
libwireshark.dll
to dissect transport and application protocols.
Do it for selected processes only. -
Deny certain applications to use
AF_INET6
protocols (return-1
onsocket(AF_INET6,...)
. -
Add a Json type config feature to support the above features. E.g.:
wireshark_dissect { wget.exe : 1 # Wireshark dissection in wget and curl only. curl.exe : 1 } exclude_trace { select: [curl.exe, wget.exe] # Exclude trace of `select()` and `inet_ntoa()` in curl/wget. inet_ntoa: [curl.exe, wget.exe] htons: [ * ] # Exclude trace of `htons` globally. } deny_ipv6 { pycurl.pyd : 1 # Deny AF_INET6 sockets in scripts using the PyCurl module. python27.dll : 1 # And in other Python scripts too. }
-
Make a GUI trace viewer for it. Ref: http://www.viksoe.dk/code/windowless1.htm
-
Make it possible to switch network stacks at run-time (select amongst Winsock2, lwIP, SwsSock and/or Cyclone TCP (ported to Win32)).
G. Vanem <[email protected]>
2013 - 2018.
-
[1] Nmap; "Network Mapper" is a free and open source (license) utility for network discovery and security auditing. Ref. http://nmap.org/download.html
-
[2] A C library for asynchronous DNS requests (including name resolves) Ref. http://c-ares.haxx.se/
-
[3] This site or product includes IP2Location LITE data available from http://lite.ip2location.com.
PS. This file is written with the aid of the Atom editor and it's Markdown-Preview. A real time-saver.