--- trunk/reactos/apps/utils/net/netstat/netstat.c 2005-10-10 15:42:08 UTC (rev 18396)
+++ trunk/reactos/apps/utils/net/netstat/netstat.c 2005-10-10 17:49:19 UTC (rev 18397)
@@ -1,254 +1,590 @@
/*
- * netstat - display IP stack statistics.
+ * ReactOS Win32 Applications
+ * Copyright (C) 2005 ReactOS Team
*
- * This source code is in the PUBLIC DOMAIN and has NO WARRANTY.
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
*
- * Robert Dickenson <robd@reactos.org>, August 15, 2002.
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
-// Extensive reference made and use of source to netstatp by:
-// Copyright (C) 1998-2002 Mark Russinovich
-// www.sysinternals.com
+/*
+ * COPYRIGHT: See COPYING in the top level directory
+ * PROJECT: ReactOS netstat utility
+ * FILE: apps/utils/net/netstat/netstat.c
+ * PURPOSE: display IP stack statistics
+ * PROGRAMMERS: Ged Murphy (gedmurphy@gmail.com)
+ * REVISIONS:
+ * Ged Murphy 19/09/05 Created
+ * Some ideas/code taken from Rob Dickinson's original app
+ *
+ */
-#define ANY_SIZE 1
+/*
+ * TODO:
+ * rewrite DisplayOutput
+ * sort function return values. BOOL is crap
+ * implement -b, -o and -v
+ * clean up GetPortName and GetIpHostName
+ * command line parser needs more work
+ */
#include <windows.h>
#include <winsock.h>
#include <tchar.h>
#include <stdio.h>
-#include <ctype.h>
-#include <time.h>
-
-#include <iptypes.h>
-#include <ipexport.h>
-#include <tlhelp32.h>
#include <iphlpapi.h>
-#include <snmp.h>
+#include "netstat.h"
-#include "trace.h"
-#include "resource.h"
+CHAR localname[HOSTNAMELEN], remotename[HOSTNAMELEN];
+CHAR remoteport[PORTNAMELEN], localport[PORTNAMELEN];
+CHAR localaddr[ADDRESSLEN], remoteaddr[ADDRESSLEN];
+enum ProtoType {IP, TCP, UDP, ICMP} Protocol;
+DWORD Interval; /* time to pause between printing output */
-#define MAX_RESLEN 4000
-
-//
-// Possible TCP endpoint states
-//
-static char TcpState[][32] = {
- "???",
- "CLOSED",
- "LISTENING",
- "SYN_SENT",
- "SYN_RCVD",
- "ESTABLISHED",
- "FIN_WAIT1",
- "FIN_WAIT2",
- "CLOSE_WAIT",
- "CLOSING",
- "LAST_ACK",
- "TIME_WAIT",
- "DELETE_TCB"
+/* TCP endpoint states */
+TCHAR TcpState[][32] = {
+ _T("???"),
+ _T("CLOSED"),
+ _T("LISTENING"),
+ _T("SYN_SENT"),
+ _T("SYN_RCVD"),
+ _T("ESTABLISHED"),
+ _T("FIN_WAIT1"),
+ _T("FIN_WAIT2"),
+ _T("CLOSE_WAIT"),
+ _T("CLOSING"),
+ _T("LAST_ACK"),
+ _T("TIME_WAIT"),
+ _T("DELETE_TCB")
};
-static VOID PrintError(DWORD ErrorCode)
+
+/*
+ *
+ * Parse command line parameters and set any options
+ *
+ */
+BOOL ParseCmdline(int argc, char* argv[])
{
- LPVOID lpMsgBuf;
+ INT i;
- FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
- NULL, ErrorCode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
- (LPTSTR)&lpMsgBuf, 0, NULL);
- printf("%s\n", (TCHAR*)lpMsgBuf);
- LocalFree(lpMsgBuf);
+ TCHAR Proto[5];
+
+ if ((argc == 1) || (isdigit(*argv[1])))
+ bNoOptions = TRUE;
+
+ /* Parse command line for options we have been given. */
+ for (i = 1; i < argc; i++)
+ {
+ if ( (argc > 1)&&(argv[i][0] == '-') )
+ {
+ TCHAR c;
+
+ while ((c = *++argv[i]) != '\0')
+ {
+ switch (tolower(c))
+ {
+ case 'a' :
+ //_tprintf(_T("got a\n"));
+ bDoShowAllCons = TRUE;
+ break;
+ case 'e' :
+ //_tprintf(_T("got e\n"));
+ bDoShowEthStats = TRUE;
+ break;
+ case 'n' :
+ //_tprintf(_T("got n\n"));
+ bDoShowNumbers = TRUE;
+ break;
+ case 's' :
+ //_tprintf(_T("got s\n"));
+ bDoShowProtoStats = TRUE;
+ break;
+ case 'p' :
+ //_tprintf(_T("got p\n"));
+ bDoShowProtoCons = TRUE;
+
+ strncpy(Proto, (++argv)[i], sizeof(Proto));
+ if (!_tcsicmp( "IP", Proto ))
+ Protocol = IP;
+ else if (!_tcsicmp( "ICMP", Proto ))
+ Protocol = ICMP;
+ else if (!_tcsicmp( "TCP", Proto ))
+ Protocol = TCP;
+ else if (!_tcsicmp( "UDP", Proto ))
+ Protocol = UDP;
+ else
+ {
+ Usage();
+ return EXIT_FAILURE;
+ }
+ (--argv)[i]; /* move pointer back down to previous argv */
+ break;
+ case 'r' :
+ bDoShowRouteTable = FALSE;
+ break;
+ case 'v' :
+ _tprintf(_T("got v\n"));
+ bDoDispSeqComp = TRUE;
+ break;
+ default :
+ Usage();
+ return EXIT_FAILURE;
+ }
+ }
+ }
+ else if (isdigit(*argv[i]))
+ {
+ _stscanf(argv[i], "%lu", &Interval);
+ bLoopOutput = TRUE;
+ }
+// else
+// {
+// Usage();
+// EXIT_FAILURE;
+// }
+ }
+
+ return EXIT_SUCCESS;
}
-#if 0
-static void ShowTcpStatistics()
+/* Simulate Microsofts netstat utility output. It's a bit
+ * ugly and over nested, but it is a fairly acurate simulation
+ * It was easier for testing, I'll rewrite it later with flags
+ * For now, it works*/
+BOOL DisplayOutput()
+// FIXME: This whole function needs rewriting
{
- MIB_TCPSTATS TcpStatsMIB;
- GetTcpStatistics(&TcpStatsMIB);
+ if (bNoOptions)
+ {
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowTcpTable();
+ return EXIT_SUCCESS;
+ }
- _tprintf(_T("TCP/IP Statistics\t\n"));
- _tprintf(_T(" time-out algorithm:\t\t%lu\n"), TcpStatsMIB.dwRtoAlgorithm);
- _tprintf(_T(" minimum time-out:\t\t%lu\n"), TcpStatsMIB.dwRtoMin);
- _tprintf(_T(" maximum time-out:\t\t%lu\n"), TcpStatsMIB.dwRtoMax);
- _tprintf(_T(" maximum connections:\t\t%lu\n"), TcpStatsMIB.dwMaxConn);
- _tprintf(_T(" active opens:\t\t\t%lu\n"), TcpStatsMIB.dwActiveOpens);
- _tprintf(_T(" passive opens:\t\t\t%lu\n"), TcpStatsMIB.dwPassiveOpens);
- _tprintf(_T(" failed attempts:\t\t%lu\n"), TcpStatsMIB.dwAttemptFails);
- _tprintf(_T(" established connections reset:\t%lu\n"), TcpStatsMIB.dwEstabResets);
- _tprintf(_T(" established connections:\t%lu\n"), TcpStatsMIB.dwCurrEstab);
- _tprintf(_T(" segments received:\t\t%lu\n"), TcpStatsMIB.dwInSegs);
- _tprintf(_T(" segment sent:\t\t\t%lu\n"), TcpStatsMIB.dwOutSegs);
- _tprintf(_T(" segments retransmitted:\t\t%lu\n"), TcpStatsMIB.dwRetransSegs);
- _tprintf(_T(" incoming errors:\t\t%lu\n"), TcpStatsMIB.dwInErrs);
- _tprintf(_T(" outgoing resets:\t\t%lu\n"), TcpStatsMIB.dwOutRsts);
- _tprintf(_T(" cumulative connections:\t\t%lu\n"), TcpStatsMIB.dwNumConns);
+ if (bDoShowEthStats)
+ {
+ ShowEthernetStatistics();
+ return EXIT_SUCCESS;
+ }
+
+ if (bDoShowRouteTable)
+ {
+ if (system("route print") == -1)
+ {
+ //mingw doesn't have lib for _tsystem
+ _tprintf(_T("cannot find 'route.exe'\n"));
+ return EXIT_FAILURE;
+ }
+ }
+
+ /* output connections: -a */
+ if (bDoShowAllCons)
+ {
+ /* filter out certain protocols: -p */
+ if (bDoShowProtoCons)
+ {
+ /* do we want to list the stats: -s */
+ if (bDoShowProtoStats)
+ {
+ switch (Protocol)
+ {
+ case IP :
+ ShowIpStatistics();
+ break;
+ case ICMP :
+ ShowIcmpStatistics();
+ break;
+ case TCP :
+ ShowTcpStatistics();
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowTcpTable();
+ break;
+ case UDP :
+ ShowUdpStatistics();
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowUdpTable();
+ break;
+ default :
+ break;
+ }
+ return EXIT_SUCCESS;
+ }
+ else
+ {
+ switch (Protocol)
+ {
+ case IP :
+ break;
+ case ICMP :
+ ShowIcmpStatistics();
+ break;
+ case TCP :
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowTcpTable();
+ break;
+ case UDP :
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowUdpTable();
+ break;
+ default :
+ break;
+ }
+ return EXIT_SUCCESS;
+ }
+
+ }
+ else
+ {
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowTcpTable();
+ ShowUdpTable();
+ return EXIT_SUCCESS;
+ }
+ }
+
+ /* do we want to list the stats: -s */
+ if (bDoShowProtoStats)
+ {
+ if (bDoShowProtoCons) // -p
+ {
+ /* show individual protocols only */
+ switch (Protocol)
+ {
+ case IP :
+ ShowIpStatistics();
+ break;
+ case ICMP :
+ ShowIcmpStatistics();
+ break;
+ case TCP :
+ ShowTcpStatistics();
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowTcpTable();
+ break;
+ case UDP :
+ ShowUdpStatistics();
+ _tprintf(_T("\nActive Connections\n"));
+ _tprintf(_T("\n Proto Local Address Foreign Address State\n"));
+ ShowUdpTable();
+ break;
+ default :
+ break;
+ }
+ return EXIT_SUCCESS;
+ }
+ else
+ {
+ /* list the lot */
+ ShowIpStatistics();
+ ShowIcmpStatistics();
+ ShowTcpStatistics();
+ ShowUdpStatistics();
+ return EXIT_SUCCESS;
+ }
+ }
+ return EXIT_SUCCESS;
}
-static void ShowUdpStatistics()
+
+/* format message string and display output */
+DWORD DoFormatMessage(DWORD ErrorCode)
{
- MIB_UDPSTATS UDPStatsMIB;
- GetUdpStatistics(&UDPStatsMIB);
+ LPVOID lpMsgBuf;
+ DWORD RetVal;
- _tprintf(_T("UDP Statistics\t\n"));
- _tprintf(_T(" received datagrams:\t\t\t%lu\n"), UDPStatsMIB.dwInDatagrams);
- _tprintf(_T(" datagrams for which no port exists:\t%lu\n"), UDPStatsMIB.dwNoPorts);
- _tprintf(_T(" errors on received datagrams:\t\t%lu\n"), UDPStatsMIB.dwInErrors);
- _tprintf(_T(" sent datagrams:\t\t\t\t%lu\n"), UDPStatsMIB.dwOutDatagrams);
- _tprintf(_T(" number of entries in listener table:\t%lu\n"), UDPStatsMIB.dwNumAddrs);
+ if ((RetVal = FormatMessage(
+ FORMAT_MESSAGE_ALLOCATE_BUFFER |
+ FORMAT_MESSAGE_FROM_SYSTEM |
+ FORMAT_MESSAGE_IGNORE_INSERTS,
+ NULL,
+ ErrorCode,
+ MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), /* Default language */
+ (LPTSTR) &lpMsgBuf,
+ 0,
+ NULL )))
+ {
+ _tprintf(_T("%s"), (LPTSTR)lpMsgBuf);
+
+ LocalFree(lpMsgBuf);
+ /* return number of TCHAR's stored in output buffer
+ * excluding '\0' - as FormatMessage does*/
+ return RetVal;
+ }
+ else
+ return 0;
}
-static void ShowIpStatistics()
+
+VOID ShowIpStatistics()
{
- MIB_IPSTATS IPStatsMIB;
- GetIpStatistics(&IPStatsMIB);
+ PMIB_IPSTATS pIpStats;
+ DWORD dwRetVal;
- _tprintf(_T("IP Statistics\t\n"));
- _tprintf(_T(" IP forwarding enabled or disabled:\t%lu\n"), IPStatsMIB.dwForwarding);
- _tprintf(_T(" default time-to-live:\t\t\t%lu\n"), IPStatsMIB.dwDefaultTTL);
- _tprintf(_T(" datagrams received:\t\t\t%lu\n"), IPStatsMIB.dwInReceives);
- _tprintf(_T(" received header errors:\t\t\t%lu\n"), IPStatsMIB.dwInHdrErrors);
- _tprintf(_T(" received address errors:\t\t%lu\n"), IPStatsMIB.dwInAddrErrors);
- _tprintf(_T(" datagrams forwarded:\t\t\t%lu\n"), IPStatsMIB.dwForwDatagrams);
- _tprintf(_T(" datagrams with unknown protocol:\t%lu\n"), IPStatsMIB.dwInUnknownProtos);
- _tprintf(_T(" received datagrams discarded:\t\t%lu\n"), IPStatsMIB.dwInDiscards);
- _tprintf(_T(" received datagrams delivered:\t\t%lu\n"), IPStatsMIB.dwInDelivers);
- _tprintf(_T(" sent datagrams discarded:\t\t%lu\n"), IPStatsMIB.dwOutDiscards);
- _tprintf(_T(" datagrams for which no route exists:\t%lu\n"), IPStatsMIB.dwOutNoRoutes);
- _tprintf(_T(" datagrams for which frags didn't arrive:%lu\n"), IPStatsMIB.dwReasmTimeout);
- _tprintf(_T(" datagrams requiring reassembly:\t\t%lu\n"), IPStatsMIB.dwReasmReqds);
- _tprintf(_T(" successful reassemblies:\t\t%lu\n"), IPStatsMIB.dwReasmOks);
- _tprintf(_T(" failed reassemblies:\t\t\t%lu\n"), IPStatsMIB.dwReasmFails);
- _tprintf(_T(" successful fragmentations:\t\t%lu\n"), IPStatsMIB.dwFragOks);
- _tprintf(_T(" failed fragmentations:\t\t\t%lu\n"), IPStatsMIB.dwFragFails);
- _tprintf(_T(" datagrams fragmented:\t\t\t%lu\n"), IPStatsMIB.dwFragCreates);
- _tprintf(_T(" number of interfaces on computer:\t%lu\n"), IPStatsMIB.dwNumIf);
- _tprintf(_T(" number of IP address on computer:\t%lu\n"), IPStatsMIB.dwNumAddr);
- _tprintf(_T(" number of routes in routing table:\t%lu\n"), IPStatsMIB.dwNumRoutes);
+ pIpStats = (MIB_IPSTATS*) malloc(sizeof(MIB_IPSTATS));
+
+ if ((dwRetVal = GetIpStatistics(pIpStats)) == NO_ERROR)
+ {
+ _tprintf(_T("\nIPv4 Statistics\n\n"));
+ _tprintf(_T(" %-34s = %lu\n"), _T("Packets Recieved"), pIpStats->dwInReceives);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Received Header Errors"), pIpStats->dwInHdrErrors);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Received Address Errors"), pIpStats->dwInAddrErrors);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Datagrams Forwarded"), pIpStats->dwForwDatagrams);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Unknown Protocols Recieved"), pIpStats->dwInUnknownProtos);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Received Packets Discarded"), pIpStats->dwInDiscards);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Recieved Packets Delivered"), pIpStats->dwInDelivers);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Output Requests"), pIpStats->dwOutRequests);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Routing Discards"), pIpStats->dwRoutingDiscards);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Discarded Output Packets"), pIpStats->dwOutDiscards);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Output Packets No Route"), pIpStats->dwOutNoRoutes);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Reassembly Required"), pIpStats->dwReasmReqds);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Reassembly Succesful"), pIpStats->dwReasmOks);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Reassembly Failures"), pIpStats->dwReasmFails);
+ // _tprintf(_T(" %-34s = %lu\n"), _T("Datagrams succesfully fragmented"), NULL); /* FIXME: what is this one? */
+ _tprintf(_T(" %-34s = %lu\n"), _T("Datagrams Failing Fragmentation"), pIpStats->dwFragFails);
+ _tprintf(_T(" %-34s = %lu\n"), _T("Fragments Created"), pIpStats->dwFragCreates);
+ }
+ else
+ DoFormatMessage(dwRetVal);
}
-static void ShowNetworkParams()
+VOID ShowIcmpStatistics()
{
- FIXED_INFO* FixedInfo;
- IP_ADDR_STRING* pIPAddr;
- ULONG ulOutBufLen;
+ PMIB_ICMP pIcmpStats;
DWORD dwRetVal;
- _tprintf(_T("Network Parameters\t\n"));
+ pIcmpStats = (MIB_ICMP*) malloc(sizeof(MIB_ICMP));
- FixedInfo = (FIXED_INFO*)GlobalAlloc(GPTR, sizeof(FIXED_INFO));
- ulOutBufLen = sizeof(FIXED_INFO);
- if (ERROR_BUFFER_OVERFLOW == GetNetworkParams(FixedInfo, &ulOutBufLen)) {
- GlobalFree(FixedInfo);
- FixedInfo =(FIXED_INFO*)GlobalAlloc(GPTR, ulOutBufLen);
+ if ((dwRetVal = GetIcmpStatistics(pIcmpStats)) == NO_ERROR)
+ {
+ _tprintf(_T("\nICMPv4 Statistics\n\n"));
+ _tprintf(_T(" Received Sent\n"));
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Messages"),
+ pIcmpStats->stats.icmpInStats.dwMsgs, pIcmpStats->stats.icmpOutStats.dwMsgs);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Errors"),
+ pIcmpStats->stats.icmpInStats.dwErrors, pIcmpStats->stats.icmpOutStats.dwErrors);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Destination Unreachable"),
+ pIcmpStats->stats.icmpInStats.dwDestUnreachs, pIcmpStats->stats.icmpOutStats.dwDestUnreachs);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Time Exceeded"),
+ pIcmpStats->stats.icmpInStats.dwTimeExcds, pIcmpStats->stats.icmpOutStats.dwTimeExcds);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Parameter Problems"),
+ pIcmpStats->stats.icmpInStats.dwParmProbs, pIcmpStats->stats.icmpOutStats.dwParmProbs);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Source Quenches"),
+ pIcmpStats->stats.icmpInStats.dwSrcQuenchs, pIcmpStats->stats.icmpOutStats.dwSrcQuenchs);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Redirects"),
+ pIcmpStats->stats.icmpInStats.dwRedirects, pIcmpStats->stats.icmpOutStats.dwRedirects);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Echos"),
+ pIcmpStats->stats.icmpInStats.dwEchos, pIcmpStats->stats.icmpOutStats.dwEchos);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Echo Replies"),
+ pIcmpStats->stats.icmpInStats.dwEchoReps, pIcmpStats->stats.icmpOutStats.dwEchoReps);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Timestamps"),
+ pIcmpStats->stats.icmpInStats.dwTimestamps, pIcmpStats->stats.icmpOutStats.dwTimestamps);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Timestamp Replies"),
+ pIcmpStats->stats.icmpInStats.dwTimestampReps, pIcmpStats->stats.icmpOutStats.dwTimestampReps);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Address Masks"),
+ pIcmpStats->stats.icmpInStats.dwAddrMasks, pIcmpStats->stats.icmpOutStats.dwAddrMasks);
+ _tprintf(_T(" %-25s %-11lu %lu\n"), _T("Address Mask Replies"),
+ pIcmpStats->stats.icmpInStats.dwAddrMaskReps, pIcmpStats->stats.icmpOutStats.dwAddrMaskReps);
}
- if ((dwRetVal = GetNetworkParams(FixedInfo, &ulOutBufLen))) {
- _tprintf(_T("Call to GetNetworkParams failed. Return Value: 0x%08lx\n"), dwRetVal);
- } else {
- printf(" Host Name: %s", FixedInfo->HostName);
- printf("\n Domain Name: %s", FixedInfo->DomainName);
- printf("\n DNS Servers:\t%s\n", FixedInfo->DnsServerList.IpAddress.String);
- pIPAddr = FixedInfo->DnsServerList.Next;
- while (pIPAddr) {
- printf("\t\t\t%s\n", pIPAddr->IpAddress.String);
- pIPAddr = pIPAddr->Next;
- }
+ else
+ DoFormatMessage(dwRetVal);
+
+}
+
+VOID ShowTcpStatistics()
+{
+ PMIB_TCPSTATS pTcpStats;
+ DWORD dwRetVal;
+
+ pTcpStats = (MIB_TCPSTATS*) malloc(sizeof(MIB_TCPSTATS));
+
+ if ((dwRetVal = GetTcpStatistics(pTcpStats)) == NO_ERROR)
+ {
+ _tprintf(_T("\nTCP Statistics for IPv4\n\n"));
+ _tprintf(_T(" %-35s = %lu\n"), _T("Active Opens"), pTcpStats->dwActiveOpens);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Passive Opens"), pTcpStats->dwPassiveOpens);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Failed Connection Attempts"), pTcpStats->dwAttemptFails);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Reset Connections"), pTcpStats->dwEstabResets);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Current Connections"), pTcpStats->dwCurrEstab);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Segments Recieved"), pTcpStats->dwInSegs);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Segments Sent"), pTcpStats->dwOutSegs);
+ _tprintf(_T(" %-35s = %lu\n"), _T("Segments Retransmitted"), pTcpStats->dwRetransSegs);
}
+ else
+ DoFormatMessage(dwRetVal);
}
-static void ShowAdapterInfo()
+VOID ShowUdpStatistics()
{
- IP_ADAPTER_INFO* pAdaptorInfo;
- ULONG ulOutBufLen;
+ PMIB_UDPSTATS pUdpStats;
DWORD dwRetVal;
- _tprintf(_T("\nAdaptor Information\t\n"));
- pAdaptorInfo = (IP_ADAPTER_INFO*)GlobalAlloc(GPTR, sizeof(IP_ADAPTER_INFO));
- ulOutBufLen = sizeof(IP_ADAPTER_INFO);
+ pUdpStats = (MIB_UDPSTATS*) malloc(sizeof(MIB_UDPSTATS));
- if (ERROR_BUFFER_OVERFLOW == GetAdaptersInfo(pAdaptorInfo, &ulOutBufLen)) {
- GlobalFree(pAdaptorInfo);
- pAdaptorInfo = (IP_ADAPTER_INFO*)GlobalAlloc(GPTR, ulOutBufLen);
+ if ((dwRetVal = GetUdpStatistics(pUdpStats)) == NO_ERROR)
+ {
+ _tprintf(_T("\nUDP Statistics for IPv4\n\n"));
+ _tprintf(_T(" %-21s = %lu\n"), _T("Datagrams Recieved"), pUdpStats->dwInDatagrams);
+ _tprintf(_T(" %-21s = %lu\n"), _T("No Ports"), pUdpStats->dwNoPorts);
+ _tprintf(_T(" %-21s = %lu\n"), _T("Recieve Errors"), pUdpStats->dwInErrors);
+ _tprintf(_T(" %-21s = %lu\n"), _T("Datagrams Sent"), pUdpStats->dwOutDatagrams);
}
- if ((dwRetVal = GetAdaptersInfo(pAdaptorInfo, &ulOutBufLen))) {
- _tprintf(_T("Call to GetAdaptersInfo failed. Return Value: 0x%08lx\n"), dwRetVal);
- } else {
- while (pAdaptorInfo) {
- printf(" AdapterName: %s\n", pAdaptorInfo->AdapterName);
- printf(" Description: %s\n", pAdaptorInfo->Description);
- pAdaptorInfo = pAdaptorInfo->Next;
+ else
+ DoFormatMessage(dwRetVal);
+}
+
+VOID ShowEthernetStatistics()
+{
+ PMIB_IFTABLE pIfTable;
+ DWORD dwSize = 0;
+ DWORD dwRetVal = 0;
+
+ pIfTable = (MIB_IFTABLE*) malloc(sizeof(MIB_IFTABLE));
+
+ if (GetIfTable(pIfTable, &dwSize, 0) == ERROR_INSUFFICIENT_BUFFER)
+ {
+ GlobalFree(pIfTable);
+ pIfTable = (MIB_IFTABLE*) malloc(dwSize);
+
+ if ((dwRetVal = GetIfTable(pIfTable, &dwSize, 0)) == NO_ERROR)
+ {
+ _tprintf(_T("Interface Statistics\n\n"));
+ _tprintf(_T(" Received Sent\n\n"));
+ _tprintf(_T("%-20s %14lu %15lu\n"), _T("Bytes"),
+ pIfTable->table[0].dwInOctets, pIfTable->table[0].dwOutOctets);
+ _tprintf(_T("%-20s %14lu %15lu\n"), _T("Unicast packets"),
+ pIfTable->table[0].dwInUcastPkts, pIfTable->table[0].dwOutUcastPkts);
+ _tprintf(_T("%-20s %14lu %15lu\n"), _T("Non-unicast packets"),
+ pIfTable->table[0].dwInNUcastPkts, pIfTable->table[0].dwOutNUcastPkts);
+ _tprintf(_T("%-20s %14lu %15lu\n"), _T("Discards"),
+ pIfTable->table[0].dwInDiscards, pIfTable->table[0].dwOutDiscards);
+ _tprintf(_T("%-20s %14lu %15lu\n"), _T("Errors"),
+ pIfTable->table[0].dwInErrors, pIfTable->table[0].dwOutErrors);
+ _tprintf(_T("%-20s %14lu\n"), _T("Unknown Protocols"),
+ pIfTable->table[0].dwInUnknownProtos);
}
+ else
+ DoFormatMessage(dwRetVal);
}
}
-#endif
-// Maximum string lengths for ASCII ip address and port names
-//
-#define HOSTNAMELEN 256
-#define PORTNAMELEN 256
-#define ADDRESSLEN HOSTNAMELEN+PORTNAMELEN
+VOID ShowTcpTable()
+{
+ PMIB_TCPTABLE tcpTable;
+ DWORD error, dwSize;
+ DWORD i;
-//
-// Our option flags
-//
-#define FLAG_SHOW_ALL_ENDPOINTS 1
-#define FLAG_SHOW_ETH_STATS 2
-#define FLAG_SHOW_NUMBERS 3
-#define FLAG_SHOW_PROT_CONNECTIONS 4
-#define FLAG_SHOW_ROUTE_TABLE 5
-#define FLAG_SHOW_PROT_STATS 6
-#define FLAG_SHOW_INTERVAL 7
+ // Get the table of TCP endpoints
+ dwSize = 0;
+ error = GetTcpTable(NULL, &dwSize, TRUE);
+ if (error != ERROR_INSUFFICIENT_BUFFER)
+ {
+ printf("Failed to snapshot TCP endpoints.\n");
+ DoFormatMessage(error);
+ exit(EXIT_FAILURE);
+ }
+ tcpTable = (PMIB_TCPTABLE)malloc(dwSize);
+ error = GetTcpTable(tcpTable, &dwSize, TRUE );
+ if (error)
+ {
+ printf("Failed to snapshot TCP endpoints table.\n");
+ DoFormatMessage(error);
+ exit(EXIT_FAILURE);
+ }
+ // Dump the TCP table
+ for (i = 0; i < tcpTable->dwNumEntries; i++)
+ {
+ if (bDoShowAllCons || (tcpTable->table[i].dwState ==
+ MIB_TCP_STATE_ESTAB))
+ {
+ sprintf(localaddr, "%s:%s",
+ GetIpHostName(TRUE, tcpTable->table[i].dwLocalAddr, localname, HOSTNAMELEN),
+ GetPortName(tcpTable->table[i].dwLocalPort, "tcp", localport, PORTNAMELEN));
+ sprintf(remoteaddr, "%s:%s",
+ GetIpHostName(FALSE, tcpTable->table[i].dwRemoteAddr, remotename, HOSTNAMELEN),
+ tcpTable->table[i].dwRemoteAddr ?
+ GetPortName(tcpTable->table[i].dwRemotePort, "tcp", remoteport, PORTNAMELEN):
+ "0");
+ _tprintf(_T(" %-6s %-22s %-22s %s\n"), _T("TCP"), localaddr, remoteaddr, TcpState[tcpTable->table[i].dwState]);
+ }
+ }
+}
-// Undocumented extended information structures available only on XP and higher
-typedef struct {
- DWORD dwState; // state of the connection
- DWORD dwLocalAddr; // address on local computer
- DWORD dwLocalPort; // port number on local computer
- DWORD dwRemoteAddr; // address on remote computer
- DWORD dwRemotePort; // port number on remote computer
- DWORD dwProcessId;
-} MIB_TCPEXROW, *PMIB_TCPEXROW;
+VOID ShowUdpTable()
+{
+ PMIB_UDPTABLE udpTable;
+ DWORD error, dwSize;
+ DWORD i;
-typedef struct {
- DWORD dwNumEntries;
- MIB_TCPEXROW table[ANY_SIZE];
-} MIB_TCPEXTABLE, *PMIB_TCPEXTABLE;
+ // Get the table of UDP endpoints
+ dwSize = 0;
+ error = GetUdpTable(NULL, &dwSize, TRUE);
+ if (error != ERROR_INSUFFICIENT_BUFFER)
+ {
+ printf("Failed to snapshot UDP endpoints.\n");
+ DoFormatMessage(error);
+ exit(EXIT_FAILURE);
+ }
+ udpTable = (PMIB_UDPTABLE)malloc(dwSize);
+ error = GetUdpTable(udpTable, &dwSize, TRUE);
+ if (error)
+ {
+ printf("Failed to snapshot UDP endpoints table.\n");
+ DoFormatMessage(error);
+ exit(EXIT_FAILURE);
+ }
-typedef struct {
- DWORD dwLocalAddr; // address on local computer
- DWORD dwLocalPort; // port number on local computer
- DWORD dwProcessId;
-} MIB_UDPEXROW, *PMIB_UDPEXROW;
+ // Dump the UDP table
+ for (i = 0; i < udpTable->dwNumEntries; i++)
+ {
+ sprintf(localaddr, "%s:%s",
+ GetIpHostName(TRUE, udpTable->table[i].dwLocalAddr, localname, HOSTNAMELEN),
+ GetPortName(udpTable->table[i].dwLocalPort, "tcp", localport, PORTNAMELEN));
+ _tprintf(_T(" %-6s %-22s %-22s\n"), _T("UDP"), localaddr, _T(":*:"));
+ }
+}
-typedef struct {
- DWORD dwNumEntries;
- MIB_UDPEXROW table[ANY_SIZE];
-} MIB_UDPEXTABLE, *PMIB_UDPEXTABLE;
-
//
// GetPortName
//
// Translate port numbers into their text equivalent if there is one
//
-static PCHAR
-GetPortName(DWORD Flags, UINT port, PCHAR proto, PCHAR name, int namelen)
+PCHAR
+GetPortName(UINT port, PCHAR proto, PCHAR name, int namelen)
{
- struct servent *psrvent;
+ struct servent *psrvent;
- if (Flags & FLAG_SHOW_NUMBERS) {
- sprintf(name, "%d", htons((WORD)port));
- return name;
- }
- // Try to translate to a name
- if ((psrvent = getservbyport(port, proto))) {
- strcpy(name, psrvent->s_name );
- } else {
- sprintf(name, "%d", htons((WORD)port));
- }
- return name;
+ if (bDoShowNumbers) {
+ sprintf(name, "%d", htons((WORD)port));
+ return name;
+ }
+ // Try to translate to a name
+ if ((psrvent = getservbyport(port, proto))) {
+ strcpy(name, psrvent->s_name );
+ } else {
+ sprintf(name, "%d", htons((WORD)port));
+ }
+ return name;
}
@@ -257,265 +593,100 @@
//
// Translate IP addresses into their name-resolved form if possible.
//
-static PCHAR
-GetIpHostName(DWORD Flags, BOOL local, UINT ipaddr, PCHAR name, int namelen)
+PCHAR
+GetIpHostName(BOOL local, UINT ipaddr, PCHAR name, int namelen)
{
-// struct hostent *phostent;
- UINT nipaddr;
+// struct hostent *phostent;
+ UINT nipaddr;
- // Does the user want raw numbers?
- nipaddr = htonl(ipaddr);
- if (Flags & FLAG_SHOW_NUMBERS ) {
- sprintf(name, "%d.%d.%d.%d",
- (nipaddr >> 24) & 0xFF,
- (nipaddr >> 16) & 0xFF,
- (nipaddr >> 8) & 0xFF,
- (nipaddr) & 0xFF);
- return name;
- }
+ // Does the user want raw numbers?
+ nipaddr = htonl(ipaddr);
+ if (bDoShowNumbers) {
+ sprintf(name, "%d.%d.%d.%d",
+ (nipaddr >> 24) & 0xFF,
+ (nipaddr >> 16) & 0xFF,
+ (nipaddr >> 8) & 0xFF,
+ (nipaddr) & 0xFF);
+ return name;
+ }
- name[0] = _T('\0');
+ name[0] = _T('\0');
- // Try to translate to a name
- if (!ipaddr) {
- if (!local) {
- sprintf(name, "%d.%d.%d.%d",
- (nipaddr >> 24) & 0xFF,
- (nipaddr >> 16) & 0xFF,
- (nipaddr >> 8) & 0xFF,
- (nipaddr) & 0xFF);
- } else {
- //gethostname(name, namelen);
- }
- } else if (ipaddr == 0x0100007f) {
- if (local) {
- //gethostname(name, namelen);
- } else {
- strcpy(name, "localhost");
- }
-// } else if (phostent = gethostbyaddr((char*)&ipaddr, sizeof(nipaddr), PF_INET)) {
-// strcpy(name, phostent->h_name);
- } else {
-#if 0
- int i1, i2, i3, i4;
-
- i1 = (nipaddr >> 24) & 0x000000FF;
- i2 = (nipaddr >> 16) & 0x000000FF;
- i3 = (nipaddr >> 8) & 0x000000FF;
- i4 = (nipaddr) & 0x000000FF;
-
- i1 = 10;
- i2 = 20;
- i3 = 30;
- i4 = 40;
-
- sprintf(name, "%d.%d.%d.%d", i1,i2,i3,i4);
-#else
- sprintf(name, "%d.%d.%d.%d",
- ((nipaddr >> 24) & 0x000000FF),
- ((nipaddr >> 16) & 0x000000FF),
- ((nipaddr >> 8) & 0x000000FF),
- ((nipaddr) & 0x000000FF));
-#endif
- }
- return name;
+ // Try to translate to a name
+ if (!ipaddr) {
+ if (!local) {
+ sprintf(name, "%d.%d.%d.%d",
+ (nipaddr >> 24) & 0xFF,
+ (nipaddr >> 16) & 0xFF,
+ (nipaddr >> 8) & 0xFF,
+ (nipaddr) & 0xFF);
+ } else {
+ //gethostname(name, namelen);
+ }
+ } else if (ipaddr == 0x0100007f) {
+ if (local) {
+ //gethostname(name, namelen);
+ } else {
+ strcpy(name, "localhost");
+ }
+// } else if (phostent = gethostbyaddr((char*)&ipaddr, sizeof(nipaddr), PF_INET)) {
+// strcpy(name, phostent->h_name);
+ } else {
+ sprintf(name, "%d.%d.%d.%d",
+ ((nipaddr >> 24) & 0x000000FF),
+ ((nipaddr >> 16) & 0x000000FF),
+ ((nipaddr >> 8) & 0x000000FF),
+ ((nipaddr) & 0x000000FF));
+ }
+ return name;
}
-static BOOLEAN usage(void)
+VOID Usage()
{
- TCHAR buffer[MAX_RESLEN];
-
- LoadString(GetModuleHandle(NULL), IDS_APP_USAGE, buffer, sizeof(buffer)/sizeof(buffer[0]));
- _fputts(buffer, stderr);
- return FALSE;
+ _tprintf(_T("Displays current TCP/IP protocol statistics and network connections.\n\n"
+ "NETSTAT [-a] [-e] [-n] [-s] [-p proto] [-r] [interval]\n\n"
+ " -a Displays all connections and listening ports.\n"
+ " -e Displays Ethernet statistics. May be combined with -s\n"
+ " option\n"
+ " -n Displays address and port numbers in numeric form.\n"
+ " -p proto Shows connections for protocol 'proto' TCP or UDP.\n"
+ " If used with the -s option to display\n"
+ " per-protocol statistics, 'proto' may be TCP, UDP, or IP.\n"
+ " -r Displays the current routing table.\n"
+ " -s Displays per-protocol statistics. By default, Statistics are\n"
+ " shown for IP, ICMP, TCP and UDP;\n"
+ " the -p option may be used to specify a subset of the default.\n"
+ " interval Redisplays selected statistics every 'interval' seconds.\n"
+ " Press CTRL+C to stop redisplaying. By default netstat will\n"
+ " print the current information only once.\n"));
}
-//
-// GetOptions
-//
-// Parses the command line arguments.
-//
-static BOOLEAN
-GetOptions(int argc, char *argv[], PDWORD pFlags)
-{
- int i, j;
- BOOLEAN skipArgument;
- *pFlags = 0;
- for (i = 1; i < argc; i++) {
- skipArgument = FALSE;
- switch (argv[i][0]) {
- case '-':
- case '/':
- j = 1;
- while (argv[i][j]) {
- switch (toupper(argv[i][j])) {
- case 'A':
- *pFlags |= FLAG_SHOW_ALL_ENDPOINTS;
- break;
- case 'E':
- *pFlags |= FLAG_SHOW_ETH_STATS;
- break;
- case 'N':
- *pFlags |= FLAG_SHOW_NUMBERS;
- break;
- case 'P':
- *pFlags |= FLAG_SHOW_PROT_CONNECTIONS;
- break;
- case 'R':
- *pFlags |= FLAG_SHOW_ROUTE_TABLE;
- break;
- case 'S':
- *pFlags |= FLAG_SHOW_PROT_STATS;
- break;
- default:
- return usage();
- }
- if (skipArgument) break;
- j++;
- }
- break;
- case 'i':
- *pFlags |= FLAG_SHOW_INTERVAL;
- break;
- default:
- return usage();
- }
- }
- return TRUE;
-}
-#if 1
- CHAR localname[HOSTNAMELEN], remotename[HOSTNAMELEN];
- CHAR remoteport[PORTNAMELEN], localport[PORTNAMELEN];
- CHAR localaddr[ADDRESSLEN], remoteaddr[ADDRESSLEN];
-
-int main(int argc, char *argv[])
-{
- PMIB_TCPTABLE tcpTable;
- PMIB_UDPTABLE udpTable;
- DWORD error, dwSize;
- DWORD i, flags;
-
- // Get options
- if (!GetOptions(argc, argv, &flags)) {
- return -1;
- } else {
- // Get the table of TCP endpoints
- dwSize = 0;
- error = GetTcpTable(NULL, &dwSize, TRUE);
- if (error != ERROR_INSUFFICIENT_BUFFER) {
- printf("Failed to snapshot TCP endpoints.\n");
- PrintError(error);
- return -1;
- }
- tcpTable = (PMIB_TCPTABLE)malloc(dwSize);
- error = GetTcpTable(tcpTable, &dwSize, TRUE );
- if (error) {
[truncated at 1000 lines; 356 more skipped]