ifconfig.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553
  1. /* ifconfig.c - Configure network interface.
  2. *
  3. * Copyright 2012 Ranjan Kumar <ranjankumar.bth@gmail.com>
  4. * Copyright 2012 Kyungwan Han <asura321@gamil.com>
  5. * Reviewed by Kyungsu Kim <kaspyx@gmail.com>
  6. *
  7. * Not in SUSv4.
  8. *
  9. * Obsolete fields included for historical purposes:
  10. * irq|io_addr|mem_start ADDR - micromanage obsolete hardware
  11. * outfill|keepalive INTEGER - SLIP analog dialup line quality monitoring
  12. * metric INTEGER - added to Linux 0.9.10 with comment "never used", still true
  13. USE_IFCONFIG(NEWTOY(ifconfig, "^?aS", TOYFLAG_SBIN))
  14. config IFCONFIG
  15. bool "ifconfig"
  16. default y
  17. help
  18. usage: ifconfig [-aS] [INTERFACE [ACTION...]]
  19. Display or configure network interface.
  20. With no arguments, display active interfaces. First argument is interface
  21. to operate on, one argument by itself displays that interface.
  22. -a All interfaces displayed, not just active ones
  23. -S Short view, one line per interface
  24. Standard ACTIONs to perform on an INTERFACE:
  25. ADDR[/MASK] - set IPv4 address (1.2.3.4/5) and activate interface
  26. add|del ADDR[/LEN] - add/remove IPv6 address (1111::8888/128)
  27. up|down - activate or deactivate interface
  28. Advanced ACTIONs (default values usually suffice):
  29. default - remove IPv4 address
  30. netmask ADDR - set IPv4 netmask via 255.255.255.0 instead of /24
  31. txqueuelen LEN - number of buffered packets before output blocks
  32. mtu LEN - size of outgoing packets (Maximum Transmission Unit)
  33. broadcast ADDR - Set broadcast address
  34. pointopoint ADDR - PPP and PPPOE use this instead of "route add default gw"
  35. hw TYPE ADDR - set hardware (mac) address (type = ether|infiniband)
  36. rename NEWNAME - rename interface
  37. Flags you can set on an interface (or -remove by prefixing with -):
  38. arp - don't use Address Resolution Protocol to map LAN routes
  39. promisc - don't discard packets that aren't to this LAN hardware address
  40. multicast - force interface into multicast mode if the driver doesn't
  41. allmulti - promisc for multicast packets
  42. */
  43. #define FOR_ifconfig
  44. #include "toys.h"
  45. #include <net/if_arp.h>
  46. #include <net/ethernet.h>
  47. GLOBALS(
  48. int sockfd;
  49. )
  50. // Convert hostname to binary address for AF_INET or AF_INET6
  51. // return /prefix (or range max if none)
  52. static int get_addrinfo(char *host, sa_family_t af, void *addr)
  53. {
  54. struct addrinfo hints, *result, *rp = 0;
  55. int status, len;
  56. char *from, *slash;
  57. memset(&hints, 0 , sizeof(struct addrinfo));
  58. hints.ai_family = af;
  59. hints.ai_socktype = SOCK_STREAM;
  60. slash = strchr(host, '/');
  61. if (slash) *slash = 0;
  62. status = getaddrinfo(host, NULL, &hints, &result);
  63. if (!status)
  64. for (rp = result; rp; rp = rp->ai_next)
  65. if (rp->ai_family == af) break;
  66. if (!rp) error_exit("bad address '%s' : %s", host, gai_strerror(status));
  67. // ai_addr isn't struct in_addr or in6_addr, it's struct sockaddr. Of course.
  68. // You'd think ipv4 and ipv6 would have some basic compatibility, but no.
  69. from = ((char *)rp->ai_addr) + 4;
  70. if (af == AF_INET6) {
  71. len = 16;
  72. from += 4; // skip "flowinfo" field ipv6 puts before address
  73. } else len = 4;
  74. memcpy(addr, from, len);
  75. freeaddrinfo(result);
  76. len = -1;
  77. if (slash) len = atolx_range(slash+1, 0, (af == AF_INET) ? 32 : 128);
  78. return len;
  79. }
  80. static void display_ifconfig(char *name, int always, unsigned long long val[])
  81. {
  82. struct ifreq ifre;
  83. struct sockaddr_in *si = (void *)&ifre.ifr_addr;
  84. struct {
  85. int type;
  86. char *title;
  87. } types[] = {
  88. {ARPHRD_LOOPBACK, "Local Loopback"}, {ARPHRD_ETHER, "Ethernet"},
  89. {ARPHRD_PPP, "Point-to-Point Protocol"}, {ARPHRD_INFINIBAND, "InfiniBand"},
  90. {ARPHRD_SIT, "IPv6-in-IPv4"}, {-1, "UNSPEC"}
  91. };
  92. int i;
  93. char *pp;
  94. FILE *fp;
  95. short flags;
  96. xstrncpy(ifre.ifr_name, name, IFNAMSIZ);
  97. if (ioctl(TT.sockfd, SIOCGIFFLAGS, &ifre)<0) perror_exit_raw(name);
  98. flags = ifre.ifr_flags;
  99. if (!always && !(flags & IFF_UP)) return;
  100. if (toys.optflags&FLAG_S) {
  101. unsigned uu = 0;
  102. int len;
  103. ioctl(TT.sockfd, SIOCGIFADDR, &ifre);
  104. len = printf("%*s %s", -9, name, inet_ntoa(si->sin_addr));
  105. if (!ioctl(TT.sockfd, SIOCGIFNETMASK, &ifre))
  106. uu = htonl(*(unsigned *)&(si->sin_addr));
  107. for (i = 0; uu; i++) uu <<= 1;
  108. len += printf("/%d", i);
  109. printf("%*c", 29-len, ' ');
  110. }
  111. // Query hardware type and hardware address.
  112. // Not xioctl because you don't have permission for this on Android.
  113. ioctl(TT.sockfd, SIOCGIFHWADDR, &ifre);
  114. if (toys.optflags&FLAG_S)
  115. for (i=0; i<6; i++) printf(":%02x"+!i, ifre.ifr_hwaddr.sa_data[i]);
  116. else {
  117. for (i=0; i < ARRAY_LEN(types)-1; i++)
  118. if (ifre.ifr_hwaddr.sa_family == types[i].type) break;
  119. xprintf("%-9s Link encap:%s ", name, types[i].title);
  120. if(ifre.ifr_hwaddr.sa_family == ARPHRD_ETHER) {
  121. xprintf("HWaddr ");
  122. for (i=0; i<6; i++) xprintf(":%02x"+!i, ifre.ifr_hwaddr.sa_data[i]);
  123. }
  124. sprintf(toybuf, "/sys/class/net/%.15s/device/driver", name);
  125. if (readlink0(toybuf, toybuf, sizeof(toybuf))>0)
  126. if ((pp = strrchr(toybuf, '/'))) xprintf(" Driver %s", pp+1);
  127. xputc('\n');
  128. }
  129. // If an address is assigned record that.
  130. ifre.ifr_addr.sa_family = AF_INET;
  131. memset(&ifre.ifr_addr, 0, sizeof(ifre.ifr_addr));
  132. ioctl(TT.sockfd, SIOCGIFADDR, &ifre);
  133. pp = (char *)&ifre.ifr_addr;
  134. for (i = 0; i<sizeof(ifre.ifr_addr); i++) if (pp[i]) break;
  135. if (!(toys.optflags&FLAG_S) && i != sizeof(ifre.ifr_addr)) {
  136. struct sockaddr_in *si = (struct sockaddr_in *)&ifre.ifr_addr;
  137. struct {
  138. char *name;
  139. int flag, ioctl;
  140. } addr[] = {
  141. {"addr", 0, 0},
  142. {"P-t-P", IFF_POINTOPOINT, SIOCGIFDSTADDR},
  143. {"Bcast", IFF_BROADCAST, SIOCGIFBRDADDR},
  144. {"Mask", 0, SIOCGIFNETMASK}
  145. };
  146. // TODO: can this be ipv6? Why are we checking here when ipv6 source later?
  147. xprintf("%10c%s", ' ', (si->sin_family == AF_INET) ? "inet" :
  148. (si->sin_family == AF_INET6) ? "inet6" : "unspec");
  149. for (i=0; i<ARRAY_LEN(addr); i++) {
  150. if (!addr[i].flag || (flags & addr[i].flag)) {
  151. if (addr[i].ioctl && ioctl(TT.sockfd, addr[i].ioctl, &ifre))
  152. si->sin_family = 0;
  153. xprintf(" %s:%s ", addr[i].name,
  154. (si->sin_family == 0xFFFF || !si->sin_family)
  155. ? "[NOT SET]" : inet_ntoa(si->sin_addr));
  156. }
  157. }
  158. xputc('\n');
  159. }
  160. fp = fopen(pp = "/proc/net/if_inet6", "r");
  161. if (fp) {
  162. char iface_name[IFNAMSIZ];
  163. int plen, iscope;
  164. while (fgets(toybuf, sizeof(toybuf), fp)) {
  165. int nitems;
  166. char ipv6_addr[40];
  167. nitems = sscanf(toybuf, "%32s %*08x %02x %02x %*02x %15s\n",
  168. ipv6_addr, &plen, &iscope, iface_name);
  169. if (nitems<0 && feof(fp)) break;
  170. if (nitems != 4) perror_exit("bad %s", pp);
  171. if (!strcmp(name, iface_name)) {
  172. struct sockaddr_in6 s6;
  173. char *ptr = ipv6_addr+sizeof(ipv6_addr)-1;
  174. // convert giant hex string into colon-spearated ipv6 address by
  175. // inserting ':' every 4 characters.
  176. for (i = 32; i; i--)
  177. if ((*(ptr--) = ipv6_addr[i])) if (!(i&3)) *(ptr--) = ':';
  178. // Convert to binary and back to get abbreviated :: version
  179. if (inet_pton(AF_INET6, ipv6_addr, (void *)&s6.sin6_addr) > 0) {
  180. if (inet_ntop(AF_INET6, &s6.sin6_addr, toybuf, sizeof(toybuf))) {
  181. char *scopes[] = {"Global","Host","Link","Site","Compat"},
  182. *scope = "Unknown";
  183. for (i=0; i<ARRAY_LEN(scopes); i++)
  184. if (iscope == (!!i)<<(i+3)) scope = scopes[i];
  185. if (toys.optflags&FLAG_S) xprintf(" %s/%d@%c", toybuf, plen,*scope);
  186. else xprintf("%10cinet6 addr: %s/%d Scope: %s\n",
  187. ' ', toybuf, plen, scope);
  188. }
  189. }
  190. }
  191. }
  192. fclose(fp);
  193. }
  194. if (toys.optflags&FLAG_S) {
  195. xputc('\n');
  196. return;
  197. }
  198. xprintf("%10c", ' ');
  199. if (flags) {
  200. unsigned short mask = 1;
  201. char **s, *str[] = {
  202. "UP", "BROADCAST", "DEBUG", "LOOPBACK", "POINTOPOINT", "NOTRAILERS",
  203. "RUNNING", "NOARP", "PROMISC", "ALLMULTI", "MASTER", "SLAVE", "MULTICAST",
  204. "PORTSEL", "AUTOMEDIA", "DYNAMIC", NULL
  205. };
  206. for (s = str; *s; s++) {
  207. if (flags & mask) xprintf("%s ", *s);
  208. mask <<= 1;
  209. }
  210. } else xprintf("[NO FLAGS] ");
  211. if (ioctl(TT.sockfd, SIOCGIFMTU, &ifre) < 0) ifre.ifr_mtu = 0;
  212. xprintf(" MTU:%d", ifre.ifr_mtu);
  213. if (ioctl(TT.sockfd, SIOCGIFMETRIC, &ifre) < 0) ifre.ifr_metric = 0;
  214. if (!ifre.ifr_metric) ifre.ifr_metric = 1;
  215. xprintf(" Metric:%d", ifre.ifr_metric);
  216. // non-virtual interface
  217. if (val) {
  218. char *label[] = {"RX bytes", "RX packets", "errors", "dropped", "overruns",
  219. "frame", 0, 0, "TX bytes", "TX packets", "errors", "dropped", "overruns",
  220. "collisions", "carrier", 0, "txqueuelen"};
  221. signed char order[] = {-1, 1, 2, 3, 4, 5, -1, 9, 10, 11, 12, 14, -1,
  222. 13, 16, -1, 0, 8};
  223. int i;
  224. // Query txqueuelen
  225. if (ioctl(TT.sockfd, SIOCGIFTXQLEN, &ifre) >= 0) val[16] = ifre.ifr_qlen;
  226. else val[16] = -1;
  227. for (i = 0; i<sizeof(order); i++) {
  228. int j = order[i];
  229. if (j < 0) xprintf("\n%10c", ' ');
  230. else xprintf("%s:%llu ", label[j], val[j]);
  231. }
  232. }
  233. xputc('\n');
  234. if(!ioctl(TT.sockfd, SIOCGIFMAP, &ifre) && (ifre.ifr_map.irq ||
  235. ifre.ifr_map.mem_start || ifre.ifr_map.dma || ifre.ifr_map.base_addr))
  236. {
  237. xprintf("%10c", ' ');
  238. if(ifre.ifr_map.irq) xprintf("Interrupt:%d ", ifre.ifr_map.irq);
  239. if(ifre.ifr_map.base_addr >= 0x100) // IO_MAP_INDEX
  240. xprintf("Base address:0x%x ", ifre.ifr_map.base_addr);
  241. if(ifre.ifr_map.mem_start)
  242. xprintf("Memory:%lx-%lx ", ifre.ifr_map.mem_start, ifre.ifr_map.mem_end);
  243. if(ifre.ifr_map.dma) xprintf("DMA chan:%x ", ifre.ifr_map.dma);
  244. xputc('\n');
  245. }
  246. xputc('\n');
  247. }
  248. static void show_iface(char *iface_name)
  249. {
  250. char *name;
  251. struct string_list *ifaces = 0, *sl;
  252. int i, j;
  253. FILE *fp;
  254. fp = xfopen("/proc/net/dev", "r");
  255. for (i=0; fgets(toybuf, sizeof(toybuf), fp); i++) {
  256. char *buf = toybuf;
  257. unsigned long long val[17];
  258. if (i<2) continue;
  259. while (isspace(*buf)) buf++;
  260. name = strsep(&buf, ":");
  261. if(!buf) error_exit("bad name %s", name);
  262. errno = 0;
  263. for (j=0; j<16 && !errno; j++) val[j] = strtoll(buf, &buf, 0);
  264. if (errno) perror_exit("bad %s at %s", name, buf);
  265. if (iface_name) {
  266. if (!strcmp(iface_name, name)) {
  267. display_ifconfig(iface_name, 1, val);
  268. return;
  269. }
  270. } else {
  271. sl = xmalloc(sizeof(*sl)+strlen(name)+1);
  272. strcpy(sl->str, name);
  273. sl->next = ifaces;
  274. ifaces = sl;
  275. display_ifconfig(sl->str, toys.optflags & FLAG_a, val);
  276. }
  277. }
  278. fclose(fp);
  279. if (iface_name) display_ifconfig(iface_name, 1, 0);
  280. else {
  281. struct ifconf ifcon;
  282. struct ifreq *ifre;
  283. int num;
  284. // Loop until buffer's big enough
  285. ifcon.ifc_buf = NULL;
  286. for (num = 30;;num += 10) {
  287. ifcon.ifc_len = sizeof(struct ifreq)*num;
  288. ifcon.ifc_buf = xrealloc(ifcon.ifc_buf, ifcon.ifc_len);
  289. xioctl(TT.sockfd, SIOCGIFCONF, &ifcon);
  290. if (ifcon.ifc_len != sizeof(struct ifreq)*num) break;
  291. }
  292. ifre = ifcon.ifc_req;
  293. for(num = 0; num < ifcon.ifc_len && ifre; num += sizeof(struct ifreq), ifre++)
  294. {
  295. // Skip duplicates
  296. for(sl = ifaces; sl; sl = sl->next)
  297. if(!strcmp(sl->str, ifre->ifr_name)) break;
  298. if(!sl) display_ifconfig(ifre->ifr_name, toys.optflags & FLAG_a, 0);
  299. }
  300. free(ifcon.ifc_buf);
  301. }
  302. llist_traverse(ifaces, free);
  303. }
  304. // Encode offset and size of field into an int, and make result negative
  305. #define IFREQ_OFFSZ(x) -(int)((offsetof(struct ifreq, x)<<16) + sizeof(ifre.x))
  306. void ifconfig_main(void)
  307. {
  308. char **argv = toys.optargs;
  309. struct ifreq ifre;
  310. int i;
  311. TT.sockfd = xsocket(AF_INET, SOCK_DGRAM, 0);
  312. if(toys.optc < 2) {
  313. show_iface(*argv);
  314. return;
  315. }
  316. // Open interface
  317. memset(&ifre, 0, sizeof(struct ifreq));
  318. xstrncpy(ifre.ifr_name, *argv, IFNAMSIZ);
  319. // Perform operations on interface
  320. while(*++argv) {
  321. // Table of known operations
  322. struct argh {
  323. char *name;
  324. int on, off; // set, clear
  325. } try[] = {
  326. {0, IFF_UP|IFF_RUNNING, SIOCSIFADDR},
  327. {"up", IFF_UP|IFF_RUNNING, 0},
  328. {"down", 0, IFF_UP},
  329. {"arp", 0, IFF_NOARP},
  330. {"promisc", IFF_PROMISC, 0},
  331. {"allmulti", IFF_ALLMULTI, 0},
  332. {"multicast", IFF_MULTICAST, 0},
  333. {"pointopoint", IFF_POINTOPOINT, SIOCSIFDSTADDR},
  334. {"broadcast", IFF_BROADCAST, SIOCSIFBRDADDR},
  335. {"netmask", 0, SIOCSIFNETMASK},
  336. {"dstaddr", 0, SIOCSIFDSTADDR},
  337. {"mtu", IFREQ_OFFSZ(ifr_mtu), SIOCSIFMTU},
  338. {"rename", IFREQ_OFFSZ(ifr_newname), SIOCSIFNAME},
  339. {"keepalive", IFREQ_OFFSZ(ifr_data), SIOCDEVPRIVATE}, // SIOCSKEEPALIVE
  340. {"outfill", IFREQ_OFFSZ(ifr_data), SIOCDEVPRIVATE+2}, // SIOCSOUTFILL
  341. {"metric", IFREQ_OFFSZ(ifr_metric), SIOCSIFMETRIC},
  342. {"txqueuelen", IFREQ_OFFSZ(ifr_qlen), SIOCSIFTXQLEN},
  343. {"mem_start", IFREQ_OFFSZ(ifr_map.mem_start), SIOCSIFMAP},
  344. {"io_addr", IFREQ_OFFSZ(ifr_map.base_addr), SIOCSIFMAP},
  345. {"irq", IFREQ_OFFSZ(ifr_map.irq), SIOCSIFMAP},
  346. {"inet", 0, 0},
  347. {"inet6", 0, 0}
  348. };
  349. char *s = *argv;
  350. int rev = (*s == '-');
  351. s += rev;
  352. // "set hardware address" is oddball enough to special case
  353. if (!strcmp(*argv, "hw")) {
  354. char *hw_addr, *ptr, *p;
  355. struct sockaddr *sock = &ifre.ifr_hwaddr;
  356. int count = 6;
  357. ptr = p = (char *)sock->sa_data;
  358. memset(sock, 0, sizeof(struct sockaddr));
  359. if (argv[1]) {
  360. if (!strcmp("ether", *++argv)) sock->sa_family = ARPHRD_ETHER;
  361. else if (!strcmp("infiniband", *argv)) {
  362. sock->sa_family = ARPHRD_INFINIBAND;
  363. count = 20;
  364. p = ptr = toybuf;
  365. }
  366. }
  367. if (!sock->sa_family || !argv[1]) help_exit("bad hw '%s'", *argv);
  368. hw_addr = *++argv;
  369. // Parse and verify address.
  370. while (*hw_addr && (p-ptr) < count) {
  371. int val, len = 0;
  372. if (*hw_addr == ':') hw_addr++;
  373. sscanf(hw_addr, "%2x%n", &val, &len);
  374. if (!len || len > 2) break; // 1 nibble can be set e.g. C2:79:38:95:D:A
  375. hw_addr += len;
  376. *p++ = val;
  377. }
  378. if ((p-ptr) != count || *hw_addr)
  379. error_exit("bad hw-addr '%s'", *argv);
  380. // the linux kernel's "struct sockaddr" (include/linux/socket.h in the
  381. // kernel source) only has 14 bytes of sa_data, and an infiniband address
  382. // is 20. So if we go through the ioctl, the kernel will truncate
  383. // infiniband addresses, meaning we have to go through sysfs instead.
  384. if (sock->sa_family == ARPHRD_INFINIBAND && !strchr(ifre.ifr_name, '/')) {
  385. int fd;
  386. sprintf(toybuf, "/sys/class/net/%s/address", ifre.ifr_name);
  387. fd = xopen(toybuf, O_RDWR);
  388. xwrite(fd, *argv, strlen(*argv));
  389. close(fd);
  390. } else xioctl(TT.sockfd, SIOCSIFHWADDR, &ifre);
  391. continue;
  392. // Add/remove ipv6 address to interface
  393. } else if (!strcmp(*argv, "add") || !strcmp(*argv, "del")) {
  394. struct ifreq_inet6 {
  395. struct in6_addr addr;
  396. unsigned prefix;
  397. int index;
  398. } ifre6;
  399. int plen, fd6 = xsocket(AF_INET6, SOCK_DGRAM, 0);
  400. if (!argv[1]) help_exit("%s", *argv);
  401. plen = get_addrinfo(argv[1], AF_INET6, &ifre6.addr);
  402. if (plen < 0) plen = 128;
  403. xioctl(fd6, SIOCGIFINDEX, &ifre);
  404. ifre6.index = ifre.ifr_ifindex;
  405. ifre6.prefix = plen;
  406. xioctl(fd6, **(argv++)=='a' ? SIOCSIFADDR : SIOCDIFADDR, &ifre6);
  407. close(fd6);
  408. continue;
  409. // Iterate through table to find/perform operation
  410. } else for (i = 0; i<ARRAY_LEN(try); i++) {
  411. struct argh *t = try+i;
  412. int on = t->on, off = t->off;
  413. // First entry in list assigns address to interface (no command name)
  414. if (!t->name) {
  415. if (isdigit(**argv) || !strcmp(*argv, "default")) argv--;
  416. else continue;
  417. } else if (strcmp(t->name, s)) continue;
  418. // Is this an SIOCSI entry?
  419. if ((off|0xff) == 0x89ff) {
  420. if (!rev) {
  421. if (!*++argv) error_exit("%s needs argument", t->name);
  422. // Assign value to ifre field and call ioctl? (via IFREQ_OFFSZ.)
  423. if (on < 0) {
  424. void *dest = ((on = -on)>>16)+(char *)&ifre;
  425. if (off == SIOCSIFMAP) xioctl(TT.sockfd, SIOCGIFMAP, &ifre);
  426. if (SIOCSIFNAME) xstrncpy(dest, *argv, on&0xffff);
  427. else poke(dest, strtoul(*argv, 0, 0), on&15);
  428. xioctl(TT.sockfd, off, &ifre);
  429. break;
  430. } else {
  431. struct sockaddr_in *si = (struct sockaddr_in *)&ifre.ifr_addr;
  432. int mask = -1;
  433. si->sin_family = AF_INET;
  434. if (!strcmp(*argv, "default")) si->sin_addr.s_addr = INADDR_ANY;
  435. else mask = get_addrinfo(*argv, AF_INET, &si->sin_addr);
  436. xioctl(TT.sockfd, off, &ifre);
  437. // Handle /netmask
  438. if (mask >= 0) {
  439. // sin_addr probably isn't unaligned, but just in case...
  440. mask = htonl((~0)<<(32-mask));
  441. memcpy(&si->sin_addr, &mask, 4);
  442. xioctl(TT.sockfd, SIOCSIFNETMASK, &ifre);
  443. }
  444. }
  445. }
  446. off = 0;
  447. }
  448. // Set flags
  449. if (on || off) {
  450. xioctl(TT.sockfd, SIOCGIFFLAGS, &ifre);
  451. ifre.ifr_flags &= ~(rev ? on : off);
  452. ifre.ifr_flags |= (rev ? off : on);
  453. xioctl(TT.sockfd, SIOCSIFFLAGS, &ifre);
  454. }
  455. break;
  456. }
  457. if (i == ARRAY_LEN(try)) help_exit("bad argument '%s'", *argv);
  458. }
  459. close(TT.sockfd);
  460. }