/* * Copyright (c) 2017 Denys Vlasenko <vda.linux@googlemail.com> * * Licensed under GPLv2, see file LICENSE in this source tree. */ //config:config IPCONFIG //config: bool "ipconfig" //config: default y //config: help //config: (Auto)configure network. //applet:IF_IPCONFIG(APPLET(ipconfig, BB_DIR_BIN, BB_SUID_DROP)) //kbuild:lib-$(CONFIG_IPCONFIG) += ipconfig.o #include <net/if.h> #include "libbb.h" struct globals { int fixed; const char *hostname; }; #define G (*ptr_to_globals) #define INIT_G() do { \ SET_PTR_TO_GLOBALS(xzalloc(sizeof(G))); \ } while (0) struct dev { const char *name; uint8_t fixed; uint32_t ip_addr; uint32_t ip_netmask; uint32_t ip_server; uint32_t ip_router; }; static int parse_method(const char *method) { int fixed; fixed = (method[0] != '\0'); if (fixed) { /* if it's not "" */ fixed = index_in_strings( /* 0 */ "on""\0" /* 1 */ "any""\0" /* 2 */ "both""\0" /* 3 */ "dhcp""\0" /* 4 */ "bootp""\0" /* 5 */ "rarp""\0" /* 6 */ "none""\0" /* 7 */ "static""\0" /* 8 */ "off""\0" , method ); if (fixed > 0) fixed /= 6; } return fixed; } static uint32_t parse_addr(const char *ip) { struct in_addr in; if (inet_aton(ip, &in) == 0) bb_error_msg_and_die("bad IP address '%s'", ip); return in.s_addr; } static struct dev* find_device(llist_t *iface_list, const char *name) { while (iface_list) { struct dev *dev = (void*) iface_list->data; if (strcmp(dev->name, name) == 0) return dev; iface_list = iface_list->link; } return NULL; } static void set_from_template(struct dev *dev, struct dev *template) { if (template->ip_addr != 0) dev->ip_addr = template->ip_addr; if (template->ip_netmask != 0) dev->ip_netmask = template->ip_netmask; if (template->ip_server != 0) dev->ip_server = template->ip_server; if (template->ip_router != 0) dev->ip_router = template->ip_router; dev->fixed = template->fixed; } // "ip=PROTO" - also implies -o // "nfsaddrs=PROTO" - also implies -o // "<devname>" // "[ip=/nfsaddrs=]IP:SERVER_IP:ROUTER:NETMASK:HOSTNAME:IFACE:METHOD" // all optional. trailing empty :: can be skipped, only one : needs to be there // (to distinguish from other formats). // ":::::eth0" - dhcp on eth0 // ":" - dhcp on all ifaces // "::1.2.3.4" - dhcp on all ifaces, gateway is 1.2.3.4 (fairly nonsensical) static void add_all_devices(llist_t **iface_list, struct dev *template); static struct dev* add_device(llist_t **iface_list, char *ip) { struct dev *dev; dev = xzalloc(sizeof(*dev)); dev->fixed = G.fixed; if (strncmp("ip=", ip, 3) == 0 || strncmp("nfsaddrs=", ip, 9) == 0 ) { int fixed; ip = strchr(ip, '=') + 1; fixed = parse_method(ip); if (fixed >= 0) { add_all_devices(iface_list, dev); free(dev); return NULL; } } if (!strchr(ip, ':')) { dev->name = ip; } else { unsigned opt = 0; while (ip && *ip) { char *next = strchr(ip, ':'); if (next) *next++ = '\0'; if (opt > 6) bb_error_msg_and_die("too many options for %s", dev->name); if (ip[0]) switch (opt) { case 0: dev->ip_addr = parse_addr(ip); break; case 1: dev->ip_server = parse_addr(ip); break; case 2: dev->ip_router = parse_addr(ip); break; case 3: dev->ip_netmask = parse_addr(ip); break; case 4: if (G.hostname && strcmp(G.hostname, ip) != 0) bb_error_msg_and_die("hostname must be the same"); G.hostname = ip; break; case 5: dev->name = ip; break; case 6: dev->fixed = parse_method(ip); break; } ip = next; opt++; } } if (dev->name == NULL || strcmp(dev->name, "all") == 0 ) { add_all_devices(iface_list, dev); free(dev); return NULL; } llist_add_to_end(iface_list, dev); return dev; } static void add_all_devices(llist_t **iface_list, struct dev *template) { DIR *d; struct dirent *de; #define sys_class_net "/sys/class/net" /* All forms of "config all ifaces" imply -o */ option_mask32 |= 1; d = opendir(sys_class_net); if (!d) return; while ((de = readdir(d)) != NULL) { struct dev *dev; char *filename; char p[sizeof(long)*3]; unsigned long flags; int r; /* Exclude devices beginning with dots as well as . and .. */ if (de->d_name[0] == '.') continue; filename = xasprintf("%s/%s/flags", sys_class_net, de->d_name); r = open_read_close(filename, p, sizeof(p) - 1); free(filename); if (r < 0) continue; p[r] = '\0'; /* file's format is "0xNNNN\n" */ flags = bb_strtoul(p, NULL, 0); /* * Heuristic for if this is a reasonable boot interface. * This is the same logic the in-kernel ipconfig uses. */ if (flags & IFF_LOOPBACK) continue; if (!(flags & (IFF_BROADCAST | IFF_POINTOPOINT))) continue; if (find_device(*iface_list, de->d_name)) continue; dev = add_device(iface_list, xstrdup(de->d_name)); if (dev) set_from_template(dev, template); } closedir(d); #undef sys_class_net } //usage:#define ipconfig_trivial_usage //usage: "[-c METHOD] [-t TIMEOUT] [-on] [-i VENDOR_ID] [-p PORT] [-d] IFACE..." //usage:#define ipconfig_full_usage "\n\n" //usage: "(Auto)configure network" //usage: "\n" //usage: "\n"" -c METHOD off/none/static or on/dhcp (default)" //usage: "\n"" -t SECONDS Give up after SECONDS" //usage: "\n"" -o Stop after one interface is configured" //usage: "\n"" -n Dry run" //usage: "\n"" -i VENDOR_ID DHCP vendor id (default '')" //usage: "\n"" -p PORT DHCP port to use" //usage: "\n"" [-d] IFACE... Interface(s)" //usage: "\n" //usage: "\n"" IFACE can be:" //usage: "\n"" all - configure all interfaces" //usage: "\n"" IFACE - configure this interface" //usage: "\n"" IP:SERVER_IP:ROUTER:NETMASK:HOSTNAME:IFACE:METHOD (all optional)" // TIMEOUT defaults to infinite // -d actually is an option with an argument // (not a clue why klibc-utils has two ways to specify interfaces) int ipconfig_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; int ipconfig_main(int argc UNUSED_PARAM, char **argv) { const char *method = ""; const char *vendor_id = ""; llist_t *devname_list = NULL; llist_t *iface_list; int timeout = -1; unsigned port; unsigned opt; INIT_G(); opt = getopt32(argv, "onc:t:i:p:+d:*", &method, &timeout, &vendor_id, &port, &devname_list ); argv += optind; G.fixed = parse_method(method); if (G.fixed < 0) bb_show_usage(); iface_list = NULL; while (devname_list) add_device(&iface_list, (char*) llist_pop(&devname_list)); while (*argv) add_device(&iface_list, *argv++); while (iface_list) { struct dev *dev = (void*) iface_list->data; printf("name:'%s'\n", dev->name); printf("fixed:%u\n" , dev->fixed); printf("ip:%s/" , inet_ntoa(*(struct in_addr*)&dev->ip_addr)); printf("%s\n" , inet_ntoa(*(struct in_addr*)&dev->ip_netmask)); printf("server:%s\n", inet_ntoa(*(struct in_addr*)&dev->ip_server)); printf("router:%s\n", inet_ntoa(*(struct in_addr*)&dev->ip_router)); iface_list = iface_list->link; } bb_error_msg("hostname:'%s'", G.hostname); bb_error_msg("fixed:%u", G.fixed); return EXIT_SUCCESS; } //After device is configured, write out a "/run/net-IFACE.conf" file: // // udchcp env values: //write_option("DEVICE", dev->name); interface=eth0 //write_option("PROTO", method); //write_option("IPV4ADDR", dev->ip_addr); ip=10.43.17.38 //write_option("IPV4BROADCAST", dev->ip_broadcast); subnet=255.255.255.0 mask=24 //write_option("IPV4NETMASK", dev->ip_netmask); subnet=255.255.255.0 mask=24 //write_option("IPV4GATEWAY", dev->ip_gateway); router=10.43.17.254 //write_option("IPV4DNS0", dev->ip_nameserver[0]); dns=10.38.5.26 10.11.5.19 //write_option("IPV4DNS1", dev->ip_nameserver[1]); dns=10.38.5.26 10.11.5.19 //write_option("HOSTNAME", dev->hostname); hostname="STR" //write_option("DNSDOMAIN", dev->dnsdomainname); domain=domain.com //write_option("NISDOMAIN", dev->nisdomainname); nisdomain="STR" //write_option("ROOTSERVER", my_inet_ntoa(dev->ip_server)); serverid=10.44.6.2 //write_option("ROOTPATH", dev->bootpath); rootpath="STR" //write_option("filename", dev->filename); boot_file=/pxelinux.0 //write_option("UPTIME", dev->uptime); sysinfo()->uptime //write_option("DHCPLEASETIME", dev->dhcpleasetime); lease=44148 //write_option("DOMAINSEARCH", dev->domainsearch); search="ABC DEF" // //(write_option writes out single-quote escaped string, VAR='VAL')