aboutsummaryrefslogtreecommitdiffstats
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;
 }