summaryrefslogtreecommitdiff
path: root/net/pload/files/patch-ioctl_stats.c
blob: 83962721b2c9463c4f729808c3e2917fc312ba50 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
--- ioctl_stat.c.orig	2000-02-01 02:11:24.000000000 -0500
+++ ioctl_stat.c	2009-06-15 15:30:03.000000000 -0400
@@ -30,6 +30,12 @@
 #include <fcntl.h>		/* open */
 #include <sys/ioctl.h>		/* ioctl */
 #include <errno.h>
+#ifdef __FreeBSD__
+#include <sys/sysctl.h>
+#include <sys/socket.h>
+#include <net/if.h>
+#include <net/if_mib.h>
+#endif
 
 #ifndef STREAMS				/* Linux, FreeBSD, NetBSD, Ultrix */
 #	include <sys/socket.h>		/* socket */
@@ -39,8 +45,10 @@
 #		include <linux/if_ppp.h>
 #	else				/* most everything else */
 #		include <net/if.h>
-#		include <net/ppp_defs.h>
-#		include <net/if_ppp.h>
+#		if !defined(__FreeBSD__)
+#			include <net/ppp_defs.h>
+#			include <net/if_ppp.h>
+#		endif /* ! __FreeBSD__ */
 #	endif	/* linux && __GLIBC__ < 2 */
 #else	/* STREAMS */			/* Solaris, SunOS, OSF/1, SVR4 */
 #	include <net/ppp_defs.h>
@@ -61,8 +69,18 @@
 void ioctl_stat(if_data *ifd)
 {
 	struct ifreq ifr;
+#ifdef __FreeBSD__
+	static int		if_ix = -1;
+	struct ifmibdata	ifmd;
+	size_t			ifmd_sz = sizeof(ifmd);
+	int			nr_ifs;
+	size_t			nr_ifs_sz = sizeof(nr_ifs);
+	int			name[6];
+	int			i;
+#else
 	struct ifpppstatsreq req;
-	
+#endif
+
 	if (!ifd->s) getsocket(ifd);
 	
 	memset(&ifr, 0, sizeof(ifr));
@@ -76,7 +94,9 @@
 		return;
 	}
 	
+#ifndef __FreeBSD__
 	memset(&req, 0, sizeof(req));
+#endif
 
 #ifdef linux
 	req.stats_ptr = (caddr_t) &req.stats;
@@ -84,18 +104,36 @@
 #define ifr_name ifr__name
 #endif	
 
-	strncpy(req.ifr_name, ifd->device, sizeof(req.ifr_name));
-	if (ioctl(ifd->s, SIOCGPPPSTATS, &req) != 0)
-	{
-		/* shouldn't fail if SIOCGIFFLAGS worked... */
-		ifd->in_bytes = 0UL;
-		ifd->out_bytes = 0UL;
-		return;
+#ifdef __FreeBSD__
+	name[0] = CTL_NET;
+	name[1] = PF_LINK;
+	name[2] = NETLINK_GENERIC;
+	name[3] = IFMIB_IFDATA;
+	name[5] = IFDATA_GENERAL;
+
+	if (if_ix < 0) {
+		if (sysctlbyname("net.link.generic.system.ifcount", (void *) &nr_ifs, &nr_ifs_sz, (void *) 0, 0) < 0) {
+			return;
+		}
+		for (i = 1; i <= nr_ifs; i++) {
+			name[4] = i; /* row of the ifmib table */
+
+			if (sysctl(name, 6, (void *) &ifmd, &ifmd_sz, (void *) 0, 0) < 0) {
+				continue;
+			}
+			if (strncmp(ifmd.ifmd_name, ifr.ifr_name, strlen(ifr.ifr_name)) == 0) {
+				if_ix = i;
+				break;
+			}
+		}
 	}
-	
-	ifd->in_bytes = (unsigned long)req.stats.p.ppp_ibytes;
-	ifd->out_bytes = (unsigned long)req.stats.p.ppp_obytes;
-	
+	name[4] = if_ix;
+	if (sysctl(name, 6, (void *) &ifmd, &ifmd_sz, (void *) 0, 0) >= 0) {
+		ifd->in_bytes = ifmd.ifmd_data.ifi_ibytes;
+		ifd->out_bytes = ifmd.ifmd_data.ifi_obytes;
+	}
+#endif
+
 	return;
 }