3 #include <sys/socket.h>
9 #include <linux/sockios.h>
13 #if defined(__FreeBSD_kernel__)
21 #if defined(__APPLE__)
22 #define s6_addr16 __u6_addr.__u6_addr16
27 #define s6_addr16 _S6_un._S6_u8
29 #include <sys/sockio.h>
32 #include <netinet/in.h>
73 unsigned long long in_packets
, in_bytes
, in_errors
, in_drops
;
74 unsigned long long in_fifo
, in_frame
, in_compress
, in_multicast
;
75 unsigned long long out_bytes
, out_packets
, out_errors
, out_drops
;
76 unsigned long long out_fifo
, out_colls
, out_carrier
, out_multicast
;
80 void print_quad_ipv4(in_addr_t i
) {
83 (i
& 0xff000000) >> 24,
84 (i
& 0x00ff0000) >> 16,
85 (i
& 0x0000ff00) >> 8,
89 void print_quad_ipv6(uint16_t *a
) {
90 printf("%04x:%04x:%04x:%04x:%04x:%04x:%04x:%04x",
91 a
[0], a
[1], a
[2], a
[3], a
[4], a
[5], a
[6], a
[7]);
94 void print_quad(struct sockaddr
*adr
) {
95 switch (adr
->sa_family
) {
97 print_quad_ipv4(((struct sockaddr_in
*)adr
)->sin_addr
.s_addr
);
100 print_quad_ipv6(((struct sockaddr_in6
*)adr
)->sin6_addr
.s6_addr16
);
108 enum print_error_enum
{
117 static int do_socket_ioctl(const char *ifname
, const unsigned long int request
,
118 struct ifreq
*req
, int *ioctl_errno
,
119 const enum print_error_enum print_error
) {
122 if ((sock
= socket(PF_INET
, SOCK_DGRAM
, IPPROTO_IP
)) == -1)
124 strncpy(req
->ifr_name
, ifname
, IFNAMSIZ
);
125 req
->ifr_name
[IFNAMSIZ
- 1] = 0;
127 if ((res
= ioctl(sock
, request
, req
)) == -1) {
129 *ioctl_errno
= errno
;
130 if (print_error
== PRINT_ERROR
)
131 fprintf(stderr
, "ioctl on %s: %s\n", ifname
, strerror(errno
));
141 int if_exists(const char *iface
) {
143 return !do_socket_ioctl(iface
, SIOCGIFFLAGS
, &r
, NULL
, PRINT_NO_ERROR
);
146 #if defined(__linux__)
148 void if_flags(const char *iface
) {
156 { IFF_BROADCAST
, "Broadcast" },
157 { IFF_DEBUG
, "Debugging" },
158 { IFF_LOOPBACK
, "Loopback" },
159 { IFF_POINTOPOINT
, "Ppp" },
160 { IFF_NOTRAILERS
, "No-trailers" },
161 { IFF_RUNNING
, "Running" },
162 { IFF_NOARP
, "No-arp" },
163 { IFF_PROMISC
, "Promiscuous" },
164 { IFF_ALLMULTI
, "All-multicast" },
165 { IFF_MASTER
, "Load-master" },
166 { IFF_SLAVE
, "Load-slave" },
167 { IFF_MULTICAST
, "Multicast" },
168 { IFF_PORTSEL
, "Port-select" },
169 { IFF_AUTOMEDIA
, "Auto-detect" },
170 { IFF_DYNAMIC
, "Dynaddr" },
171 { 0xffff0000, "Unknown-flags" },
174 if (do_socket_ioctl(iface
, SIOCGIFFLAGS
, &r
, NULL
, PRINT_ERROR
))
177 for (i
= 0; i
< sizeof(flags
) / sizeof(flags
[0]); i
++)
178 printf("%s%s%s", (r
.ifr_flags
& flags
[i
].flag
) ? "On " : "Off ",
180 sizeof(flags
) / sizeof(flags
[0]) - 1 == i
? "" : "\n");
183 void if_hwaddr(const char *iface
) {
185 unsigned char *hwaddr
;
187 if (do_socket_ioctl(iface
, SIOCGIFHWADDR
, &r
, NULL
, PRINT_ERROR
))
190 hwaddr
= (unsigned char *)r
.ifr_hwaddr
.sa_data
;
191 printf("%02X:%02X:%02X:%02X:%02X:%02X",
192 hwaddr
[0], hwaddr
[1], hwaddr
[2], hwaddr
[3], hwaddr
[4], hwaddr
[5]);
197 static struct sockaddr
*if_addr_value(const char *iface
, struct ifreq
*r
,
198 unsigned long int request
) {
201 if (do_socket_ioctl(iface
, request
, r
, &e
, PRINT_NO_ERROR
)) {
202 if (e
== EADDRNOTAVAIL
)
209 struct sockaddr
*if_addr(const char *iface
, struct ifreq
*r
) {
210 return if_addr_value(iface
, r
, SIOCGIFADDR
);
213 struct sockaddr
*if_mask(const char *iface
, struct ifreq
*r
) {
214 return if_addr_value(iface
, r
, SIOCGIFNETMASK
);
217 struct sockaddr
*if_bcast(const char *iface
, struct ifreq
*r
) {
218 return if_addr_value(iface
, r
, SIOCGIFBRDADDR
);
221 struct sockaddr
*if_network(const char *iface
) {
222 struct sockaddr
*saddr
;
223 static struct ifreq req
;
226 if (!(saddr
= if_mask(iface
, &req
)))
229 mask
= ((struct sockaddr_in
*)saddr
)->sin_addr
.s_addr
;
231 if (!(saddr
= if_addr(iface
, &req
)))
234 ((struct sockaddr_in
*)saddr
)->sin_addr
.s_addr
&= mask
;
238 int if_mtu(const char *iface
) {
239 static struct ifreq req
;
241 if (do_socket_ioctl(iface
, SIOCGIFMTU
, &req
, NULL
, PRINT_ERROR
))
247 #if defined(__linux__)
249 static void skipline(FILE *fd
) {
253 } while (ch
!= '\n' && ch
!= EOF
);
256 struct if_stat
*get_stats(const char *iface
) {
258 struct if_stat
*ifstat
;
261 if (!(ifstat
= malloc(sizeof(struct if_stat
)))) {
266 if ((fd
= fopen("/proc/net/dev", "r")) == NULL
) {
267 perror("fopen(\"/proc/net/dev\")");
277 int items
= fscanf(fd
,
278 " %20[^:]:%llu %llu %llu %llu %llu %llu %llu %llu "
279 "%llu %llu %llu %llu %llu %llu %llu %llu",
281 &ifstat
->in_bytes
, &ifstat
->in_packets
,
282 &ifstat
->in_errors
, &ifstat
->in_drops
,
283 &ifstat
->in_fifo
, &ifstat
->in_frame
,
284 &ifstat
->in_compress
, &ifstat
->in_multicast
,
285 &ifstat
->out_bytes
, &ifstat
->out_packets
,
286 &ifstat
->out_errors
, &ifstat
->out_drops
,
287 &ifstat
->out_fifo
, &ifstat
->out_colls
,
288 &ifstat
->out_carrier
, &ifstat
->out_carrier
294 fprintf(stderr
, "Invalid data read, check!\n");
298 if (!strncmp(name
, iface
, sizeof(name
))) {
314 unsigned int is_stat
;
317 { "-e", DO_EXISTS
, 0, "Reports interface existence via return code" },
318 { "-p", DO_PALL
, 0, "Print out the whole config of iface" },
319 { "-pe", DO_PEXISTS
, 0, "Print out yes or no according to existence" },
320 { "-pa", DO_PADDRESS
, 0, "Print out the address" },
321 { "-pn", DO_PMASK
, 0, "Print netmask" },
322 { "-pN", DO_PNETWORK
, 0, "Print network address" },
323 { "-pb", DO_PCAST
, 0, "Print broadcast" },
324 { "-pm", DO_PMTU
, 0, "Print mtu" },
325 #if defined(__linux__)
326 { "-ph", DO_PHWADDRESS
, 0, "Print out the hardware address" },
327 { "-pf", DO_PFLAGS
, 0, "Print flags" },
328 { "-si", DO_SINALL
, 1, "Print all statistics on input" },
329 { "-sip", DO_SINPACKETS
, 1, "Print # of in packets" },
330 { "-sib", DO_SINBYTES
, 1, "Print # of in bytes" },
331 { "-sie", DO_SINERRORS
, 1, "Print # of in errors" },
332 { "-sid", DO_SINDROPS
, 1, "Print # of in drops" },
333 { "-sif", DO_SINFIFO
, 1, "Print # of in fifo overruns" },
334 { "-sic", DO_SINCOMPRESSES
, 1, "Print # of in compress" },
335 { "-sim", DO_SINMULTICAST
, 1, "Print # of in multicast" },
336 { "-so", DO_SOUTALL
, 1, "Print all statistics on output" },
337 { "-sop", DO_SOUTPACKETS
, 1, "Print # of out packets" },
338 { "-sob", DO_SOUTBYTES
, 1, "Print # of out bytes" },
339 { "-soe", DO_SOUTERRORS
, 1, "Print # of out errors" },
340 { "-sod", DO_SOUTDROPS
, 1, "Print # of out drops" },
341 { "-sof", DO_SOUTFIFO
, 1, "Print # of out fifo overruns" },
342 { "-sox", DO_SOUTCOLLS
, 1, "Print # of out collisions" },
343 { "-soc", DO_SOUTCARRIER
, 1, "Print # of out carrier loss" },
344 { "-som", DO_SOUTMULTICAST
, 1, "Print # of out multicast" },
345 { "-bips",DO_BIPS
, 1, "Print # of incoming bytes per second" },
346 { "-bops",DO_BOPS
, 1, "Print # of outgoing bytes per second" },
350 void usage(const char *name
) {
353 fprintf(stderr
, "Usage: %s [options] iface\n", name
);
354 for (i
= 0; i
< sizeof(options
) / sizeof(options
[0]); i
++) {
355 fprintf(stderr
, " %5s %s\n",
356 options
[i
].option
, options
[i
].description
);
360 void add_do(int *ndo
, int **todo
, int act
) {
361 *todo
= realloc(*todo
, (*ndo
+1) * sizeof(int));
366 static void print_addr(struct sockaddr
*sadr
) {
368 fprintf(stderr
, "Error\n");
374 struct if_stat
*ifstats
, *ifstats2
= NULL
;
376 void please_do(int ndo
, int *todo
, const char *ifname
) {
378 static struct ifreq req
;
380 // printf("I have %d items in my queue.\n",ndo);
381 for (i
=0; i
<ndo
; i
++) {
384 exit(!if_exists(ifname
));
386 printf("%s", if_exists(ifname
) ? "yes" : "no");
389 print_addr(if_addr(ifname
, &req
));
391 #if defined(__linux__)
400 print_addr(if_mask(ifname
, &req
));
403 print_addr(if_bcast(ifname
, &req
));
406 printf("%d", if_mtu(ifname
));
409 print_addr(if_network(ifname
));
412 print_addr(if_addr(ifname
, &req
));
414 print_addr(if_mask(ifname
, &req
));
416 print_addr(if_bcast(ifname
, &req
));
418 printf("%d", if_mtu(ifname
));
420 #if defined(__linux__)
422 printf("%llu",ifstats
->in_packets
);
425 printf("%llu",ifstats
->in_bytes
);
428 printf("%llu",ifstats
->in_errors
);
431 printf("%llu",ifstats
->in_drops
);
434 printf("%llu",ifstats
->in_fifo
);
437 printf("%llu",ifstats
->in_frame
);
439 case DO_SINCOMPRESSES
:
440 printf("%llu",ifstats
->in_compress
);
442 case DO_SINMULTICAST
:
443 printf("%llu",ifstats
->in_multicast
);
446 printf("%llu %llu %llu %llu %llu %llu %llu %llu",
447 ifstats
->in_bytes
, ifstats
->in_packets
,
448 ifstats
->in_errors
, ifstats
->in_drops
,
449 ifstats
->in_fifo
, ifstats
->in_frame
,
450 ifstats
->in_compress
, ifstats
->in_multicast
);
453 printf("%llu",ifstats
->out_bytes
);
456 printf("%llu",ifstats
->out_packets
);
459 printf("%llu",ifstats
->out_errors
);
462 printf("%llu",ifstats
->out_drops
);
465 printf("%llu",ifstats
->out_fifo
);
468 printf("%llu",ifstats
->out_colls
);
471 printf("%llu",ifstats
->out_carrier
);
473 case DO_SOUTMULTICAST
:
474 printf("%llu",ifstats
->out_multicast
);
477 if (ifstats2
== NULL
) {
479 ifstats2
= get_stats(ifname
);
481 printf("%llu", ifstats2
->in_bytes
-ifstats
->in_bytes
);
484 if (ifstats2
== NULL
) {
486 ifstats2
= get_stats(ifname
);
488 printf("%llu", ifstats2
->out_bytes
-ifstats
->out_bytes
);
491 printf("%llu %llu %llu %llu %llu %llu %llu %llu",
492 ifstats
->out_bytes
, ifstats
->out_packets
,
493 ifstats
->out_errors
, ifstats
->out_drops
,
494 ifstats
->out_fifo
, ifstats
->out_colls
,
495 ifstats
->out_carrier
, ifstats
->out_multicast
);
499 printf("Unknown command: %d", todo
[i
]);
506 int main(int argc
, char *argv
[]) {
512 unsigned int i
, found
;
519 while (narg
< argc
- 1) {
524 for (i
= 0; i
< sizeof(options
) / sizeof(options
[0]); i
++) {
525 if (!strcmp(argv
[narg
], options
[i
].option
)) {
526 add_do(&ndo
, &todo
, options
[i
].flag
);
527 do_stats
|= options
[i
].is_stat
;
536 if (argv
[narg
][0] == '-') {
546 if (narg
+ 1 < argc
|| !ifname
) {
551 #if defined(__linux__)
552 if (do_stats
&& (ifstats
= get_stats(ifname
)) == NULL
) {
553 fprintf(stderr
, "Error getting statistics for %s\n", ifname
);
558 please_do(ndo
, todo
, ifname
);