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