/* Copyright (C) 2003,2006 Juan Lang * Copyright (C) 2007 TransGaming Technologies Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA * * This file implements statistics getting using the /proc filesystem exported * by Linux, and maybe other OSes. */ #include "config.h" #include "wine/port.h" #include "wine/debug.h" #include #include #include #include #include #ifdef HAVE_ALIAS_H #include #endif #ifdef HAVE_SYS_SOCKET_H #include #endif #ifdef HAVE_SYS_SOCKETVAR_H #include #endif #ifdef HAVE_NETINET_IN_H #include #endif #ifdef HAVE_ARPA_INET_H #include #endif #ifdef HAVE_NET_IF_H #include #endif #ifdef HAVE_NET_IF_DL_H #include #endif #ifdef HAVE_NET_IF_TYPES_H #include #endif #ifdef HAVE_NET_ROUTE_H #include #endif #ifdef HAVE_NET_IF_ARP_H #include #endif #ifdef HAVE_NETINET_IF_ETHER_H #include #endif #ifdef HAVE_NETINET_IP_H #include #endif #ifdef HAVE_NETINET_TCP_H #include #endif #ifdef HAVE_NETINET_TCP_FSM_H #include #endif #ifdef HAVE_NETINET_IN_PCB_H #include #endif #ifdef HAVE_NETINET_TCP_VAR_H #include #endif #ifdef HAVE_NETINET_TCP_TIMER_H #include #endif #ifdef HAVE_NETINET_IN_SYSTM_H #include #endif #ifdef HAVE_NETINET_IP_ICMP_H #include #endif #ifdef HAVE_NETINET_ICMP_VAR_H #include #endif #ifdef HAVE_NETINET_IP_VAR_H #include #endif #ifdef HAVE_NETINET_UDP_H #include #endif #ifdef HAVE_NETINET_UDP_VAR_H #include #endif #ifdef HAVE_SYS_SYSCTL_H #include #endif #ifndef ROUNDUP #define ROUNDUP(a) \ ((a) > 0 ? (1 + (((a) - 1) | (sizeof(long) - 1))) : sizeof(long)) #endif #ifndef ADVANCE #define ADVANCE(x, n) (x += ROUNDUP(((struct sockaddr *)n)->sa_len)) #endif #include "windef.h" #include "winbase.h" #include "iprtrmib.h" #include "ifenum.h" #include "ipstats.h" #ifndef HAVE_NETINET_TCP_FSM_H #define TCPS_ESTABLISHED 1 #define TCPS_SYN_SENT 2 #define TCPS_SYN_RECEIVED 3 #define TCPS_FIN_WAIT_1 4 #define TCPS_FIN_WAIT_2 5 #define TCPS_TIME_WAIT 6 #define TCPS_CLOSED 7 #define TCPS_CLOSE_WAIT 8 #define TCPS_LAST_ACK 9 #define TCPS_LISTEN 10 #define TCPS_CLOSING 11 #endif #ifndef RTF_MULTICAST #define RTF_MULTICAST 0 /* Not available on NetBSD/OpenBSD */ #endif WINE_DEFAULT_DEBUG_CHANNEL(iphlpapi); DWORD getInterfaceStatsByName(const char *name, PMIB_IFROW entry) { #if defined(HAVE_SYS_SYSCTL_H) && defined(NET_RT_IFLIST) int mib[] = {CTL_NET, PF_ROUTE, 0, AF_INET, NET_RT_IFLIST, if_nametoindex(name)}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) size_t needed; char *buf, *end; struct if_msghdr *ifm; struct if_data ifdata; if (!name || !entry) return ERROR_INVALID_PARAMETER; if(sysctl(mib, MIB_LEN, NULL, &needed, NULL, 0) == -1) { ERR ("failed to get size of iflist\n"); return ERROR_NOT_SUPPORTED; } buf = HeapAlloc (GetProcessHeap (), 0, needed); if (!buf) return ERROR_NOT_SUPPORTED; if(sysctl(mib, MIB_LEN, buf, &needed, NULL, 0) == -1) { ERR ("failed to get iflist\n"); HeapFree (GetProcessHeap (), 0, buf); return ERROR_NOT_SUPPORTED; } else for ( end = buf + needed; buf < end; buf += ifm->ifm_msglen) { ifm = (struct if_msghdr *) buf; if(ifm->ifm_type == RTM_IFINFO && ifm->ifm_data.ifi_type == IFT_ETHER) { ifdata = ifm->ifm_data; entry->dwMtu = ifdata.ifi_mtu; entry->dwSpeed = ifdata.ifi_baudrate; entry->dwInOctets = ifdata.ifi_ibytes; entry->dwInErrors = ifdata.ifi_ierrors; entry->dwInDiscards = ifdata.ifi_iqdrops; entry->dwInUcastPkts = ifdata.ifi_ipackets; entry->dwInNUcastPkts = ifdata.ifi_imcasts; entry->dwOutOctets = ifdata.ifi_obytes; entry->dwOutUcastPkts = ifdata.ifi_opackets; entry->dwOutErrors = ifdata.ifi_oerrors; HeapFree (GetProcessHeap (), 0, buf); return NO_ERROR; } } HeapFree (GetProcessHeap (), 0, buf); return ERROR_NOT_SUPPORTED; #else /* get interface stats from /proc/net/dev, no error if can't no inUnknownProtos, outNUcastPkts, outQLen */ FILE *fp; if (!name || !entry) return ERROR_INVALID_PARAMETER; fp = fopen("/proc/net/dev", "r"); if (fp) { char buf[512] = { 0 }, *ptr; int nameLen = strlen(name), nameFound = 0; ptr = fgets(buf, sizeof(buf), fp); while (ptr && !nameFound) { while (*ptr && isspace(*ptr)) ptr++; if (strncasecmp(ptr, name, nameLen) == 0 && *(ptr + nameLen) == ':') nameFound = 1; else ptr = fgets(buf, sizeof(buf), fp); } if (nameFound) { char *endPtr; ptr += nameLen + 1; if (ptr && *ptr) { entry->dwInOctets = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwInUcastPkts = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwInErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwInDiscards = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { strtoul(ptr, &endPtr, 10); /* skip */ ptr = endPtr; } if (ptr && *ptr) { strtoul(ptr, &endPtr, 10); /* skip */ ptr = endPtr; } if (ptr && *ptr) { strtoul(ptr, &endPtr, 10); /* skip */ ptr = endPtr; } if (ptr && *ptr) { entry->dwInNUcastPkts = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwOutOctets = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwOutUcastPkts = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwOutErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { entry->dwOutDiscards = strtoul(ptr, &endPtr, 10); ptr = endPtr; } } fclose(fp); } else { ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; } return NO_ERROR; #endif } DWORD getICMPStats(MIB_ICMP *stats) { #if defined(HAVE_SYS_SYSCTL_H) && defined(ICMPCTL_STATS) int mib[] = {CTL_NET, PF_INET, IPPROTO_ICMP, ICMPCTL_STATS}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) size_t needed; struct icmpstat icmp_stat; int i; if (!stats) return ERROR_INVALID_PARAMETER; needed = sizeof(icmp_stat); if(sysctl(mib, MIB_LEN, &icmp_stat, &needed, NULL, 0) == -1) { ERR ("failed to get icmpstat\n"); return ERROR_NOT_SUPPORTED; } /*in stats */ stats->stats.icmpInStats.dwMsgs = icmp_stat.icps_badcode + icmp_stat.icps_checksum + icmp_stat.icps_tooshort + icmp_stat.icps_badlen; for(i = 0; i <= ICMP_MAXTYPE; i++) stats->stats.icmpInStats.dwMsgs += icmp_stat.icps_inhist[i]; stats->stats.icmpInStats.dwErrors = icmp_stat.icps_badcode + icmp_stat.icps_tooshort + icmp_stat.icps_checksum + icmp_stat.icps_badlen; stats->stats.icmpInStats.dwDestUnreachs = icmp_stat.icps_inhist[ICMP_UNREACH]; stats->stats.icmpInStats.dwTimeExcds = icmp_stat.icps_inhist[ICMP_TIMXCEED]; stats->stats.icmpInStats.dwParmProbs = icmp_stat.icps_inhist[ICMP_PARAMPROB]; stats->stats.icmpInStats.dwSrcQuenchs = icmp_stat.icps_inhist[ICMP_SOURCEQUENCH]; stats->stats.icmpInStats.dwRedirects = icmp_stat.icps_inhist[ICMP_REDIRECT]; stats->stats.icmpInStats.dwEchos = icmp_stat.icps_inhist[ICMP_ECHO]; stats->stats.icmpInStats.dwEchoReps = icmp_stat.icps_inhist[ICMP_ECHOREPLY]; stats->stats.icmpInStats.dwTimestamps = icmp_stat.icps_inhist[ICMP_TSTAMP]; stats->stats.icmpInStats.dwTimestampReps = icmp_stat.icps_inhist[ICMP_TSTAMPREPLY]; stats->stats.icmpInStats.dwAddrMasks = icmp_stat.icps_inhist[ICMP_MASKREQ]; stats->stats.icmpInStats.dwAddrMaskReps = icmp_stat.icps_inhist[ICMP_MASKREPLY]; /* out stats */ stats->stats.icmpOutStats.dwMsgs = icmp_stat.icps_oldshort + icmp_stat.icps_oldicmp; for(i = 0; i <= ICMP_MAXTYPE; i++) stats->stats.icmpOutStats.dwMsgs += icmp_stat.icps_outhist[i]; stats->stats.icmpOutStats.dwErrors = icmp_stat.icps_oldshort + icmp_stat.icps_oldicmp; stats->stats.icmpOutStats.dwDestUnreachs = icmp_stat.icps_outhist[ICMP_UNREACH]; stats->stats.icmpOutStats.dwTimeExcds = icmp_stat.icps_outhist[ICMP_TIMXCEED]; stats->stats.icmpOutStats.dwParmProbs = icmp_stat.icps_outhist[ICMP_PARAMPROB]; stats->stats.icmpOutStats.dwSrcQuenchs = icmp_stat.icps_outhist[ICMP_SOURCEQUENCH]; stats->stats.icmpOutStats.dwRedirects = icmp_stat.icps_outhist[ICMP_REDIRECT]; stats->stats.icmpOutStats.dwEchos = icmp_stat.icps_outhist[ICMP_ECHO]; stats->stats.icmpOutStats.dwEchoReps = icmp_stat.icps_outhist[ICMP_ECHOREPLY]; stats->stats.icmpOutStats.dwTimestamps = icmp_stat.icps_outhist[ICMP_TSTAMP]; stats->stats.icmpOutStats.dwTimestampReps = icmp_stat.icps_outhist[ICMP_TSTAMPREPLY]; stats->stats.icmpOutStats.dwAddrMasks = icmp_stat.icps_outhist[ICMP_MASKREQ]; stats->stats.icmpOutStats.dwAddrMaskReps = icmp_stat.icps_outhist[ICMP_MASKREPLY]; return NO_ERROR; #else FILE *fp; if (!stats) return ERROR_INVALID_PARAMETER; memset(stats, 0, sizeof(MIB_ICMP)); /* get most of these stats from /proc/net/snmp, no error if can't */ fp = fopen("/proc/net/snmp", "r"); if (fp) { static const char hdr[] = "Icmp:"; char buf[512] = { 0 }, *ptr; do { ptr = fgets(buf, sizeof(buf), fp); } while (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1)); if (ptr) { /* last line was a header, get another */ ptr = fgets(buf, sizeof(buf), fp); if (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1) == 0) { char *endPtr; ptr += sizeof(hdr); if (ptr && *ptr) { stats->stats.icmpInStats.dwMsgs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwDestUnreachs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwTimeExcds = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwParmProbs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwSrcQuenchs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwRedirects = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwEchoReps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwTimestamps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwTimestampReps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwAddrMasks = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpInStats.dwAddrMaskReps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwMsgs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwDestUnreachs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwTimeExcds = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwParmProbs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwSrcQuenchs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwRedirects = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwEchoReps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwTimestamps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwTimestampReps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwAddrMasks = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->stats.icmpOutStats.dwAddrMaskReps = strtoul(ptr, &endPtr, 10); ptr = endPtr; } } } fclose(fp); } else { ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; } return NO_ERROR; #endif } DWORD getIPStats(PMIB_IPSTATS stats) { #if defined(HAVE_SYS_SYSCTL_H) && defined(IPCTL_STATS) int mib[] = {CTL_NET, PF_INET, IPPROTO_IP, IPCTL_STATS}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) int ip_ttl, ip_forwarding; struct ipstat ip_stat; size_t needed; if (!stats) return ERROR_INVALID_PARAMETER; needed = sizeof(ip_stat); if(sysctl(mib, MIB_LEN, &ip_stat, &needed, NULL, 0) == -1) { ERR ("failed to get ipstat\n"); return ERROR_NOT_SUPPORTED; } needed = sizeof(ip_ttl); if (sysctlbyname ("net.inet.ip.ttl", &ip_ttl, &needed, NULL, 0) == -1) { ERR ("failed to get ip Default TTL\n"); return ERROR_NOT_SUPPORTED; } needed = sizeof(ip_forwarding); if (sysctlbyname ("net.inet.ip.forwarding", &ip_forwarding, &needed, NULL, 0) == -1) { ERR ("failed to get ip forwarding\n"); return ERROR_NOT_SUPPORTED; } stats->dwForwarding = ip_forwarding; stats->dwDefaultTTL = ip_ttl; stats->dwInDelivers = ip_stat.ips_delivered; stats->dwInHdrErrors = ip_stat.ips_badhlen + ip_stat.ips_badsum + ip_stat.ips_tooshort + ip_stat.ips_badlen; stats->dwInAddrErrors = ip_stat.ips_cantforward; stats->dwInReceives = ip_stat.ips_total; stats->dwForwDatagrams = ip_stat.ips_forward; stats->dwInUnknownProtos = ip_stat.ips_noproto; stats->dwInDiscards = ip_stat.ips_fragdropped; stats->dwOutDiscards = ip_stat.ips_odropped; stats->dwReasmOks = ip_stat.ips_reassembled; stats->dwFragOks = ip_stat.ips_fragmented; stats->dwFragFails = ip_stat.ips_cantfrag; stats->dwReasmTimeout = ip_stat.ips_fragtimeout; stats->dwOutNoRoutes = ip_stat.ips_noroute; stats->dwOutRequests = ip_stat.ips_localout; stats->dwReasmReqds = ip_stat.ips_fragments; return NO_ERROR; #else FILE *fp; if (!stats) return ERROR_INVALID_PARAMETER; memset(stats, 0, sizeof(MIB_IPSTATS)); stats->dwNumIf = stats->dwNumAddr = getNumInterfaces(); stats->dwNumRoutes = getNumRoutes(); /* get most of these stats from /proc/net/snmp, no error if can't */ fp = fopen("/proc/net/snmp", "r"); if (fp) { static const char hdr[] = "Ip:"; char buf[512] = { 0 }, *ptr; do { ptr = fgets(buf, sizeof(buf), fp); } while (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1)); if (ptr) { /* last line was a header, get another */ ptr = fgets(buf, sizeof(buf), fp); if (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1) == 0) { char *endPtr; ptr += sizeof(hdr); if (ptr && *ptr) { stats->dwForwarding = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwDefaultTTL = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInReceives = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInHdrErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInAddrErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwForwDatagrams = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInUnknownProtos = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInDiscards = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInDelivers = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwOutRequests = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwOutDiscards = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwOutNoRoutes = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwReasmTimeout = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwReasmReqds = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwReasmOks = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwReasmFails = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwFragOks = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwFragFails = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwFragCreates = strtoul(ptr, &endPtr, 10); ptr = endPtr; } /* hmm, no routingDiscards */ } } fclose(fp); } else { ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; } return NO_ERROR; #endif } DWORD getTCPStats(MIB_TCPSTATS *stats) { #if defined(HAVE_SYS_SYSCTL_H) && defined(UDPCTL_STATS) #ifndef TCPTV_MIN /* got removed in Mac OS X for some reason */ #define TCPTV_MIN 2 #define TCPTV_REXMTMAX 128 #endif int mib[] = {CTL_NET, PF_INET, IPPROTO_TCP, TCPCTL_STATS}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) #define hz 1000 struct tcpstat tcp_stat; size_t needed; if (!stats) return ERROR_INVALID_PARAMETER; needed = sizeof(tcp_stat); if(sysctl(mib, MIB_LEN, &tcp_stat, &needed, NULL, 0) == -1) { ERR ("failed to get tcpstat\n"); return ERROR_NOT_SUPPORTED; } stats->dwRtoAlgorithm = MIB_TCP_RTO_VANJ; stats->dwRtoMin = TCPTV_MIN; stats->dwRtoMax = TCPTV_REXMTMAX; stats->dwMaxConn = -1; stats->dwActiveOpens = tcp_stat.tcps_connattempt; stats->dwPassiveOpens = tcp_stat.tcps_accepts; stats->dwAttemptFails = tcp_stat.tcps_conndrops; stats->dwEstabResets = tcp_stat.tcps_drops; stats->dwCurrEstab = 0; stats->dwInSegs = tcp_stat.tcps_rcvtotal; stats->dwOutSegs = tcp_stat.tcps_sndtotal - tcp_stat.tcps_sndrexmitpack; stats->dwRetransSegs = tcp_stat.tcps_sndrexmitpack; stats->dwInErrs = tcp_stat.tcps_rcvbadsum + tcp_stat.tcps_rcvbadoff + tcp_stat.tcps_rcvmemdrop + tcp_stat.tcps_rcvshort; stats->dwOutRsts = tcp_stat.tcps_sndctrl - tcp_stat.tcps_closed; stats->dwNumConns = tcp_stat.tcps_connects; return NO_ERROR; #else FILE *fp; if (!stats) return ERROR_INVALID_PARAMETER; memset(stats, 0, sizeof(MIB_TCPSTATS)); /* get from /proc/net/snmp, no error if can't */ fp = fopen("/proc/net/snmp", "r"); if (fp) { static const char hdr[] = "Tcp:"; char buf[512] = { 0 }, *ptr; do { ptr = fgets(buf, sizeof(buf), fp); } while (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1)); if (ptr) { /* last line was a header, get another */ ptr = fgets(buf, sizeof(buf), fp); if (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1) == 0) { char *endPtr; ptr += sizeof(hdr); if (ptr && *ptr) { stats->dwRtoAlgorithm = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwRtoMin = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwRtoMax = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwMaxConn = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwActiveOpens = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwPassiveOpens = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwAttemptFails = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwEstabResets = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwCurrEstab = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInSegs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwOutSegs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwRetransSegs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInErrs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwOutRsts = strtoul(ptr, &endPtr, 10); ptr = endPtr; } stats->dwNumConns = getNumTcpEntries(); } } fclose(fp); } else { ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; } return NO_ERROR; #endif } DWORD getUDPStats(MIB_UDPSTATS *stats) { #if defined(HAVE_SYS_SYSCTL_H) && defined(UDPCTL_STATS) int mib[] = {CTL_NET, PF_INET, IPPROTO_UDP, UDPCTL_STATS}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) struct udpstat udp_stat; size_t needed; if (!stats) return ERROR_INVALID_PARAMETER; needed = sizeof(udp_stat); if(sysctl(mib, MIB_LEN, &udp_stat, &needed, NULL, 0) == -1) { ERR ("failed to get udpstat\n"); return ERROR_NOT_SUPPORTED; } stats->dwInDatagrams = udp_stat.udps_ipackets; stats->dwOutDatagrams = udp_stat.udps_opackets; stats->dwNoPorts = udp_stat.udps_noport; stats->dwInErrors = udp_stat.udps_hdrops + udp_stat.udps_badsum + udp_stat.udps_fullsock + udp_stat.udps_badlen; stats->dwNumAddrs = getNumUdpEntries(); return NO_ERROR; #else FILE *fp; if (!stats) return ERROR_INVALID_PARAMETER; memset(stats, 0, sizeof(MIB_UDPSTATS)); /* get from /proc/net/snmp, no error if can't */ fp = fopen("/proc/net/snmp", "r"); if (fp) { static const char hdr[] = "Udp:"; char buf[512] = { 0 }, *ptr; do { ptr = fgets(buf, sizeof(buf), fp); } while (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1)); if (ptr) { /* last line was a header, get another */ ptr = fgets(buf, sizeof(buf), fp); if (ptr && strncasecmp(buf, hdr, sizeof(hdr) - 1) == 0) { char *endPtr; ptr += sizeof(hdr); if (ptr && *ptr) { stats->dwInDatagrams = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwNoPorts = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwInErrors = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwOutDatagrams = strtoul(ptr, &endPtr, 10); ptr = endPtr; } if (ptr && *ptr) { stats->dwNumAddrs = strtoul(ptr, &endPtr, 10); ptr = endPtr; } } } fclose(fp); } else { ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; } return NO_ERROR; #endif } static DWORD getNumWithOneHeader(const char *filename) { #if defined(HAVE_SYS_SYSCTL_H) && defined(HAVE_NETINET_IN_PCB_H) size_t Len = 0; char *Buf; struct xinpgen *pXIG, *pOrigXIG; int Protocol; DWORD NumEntries = 0; if (!strcmp (filename, "net.inet.tcp.pcblist")) Protocol = IPPROTO_TCP; else if (!strcmp (filename, "net.inet.udp.pcblist")) Protocol = IPPROTO_UDP; else { ERR ("Unsupported mib '%s', needs protocol mapping\n", filename); return 0; } if (sysctlbyname (filename, NULL, &Len, NULL, 0) < 0) { WARN ("Unable to read '%s' via sysctlbyname\n", filename); return 0; } Buf = HeapAlloc (GetProcessHeap (), 0, Len); if (!Buf) { ERR ("Out of memory!\n"); return 0; } if (sysctlbyname (filename, Buf, &Len, NULL, 0) < 0) { ERR ("Failure to read '%s' via sysctlbyname!\n", filename); HeapFree (GetProcessHeap (), 0, Buf); return 0; } /* Might be nothing here; first entry is just a header it seems */ if (Len <= sizeof (struct xinpgen)) { HeapFree (GetProcessHeap (), 0, Buf); return 0; } pOrigXIG = (struct xinpgen *)Buf; pXIG = pOrigXIG; for (pXIG = (struct xinpgen *)((char *)pXIG + pXIG->xig_len); pXIG->xig_len > sizeof (struct xinpgen); pXIG = (struct xinpgen *)((char *)pXIG + pXIG->xig_len)) { struct tcpcb *pTCPData = NULL; struct inpcb *pINData; struct xsocket *pSockData; if (Protocol == IPPROTO_TCP) { pTCPData = &((struct xtcpcb *)pXIG)->xt_tp; pINData = &((struct xtcpcb *)pXIG)->xt_inp; pSockData = &((struct xtcpcb *)pXIG)->xt_socket; } else { pINData = &((struct xinpcb *)pXIG)->xi_inp; pSockData = &((struct xinpcb *)pXIG)->xi_socket; } /* Ignore sockets for other protocols */ if (pSockData->xso_protocol != Protocol) continue; /* Ignore PCBs that were freed while generating the data */ if (pINData->inp_gencnt > pOrigXIG->xig_gen) continue; /* we're only interested in IPv4 addresses */ if (!(pINData->inp_vflag & INP_IPV4) || (pINData->inp_vflag & INP_IPV6)) continue; /* If all 0's, skip it */ if (!pINData->inp_laddr.s_addr && !pINData->inp_lport && !pINData->inp_faddr.s_addr && !pINData->inp_fport) continue; NumEntries++; } HeapFree (GetProcessHeap (), 0, Buf); return NumEntries; #else FILE *fp; int ret = 0; fp = fopen(filename, "r"); if (fp) { char buf[512] = { 0 }, *ptr; ptr = fgets(buf, sizeof(buf), fp); if (ptr) { do { ptr = fgets(buf, sizeof(buf), fp); if (ptr) ret++; } while (ptr); } fclose(fp); } else ERR ("Unable to open '%s' to count entries!\n", filename); return ret; #endif } DWORD getNumRoutes(void) { #if defined(HAVE_SYS_SYSCTL_H) && defined(NET_RT_DUMP) int mib[6] = {CTL_NET, PF_ROUTE, 0, PF_INET, NET_RT_DUMP, 0}; size_t needed; char *buf, *lim, *next; struct rt_msghdr *rtm; DWORD RouteCount = 0; if (sysctl (mib, 6, NULL, &needed, NULL, 0) < 0) { ERR ("sysctl 1 failed!\n"); return 0; } buf = HeapAlloc (GetProcessHeap (), 0, needed); if (!buf) return 0; if (sysctl (mib, 6, buf, &needed, NULL, 0) < 0) { ERR ("sysctl 2 failed!\n"); HeapFree (GetProcessHeap (), 0, buf); return 0; } lim = buf + needed; for (next = buf; next < lim; next += rtm->rtm_msglen) { rtm = (struct rt_msghdr *)next; if (rtm->rtm_type != RTM_GET) { WARN ("Got unexpected message type 0x%x!\n", rtm->rtm_type); continue; } /* Ignore all entries except for gateway routes which aren't multicast */ if (!(rtm->rtm_flags & RTF_GATEWAY) || (rtm->rtm_flags & RTF_MULTICAST)) continue; RouteCount++; } HeapFree (GetProcessHeap (), 0, buf); return RouteCount; #else return getNumWithOneHeader("/proc/net/route"); #endif } DWORD getRouteTable(PMIB_IPFORWARDTABLE *ppIpForwardTable, HANDLE heap, DWORD flags) { DWORD ret; if (!ppIpForwardTable) ret = ERROR_INVALID_PARAMETER; else { DWORD numRoutes = getNumRoutes(); DWORD size = sizeof(MIB_IPFORWARDTABLE); PMIB_IPFORWARDTABLE table; if (numRoutes > 1) size += (numRoutes - 1) * sizeof(MIB_IPFORWARDROW); table = HeapAlloc(heap, flags, size); if (table) { #if defined(HAVE_SYS_SYSCTL_H) && defined(NET_RT_DUMP) int mib[6] = {CTL_NET, PF_ROUTE, 0, PF_INET, NET_RT_DUMP, 0}; size_t needed; char *buf, *lim, *next, *addrPtr; struct rt_msghdr *rtm; if (sysctl (mib, 6, NULL, &needed, NULL, 0) < 0) { ERR ("sysctl 1 failed!\n"); HeapFree (GetProcessHeap (), 0, table); return NO_ERROR; } buf = HeapAlloc (GetProcessHeap (), 0, needed); if (!buf) { HeapFree (GetProcessHeap (), 0, table); return ERROR_OUTOFMEMORY; } if (sysctl (mib, 6, buf, &needed, NULL, 0) < 0) { ERR ("sysctl 2 failed!\n"); HeapFree (GetProcessHeap (), 0, table); HeapFree (GetProcessHeap (), 0, buf); return NO_ERROR; } *ppIpForwardTable = table; table->dwNumEntries = 0; lim = buf + needed; for (next = buf; next < lim; next += rtm->rtm_msglen) { int i; rtm = (struct rt_msghdr *)next; if (rtm->rtm_type != RTM_GET) { WARN ("Got unexpected message type 0x%x!\n", rtm->rtm_type); continue; } /* Ignore all entries except for gateway routes which aren't multicast */ if (!(rtm->rtm_flags & RTF_GATEWAY) || (rtm->rtm_flags & RTF_MULTICAST)) continue; memset (&table->table[table->dwNumEntries], 0, sizeof (MIB_IPFORWARDROW)); table->table[table->dwNumEntries].dwForwardIfIndex = rtm->rtm_index; table->table[table->dwNumEntries].dwForwardType = MIB_IPROUTE_TYPE_INDIRECT; table->table[table->dwNumEntries].dwForwardMetric1 = rtm->rtm_rmx.rmx_hopcount; table->table[table->dwNumEntries].dwForwardProto = MIB_IPPROTO_LOCAL; addrPtr = (char *)(rtm + 1); for (i = 1; i; i <<= 1) { struct sockaddr *sa; DWORD addr; if (!(i & rtm->rtm_addrs)) continue; sa = (struct sockaddr *)addrPtr; ADVANCE (addrPtr, sa); /* default routes are encoded by length-zero sockaddr */ if (sa->sa_len == 0) addr = 0; else if (sa->sa_family != AF_INET) { WARN ("Received unsupported sockaddr family 0x%x\n", sa->sa_family); addr = 0; } else { struct sockaddr_in *sin = (struct sockaddr_in *)sa; addr = sin->sin_addr.s_addr; } switch (i) { case RTA_DST: table->table[table->dwNumEntries].dwForwardDest = addr; break; case RTA_GATEWAY: table->table[table->dwNumEntries].dwForwardNextHop = addr; break; case RTA_NETMASK: table->table[table->dwNumEntries].dwForwardMask = addr; break; default: WARN ("Unexpected address type 0x%x\n", i); } } table->dwNumEntries++; } HeapFree (GetProcessHeap (), 0, buf); ret = NO_ERROR; #else FILE *fp; ret = NO_ERROR; *ppIpForwardTable = table; table->dwNumEntries = 0; /* get from /proc/net/route, no error if can't */ fp = fopen("/proc/net/route", "r"); if (fp) { char buf[512] = { 0 }, *ptr; /* skip header line */ ptr = fgets(buf, sizeof(buf), fp); while (ptr && table->dwNumEntries < numRoutes) { memset(&table->table[table->dwNumEntries], 0, sizeof(MIB_IPFORWARDROW)); ptr = fgets(buf, sizeof(buf), fp); if (ptr) { DWORD index; while (!isspace(*ptr)) ptr++; *ptr = '\0'; ptr++; if (getInterfaceIndexByName(buf, &index) == NO_ERROR) { char *endPtr; table->table[table->dwNumEntries].dwForwardIfIndex = index; if (*ptr) { table->table[table->dwNumEntries].dwForwardDest = strtoul(ptr, &endPtr, 16); ptr = endPtr; } if (ptr && *ptr) { table->table[table->dwNumEntries].dwForwardNextHop = strtoul(ptr, &endPtr, 16); ptr = endPtr; } if (ptr && *ptr) { DWORD flags = strtoul(ptr, &endPtr, 16); if (!(flags & RTF_UP)) table->table[table->dwNumEntries].dwForwardType = MIB_IPROUTE_TYPE_INVALID; else if (flags & RTF_GATEWAY) table->table[table->dwNumEntries].dwForwardType = MIB_IPROUTE_TYPE_INDIRECT; else table->table[table->dwNumEntries].dwForwardType = MIB_IPROUTE_TYPE_DIRECT; ptr = endPtr; } if (ptr && *ptr) { strtoul(ptr, &endPtr, 16); /* refcount, skip */ ptr = endPtr; } if (ptr && *ptr) { strtoul(ptr, &endPtr, 16); /* use, skip */ ptr = endPtr; } if (ptr && *ptr) { table->table[table->dwNumEntries].dwForwardMetric1 = strtoul(ptr, &endPtr, 16); ptr = endPtr; } if (ptr && *ptr) { table->table[table->dwNumEntries].dwForwardMask = strtoul(ptr, &endPtr, 16); ptr = endPtr; } /* FIXME: other protos might be appropriate, e.g. the default * route is typically set with MIB_IPPROTO_NETMGMT instead */ table->table[table->dwNumEntries].dwForwardProto = MIB_IPPROTO_LOCAL; table->dwNumEntries++; } } } fclose(fp); } else { ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; } #endif } else ret = ERROR_OUTOFMEMORY; } return ret; } DWORD getNumArpEntries(void) { #if defined(HAVE_SYS_SYSCTL_H) && defined(NET_RT_DUMP) int mib[] = {CTL_NET, PF_ROUTE, 0, AF_INET, NET_RT_FLAGS, RTF_LLINFO}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) DWORD arpEntries = 0; size_t needed; char *buf, *lim, *next; struct rt_msghdr *rtm; struct sockaddr_inarp *sinarp; struct sockaddr_dl *sdl; if (sysctl (mib, MIB_LEN, NULL, &needed, NULL, 0) == -1) { ERR ("failed to get size of arp table\n"); return 0; } buf = HeapAlloc (GetProcessHeap (), 0, needed); if (!buf) return 0; if (sysctl (mib, MIB_LEN, buf, &needed, NULL, 0) == -1) { ERR ("failed to get arp table\n"); HeapFree (GetProcessHeap (), 0, buf); return 0; } lim = buf + needed; next = buf; while(next < lim) { rtm = (struct rt_msghdr *)next; sinarp=(struct sockaddr_inarp *)(rtm + 1); sdl = (struct sockaddr_dl *)((char *)sinarp + ROUNDUP(sinarp->sin_len)); if(sdl->sdl_alen) /* arp entry */ arpEntries++; next += rtm->rtm_msglen; } HeapFree (GetProcessHeap (), 0, buf); return arpEntries; #endif return getNumWithOneHeader("/proc/net/arp"); } DWORD getArpTable(PMIB_IPNETTABLE *ppIpNetTable, HANDLE heap, DWORD flags) { DWORD ret = NO_ERROR; if (!ppIpNetTable) ret = ERROR_INVALID_PARAMETER; else { DWORD numEntries = getNumArpEntries(); DWORD size = sizeof(MIB_IPNETTABLE); PMIB_IPNETTABLE table; if (numEntries > 1) size += (numEntries - 1) * sizeof(MIB_IPNETROW); table = HeapAlloc(heap, flags, size); #if defined(HAVE_SYS_SYSCTL_H) && defined(NET_RT_DUMP) if (table) { int mib[] = {CTL_NET, PF_ROUTE, 0, AF_INET, NET_RT_FLAGS, RTF_LLINFO}; #define MIB_LEN (sizeof(mib) / sizeof(mib[0])) size_t needed; char *buf, *lim, *next; struct rt_msghdr *rtm; struct sockaddr_inarp *sinarp; struct sockaddr_dl *sdl; *ppIpNetTable = table; table->dwNumEntries = 0; if (sysctl (mib, MIB_LEN, NULL, &needed, NULL, 0) == -1) { ERR ("failed to get size of arp table\n"); return ERROR_NOT_SUPPORTED; } buf = HeapAlloc (GetProcessHeap (), 0, needed); if (!buf) return ERROR_OUTOFMEMORY; if (sysctl (mib, MIB_LEN, buf, &needed, NULL, 0) == -1) { ERR ("failed to get arp table\n"); HeapFree (GetProcessHeap (), 0, buf); return ERROR_NOT_SUPPORTED; } lim = buf + needed; next = buf; while(next < lim) { rtm = (struct rt_msghdr *)next; sinarp=(struct sockaddr_inarp *)(rtm + 1); sdl = (struct sockaddr_dl *)((char *)sinarp + ROUNDUP(sinarp->sin_len)); if(sdl->sdl_alen) /* arp entry */ { DWORD byte = strtoul(&sdl->sdl_data[sdl->sdl_alen], NULL, 16); memset(&table->table[table->dwNumEntries], 0, sizeof(MIB_IPNETROW)); table->table[table->dwNumEntries].dwAddr = sinarp->sin_addr.s_addr; table->table[table->dwNumEntries].dwIndex = sdl->sdl_index; table->table[table->dwNumEntries].dwPhysAddrLen = sdl->sdl_alen; table->table[table->dwNumEntries].bPhysAddr[ table->table[table->dwNumEntries].dwPhysAddrLen++] = byte & 0x0ff; if(rtm->rtm_rmx.rmx_expire == 0) table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_STATIC; else if(sinarp->sin_other & SIN_PROXY) table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_OTHER; else if(rtm->rtm_rmx.rmx_expire != 0) table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_DYNAMIC; else table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_INVALID; table->dwNumEntries++; } next += rtm->rtm_msglen; } HeapFree (GetProcessHeap (), 0, buf); } else ret = ERROR_OUTOFMEMORY; return ret; #endif if (table) { FILE *fp; *ppIpNetTable = table; table->dwNumEntries = 0; /* get from /proc/net/arp, no error if can't */ fp = fopen("/proc/net/arp", "r"); if (fp) { char buf[512] = { 0 }, *ptr; /* skip header line */ ptr = fgets(buf, sizeof(buf), fp); while (ptr && table->dwNumEntries < numEntries) { ptr = fgets(buf, sizeof(buf), fp); if (ptr) { char *endPtr; memset(&table->table[table->dwNumEntries], 0, sizeof(MIB_IPNETROW)); table->table[table->dwNumEntries].dwAddr = inet_addr(ptr); while (ptr && *ptr && !isspace(*ptr)) ptr++; if (ptr && *ptr) { strtoul(ptr, &endPtr, 16); /* hw type (skip) */ ptr = endPtr; } if (ptr && *ptr) { DWORD flags = strtoul(ptr, &endPtr, 16); #ifdef ATF_COM if (flags & ATF_COM) table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_DYNAMIC; else #endif #ifdef ATF_PERM if (flags & ATF_PERM) table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_STATIC; else #endif table->table[table->dwNumEntries].dwType = MIB_IPNET_TYPE_OTHER; ptr = endPtr; } while (ptr && *ptr && isspace(*ptr)) ptr++; while (ptr && *ptr && !isspace(*ptr)) { DWORD byte = strtoul(ptr, &endPtr, 16); if (endPtr && *endPtr) { endPtr++; table->table[table->dwNumEntries].bPhysAddr[ table->table[table->dwNumEntries].dwPhysAddrLen++] = byte & 0x0ff; } ptr = endPtr; } if (ptr && *ptr) { strtoul(ptr, &endPtr, 16); /* mask (skip) */ ptr = endPtr; } getInterfaceIndexByName(ptr, &table->table[table->dwNumEntries].dwIndex); table->dwNumEntries++; } } fclose(fp); } else ret = ERROR_NOT_SUPPORTED; } else ret = ERROR_OUTOFMEMORY; } return ret; } DWORD getNumUdpEntries(void) { #if defined(HAVE_SYS_SYSCTL_H) && defined(HAVE_NETINET_IN_PCB_H) return getNumWithOneHeader ("net.inet.udp.pcblist"); #else return getNumWithOneHeader("/proc/net/udp"); #endif } DWORD getUdpTable(PMIB_UDPTABLE *ppUdpTable, HANDLE heap, DWORD flags) { DWORD ret; #if defined(HAVE_SYS_SYSCTL_H) && defined(NET_RT_DUMP) ERR ("unimplemented!\n"); return ERROR_NOT_SUPPORTED; #endif if (!ppUdpTable) ret = ERROR_INVALID_PARAMETER; else { DWORD numEntries = getNumUdpEntries(); DWORD size = sizeof(MIB_UDPTABLE); PMIB_UDPTABLE table; if (numEntries > 1) size += (numEntries - 1) * sizeof(MIB_UDPROW); table = HeapAlloc(heap, flags, size); if (table) { FILE *fp; ret = NO_ERROR; *ppUdpTable = table; table->dwNumEntries = 0; /* get from /proc/net/udp, no error if can't */ fp = fopen("/proc/net/udp", "r"); if (fp) { char buf[512] = { 0 }, *ptr; /* skip header line */ ptr = fgets(buf, sizeof(buf), fp); while (ptr && table->dwNumEntries < numEntries) { memset(&table->table[table->dwNumEntries], 0, sizeof(MIB_UDPROW)); ptr = fgets(buf, sizeof(buf), fp); if (ptr) { char *endPtr; if (ptr && *ptr) { strtoul(ptr, &endPtr, 16); /* skip */ ptr = endPtr; } if (ptr && *ptr) { ptr++; table->table[table->dwNumEntries].dwLocalAddr = strtoul(ptr, &endPtr, 16); ptr = endPtr; } if (ptr && *ptr) { ptr++; table->table[table->dwNumEntries].dwLocalPort = strtoul(ptr, &endPtr, 16); ptr = endPtr; } table->dwNumEntries++; } } fclose(fp); } else ret = ERROR_NOT_SUPPORTED; } else ret = ERROR_OUTOFMEMORY; } return ret; } DWORD getNumTcpEntries(void) { #if defined(HAVE_SYS_SYSCTL_H) && defined(HAVE_NETINET_IN_PCB_H) return getNumWithOneHeader ("net.inet.tcp.pcblist"); #else return getNumWithOneHeader ("/proc/net/tcp"); #endif } /* Why not a lookup table? Because the TCPS_* constants are different on different platforms */ static DWORD TCPStateToMIBState (int state) { switch (state) { case TCPS_ESTABLISHED: return MIB_TCP_STATE_ESTAB; case TCPS_SYN_SENT: return MIB_TCP_STATE_SYN_SENT; case TCPS_SYN_RECEIVED: return MIB_TCP_STATE_SYN_RCVD; case TCPS_FIN_WAIT_1: return MIB_TCP_STATE_FIN_WAIT1; case TCPS_FIN_WAIT_2: return MIB_TCP_STATE_FIN_WAIT2; case TCPS_TIME_WAIT: return MIB_TCP_STATE_TIME_WAIT; case TCPS_CLOSE_WAIT: return MIB_TCP_STATE_CLOSE_WAIT; case TCPS_LAST_ACK: return MIB_TCP_STATE_LAST_ACK; case TCPS_LISTEN: return MIB_TCP_STATE_LISTEN; case TCPS_CLOSING: return MIB_TCP_STATE_CLOSING; default: case TCPS_CLOSED: return MIB_TCP_STATE_CLOSED; } } DWORD getTcpTable(PMIB_TCPTABLE *ppTcpTable, DWORD maxEntries, HANDLE heap, DWORD flags) { DWORD numEntries; PMIB_TCPTABLE table; #if defined(HAVE_SYS_SYSCTL_H) && defined(HAVE_NETINET_IN_PCB_H) size_t Len = 0; char *Buf; struct xinpgen *pXIG, *pOrigXIG; #else FILE *fp; char buf[512] = { 0 }, *ptr; #endif if (!ppTcpTable) return ERROR_INVALID_PARAMETER; numEntries = getNumTcpEntries (); if (!*ppTcpTable) { DWORD size = sizeof(MIB_TCPTABLE); if (numEntries > 1) size += (numEntries - 1) * sizeof (MIB_TCPROW); *ppTcpTable = HeapAlloc (heap, flags, size); if (!*ppTcpTable) { ERR ("Out of memory!\n"); return ERROR_OUTOFMEMORY; } maxEntries = numEntries; } table = *ppTcpTable; table->dwNumEntries = 0; if (!numEntries) return NO_ERROR; #if defined(HAVE_SYS_SYSCTL_H) && defined(HAVE_NETINET_IN_PCB_H) if (sysctlbyname ("net.inet.tcp.pcblist", NULL, &Len, NULL, 0) < 0) { ERR ("Failure to read net.inet.tcp.pcblist via sysctlbyname!\n"); return ERROR_OUTOFMEMORY; } Buf = HeapAlloc (GetProcessHeap (), 0, Len); if (!Buf) { ERR ("Out of memory!\n"); return ERROR_OUTOFMEMORY; } if (sysctlbyname ("net.inet.tcp.pcblist", Buf, &Len, NULL, 0) < 0) { ERR ("Failure to read net.inet.tcp.pcblist via sysctlbyname!\n"); HeapFree (GetProcessHeap (), 0, Buf); return ERROR_OUTOFMEMORY; } /* Might be nothing here; first entry is just a header it seems */ if (Len <= sizeof (struct xinpgen)) { HeapFree (GetProcessHeap (), 0, Buf); return NO_ERROR; } pOrigXIG = (struct xinpgen *)Buf; pXIG = pOrigXIG; for (pXIG = (struct xinpgen *)((char *)pXIG + pXIG->xig_len); (pXIG->xig_len > sizeof (struct xinpgen)) && (table->dwNumEntries < maxEntries); pXIG = (struct xinpgen *)((char *)pXIG + pXIG->xig_len)) { struct tcpcb *pTCPData = NULL; struct inpcb *pINData; struct xsocket *pSockData; pTCPData = &((struct xtcpcb *)pXIG)->xt_tp; pINData = &((struct xtcpcb *)pXIG)->xt_inp; pSockData = &((struct xtcpcb *)pXIG)->xt_socket; /* Ignore sockets for other protocols */ if (pSockData->xso_protocol != IPPROTO_TCP) continue; /* Ignore PCBs that were freed while generating the data */ if (pINData->inp_gencnt > pOrigXIG->xig_gen) continue; /* we're only interested in IPv4 addresses */ if (!(pINData->inp_vflag & INP_IPV4) || (pINData->inp_vflag & INP_IPV6)) continue; /* If all 0's, skip it */ if (!pINData->inp_laddr.s_addr && !pINData->inp_lport && !pINData->inp_faddr.s_addr && !pINData->inp_fport) continue; /* Fill in structure details */ table->table[table->dwNumEntries].dwLocalAddr = pINData->inp_laddr.s_addr; table->table[table->dwNumEntries].dwLocalPort = pINData->inp_lport; table->table[table->dwNumEntries].dwRemoteAddr = pINData->inp_faddr.s_addr; table->table[table->dwNumEntries].dwRemotePort = pINData->inp_fport; table->table[table->dwNumEntries].dwState = TCPStateToMIBState (pTCPData->t_state); table->dwNumEntries++; } HeapFree (GetProcessHeap (), 0, Buf); #else /* get from /proc/net/tcp, no error if can't */ fp = fopen("/proc/net/tcp", "r"); if (!fp) return ERROR_NOT_SUPPORTED; /* skip header line */ ptr = fgets(buf, sizeof(buf), fp); while (ptr && table->dwNumEntries < maxEntries) { memset(&table->table[table->dwNumEntries], 0, sizeof(MIB_TCPROW)); ptr = fgets(buf, sizeof(buf), fp); if (ptr) { char *endPtr; while (ptr && *ptr && *ptr != ':') ptr++; if (ptr && *ptr) ptr++; if (ptr && *ptr) { table->table[table->dwNumEntries].dwLocalAddr = strtoul(ptr, &endPtr, 16); ptr = endPtr; } if (ptr && *ptr) { ptr++; table->table[table->dwNumEntries].dwLocalPort = htons ((unsigned short)strtoul(ptr, &endPtr, 16)); ptr = endPtr; } if (ptr && *ptr) { table->table[table->dwNumEntries].dwRemoteAddr = strtoul(ptr, &endPtr, 16); ptr = endPtr; } if (ptr && *ptr) { ptr++; table->table[table->dwNumEntries].dwRemotePort = htons ((unsigned short)strtoul(ptr, &endPtr, 16)); ptr = endPtr; } if (ptr && *ptr) { DWORD state = strtoul(ptr, &endPtr, 16); table->table[table->dwNumEntries].dwState = TCPStateToMIBState (state); ptr = endPtr; } table->dwNumEntries++; } } fclose(fp); #endif return NO_ERROR; }