udhcpc6: make -O OPT work

Patch is based on work by tiggerswelt.net.

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
Denys Vlasenko 2017-06-28 19:18:17 +02:00
parent ae2b9f286c
commit ba4fbca8a8
6 changed files with 149 additions and 58 deletions

View File

@ -14,6 +14,7 @@ const uint8_t MAC_BCAST_ADDR[6] ALIGN2 = {
0xff, 0xff, 0xff, 0xff, 0xff, 0xff 0xff, 0xff, 0xff, 0xff, 0xff, 0xff
}; };
#if ENABLE_UDHCPC || ENABLE_UDHCPD
/* Supported options are easily added here. /* Supported options are easily added here.
* See RFC2132 for more options. * See RFC2132 for more options.
* OPTION_REQ: these options are requested by udhcpc (unless -o). * OPTION_REQ: these options are requested by udhcpc (unless -o).
@ -136,6 +137,7 @@ const char dhcp_option_strings[] ALIGN1 =
"msstaticroutes""\0"/* DHCP_MS_STATIC_ROUTES */ "msstaticroutes""\0"/* DHCP_MS_STATIC_ROUTES */
"wpad" "\0" /* DHCP_WPAD */ "wpad" "\0" /* DHCP_WPAD */
; ;
#endif
/* Lengths of the option types in binary form. /* Lengths of the option types in binary form.
* Used by: * Used by:
@ -190,21 +192,26 @@ static void log_option(const char *pfx, const uint8_t *opt)
# define log_option(pfx, opt) ((void)0) # define log_option(pfx, opt) ((void)0)
#endif #endif
unsigned FAST_FUNC udhcp_option_idx(const char *name) unsigned FAST_FUNC udhcp_option_idx(const char *name, const char *option_strings)
{ {
int n = index_in_strings(dhcp_option_strings, name); int n = index_in_strings(option_strings, name);
if (n >= 0) if (n >= 0)
return n; return n;
{ {
char buf[sizeof(dhcp_option_strings)]; char *buf, *d;
char *d = buf; const char *s;
const char *s = dhcp_option_strings;
while (s < dhcp_option_strings + sizeof(dhcp_option_strings) - 2) { s = option_strings;
while (*s)
s += strlen(s) + 1;
d = buf = xzalloc(s - option_strings);
s = option_strings;
while (!(*s == '\0' && s[1] == '\0')) {
*d++ = (*s == '\0' ? ' ' : *s); *d++ = (*s == '\0' ? ' ' : *s);
s++; s++;
} }
*d = '\0';
bb_error_msg_and_die("unknown option '%s', known options: %s", name, buf); bb_error_msg_and_die("unknown option '%s', known options: %s", name, buf);
} }
} }
@ -315,6 +322,7 @@ void FAST_FUNC udhcp_add_binary_option(struct dhcp_packet *packet, uint8_t *addo
optionptr[end + len] = DHCP_END; optionptr[end + len] = DHCP_END;
} }
#if ENABLE_UDHCPC || ENABLE_UDHCPD
/* Add an one to four byte option to a packet */ /* Add an one to four byte option to a packet */
void FAST_FUNC udhcp_add_simple_option(struct dhcp_packet *packet, uint8_t code, uint32_t data) void FAST_FUNC udhcp_add_simple_option(struct dhcp_packet *packet, uint8_t code, uint32_t data)
{ {
@ -338,6 +346,7 @@ void FAST_FUNC udhcp_add_simple_option(struct dhcp_packet *packet, uint8_t code,
bb_error_msg("can't add option 0x%02x", code); bb_error_msg("can't add option 0x%02x", code);
} }
#endif
/* Find option 'code' in opt_list */ /* Find option 'code' in opt_list */
struct option_set* FAST_FUNC udhcp_find_option(struct option_set *opt_list, uint8_t code) struct option_set* FAST_FUNC udhcp_find_option(struct option_set *opt_list, uint8_t code)
@ -451,7 +460,7 @@ static NOINLINE void attach_option(
free(allocated); free(allocated);
} }
int FAST_FUNC udhcp_str2optset(const char *const_str, void *arg) int FAST_FUNC udhcp_str2optset(const char *const_str, void *arg, const struct dhcp_optflag *optflags, const char *option_strings)
{ {
struct option_set **opt_list = arg; struct option_set **opt_list = arg;
char *opt, *val; char *opt, *val;
@ -478,7 +487,7 @@ int FAST_FUNC udhcp_str2optset(const char *const_str, void *arg)
bin_optflag.code = optcode; bin_optflag.code = optcode;
optflag = &bin_optflag; optflag = &bin_optflag;
} else { } else {
optflag = &dhcp_optflags[udhcp_option_idx(opt)]; optflag = &optflags[udhcp_option_idx(opt, option_strings)];
} }
retval = 0; retval = 0;

View File

@ -93,8 +93,10 @@ enum {
OPTION_BIN, OPTION_BIN,
OPTION_STATIC_ROUTES, OPTION_STATIC_ROUTES,
OPTION_6RD, OPTION_6RD,
#if ENABLE_FEATURE_UDHCP_RFC3397 #if ENABLE_FEATURE_UDHCP_RFC3397 || ENABLE_FEATURE_UDHCPC6_RFC3646 || ENABLE_FEATURE_UDHCPC6_RFC4704
OPTION_DNS_STRING, /* RFC1035 compressed domain name list */ OPTION_DNS_STRING, /* RFC1035 compressed domain name list */
#endif
#if ENABLE_FEATURE_UDHCP_RFC3397
OPTION_SIP_SERVERS, OPTION_SIP_SERVERS,
#endif #endif
@ -189,17 +191,21 @@ struct option_set {
struct option_set *next; struct option_set *next;
}; };
#if ENABLE_UDHCPC || ENABLE_UDHCPD
extern const struct dhcp_optflag dhcp_optflags[]; extern const struct dhcp_optflag dhcp_optflags[];
extern const char dhcp_option_strings[] ALIGN1; extern const char dhcp_option_strings[] ALIGN1;
#endif
extern const uint8_t dhcp_option_lengths[] ALIGN1; extern const uint8_t dhcp_option_lengths[] ALIGN1;
unsigned FAST_FUNC udhcp_option_idx(const char *name); unsigned FAST_FUNC udhcp_option_idx(const char *name, const char *option_strings);
uint8_t *udhcp_get_option(struct dhcp_packet *packet, int code) FAST_FUNC; uint8_t *udhcp_get_option(struct dhcp_packet *packet, int code) FAST_FUNC;
int udhcp_end_option(uint8_t *optionptr) FAST_FUNC; int udhcp_end_option(uint8_t *optionptr) FAST_FUNC;
void udhcp_add_binary_option(struct dhcp_packet *packet, uint8_t *addopt) FAST_FUNC; void udhcp_add_binary_option(struct dhcp_packet *packet, uint8_t *addopt) FAST_FUNC;
#if ENABLE_UDHCPC || ENABLE_UDHCPD
void udhcp_add_simple_option(struct dhcp_packet *packet, uint8_t code, uint32_t data) FAST_FUNC; void udhcp_add_simple_option(struct dhcp_packet *packet, uint8_t code, uint32_t data) FAST_FUNC;
#if ENABLE_FEATURE_UDHCP_RFC3397 #endif
#if ENABLE_FEATURE_UDHCP_RFC3397 || ENABLE_FEATURE_UDHCPC6_RFC3646 || ENABLE_FEATURE_UDHCPC6_RFC4704
char *dname_dec(const uint8_t *cstr, int clen, const char *pre) FAST_FUNC; char *dname_dec(const uint8_t *cstr, int clen, const char *pre) FAST_FUNC;
uint8_t *dname_enc(const uint8_t *cstr, int clen, const char *src, int *retlen) FAST_FUNC; uint8_t *dname_enc(const uint8_t *cstr, int clen, const char *src, int *retlen) FAST_FUNC;
#endif #endif
@ -284,7 +290,10 @@ void udhcp_dump_packet(struct dhcp_packet *packet) FAST_FUNC;
/* 2nd param is "uint32_t*" */ /* 2nd param is "uint32_t*" */
int FAST_FUNC udhcp_str2nip(const char *str, void *arg); int FAST_FUNC udhcp_str2nip(const char *str, void *arg);
/* 2nd param is "struct option_set**" */ /* 2nd param is "struct option_set**" */
int FAST_FUNC udhcp_str2optset(const char *str, void *arg); int FAST_FUNC udhcp_str2optset(const char *str,
void *arg,
const struct dhcp_optflag *optflags,
const char *option_strings);
void udhcp_init_header(struct dhcp_packet *packet, char type) FAST_FUNC; void udhcp_init_header(struct dhcp_packet *packet, char type) FAST_FUNC;

View File

@ -2,26 +2,49 @@
/* /*
* DHCPv6 client. * DHCPv6 client.
* *
* 2011-11. * WARNING: THIS CODE IS INCOMPLETE.
* WARNING: THIS CODE IS INCOMPLETE. IT IS NOWHERE NEAR
* TO BE READY FOR PRODUCTION USE.
* *
* Copyright (C) 2011 Denys Vlasenko. * Copyright (C) 2011-2017 Denys Vlasenko.
* *
* Licensed under GPLv2, see file LICENSE in this source tree. * Licensed under GPLv2, see file LICENSE in this source tree.
*/ */
//config:config UDHCPC6 //config:config UDHCPC6
//config: bool "udhcpc6 (DHCPv6 client, NOT READY)" //config: bool "udhcpc6 (DHCPv6 client, EXPERIMENTAL)"
//config: default n # not yet ready //config: default n # not yet ready
//config: depends on FEATURE_IPV6 //config: depends on FEATURE_IPV6
//config: help //config: help
//config: udhcpc6 is a DHCPv6 client //config: udhcpc6 is a DHCPv6 client
//config:
//config:config FEATURE_UDHCPC6_RFC3646
//config: bool "Support RFC 3646 (DNS server and search list)"
//config: default y
//config: depends on UDHCPC6
//config: help
//config: List of DNS servers and domain search list can be requested with
//config: "-O dns" and "-O search". If server gives these values,
//config: they will be set in environment variables "dns" and "search".
//config:
//config:config FEATURE_UDHCPC6_RFC4704
//config: bool "Support RFC 4704 (Client FQDN)"
//config: default y
//config: depends on UDHCPC6
//config: help
//config: You can request FQDN to be given by server using "-O fqdn".
//config:
//config:config FEATURE_UDHCPC6_RFC4833
//config: bool "Support RFC 4833 (Timezones)"
//config: default y
//config: depends on UDHCPC6
//config: help
//config: You can request POSIX timezone with "-O tz" and timezone name
//config: with "-O timezone".
//applet:IF_UDHCPC6(APPLET(udhcpc6, BB_DIR_USR_BIN, BB_SUID_DROP)) //applet:IF_UDHCPC6(APPLET(udhcpc6, BB_DIR_USR_BIN, BB_SUID_DROP))
//kbuild:lib-$(CONFIG_UDHCPC6) += d6_dhcpc.o d6_packet.o d6_socket.o common.o socket.o signalpipe.o //kbuild:lib-$(CONFIG_UDHCPC6) += d6_dhcpc.o d6_packet.o d6_socket.o common.o socket.o signalpipe.o
//kbuild:lib-$(CONFIG_FEATURE_UDHCPC6_RFC3646) += domain_codec.o
//kbuild:lib-$(CONFIG_FEATURE_UDHCPC6_RFC4704) += domain_codec.o
#include <syslog.h> #include <syslog.h>
/* Override ENABLE_FEATURE_PIDFILE - ifupdown needs our pidfile to always exist */ /* Override ENABLE_FEATURE_PIDFILE - ifupdown needs our pidfile to always exist */
@ -37,6 +60,34 @@
/* "struct client_config_t client_config" is in bb_common_bufsiz1 */ /* "struct client_config_t client_config" is in bb_common_bufsiz1 */
static const struct dhcp_optflag d6_optflags[] = {
#if ENABLE_FEATURE_UDHCPC6_RFC3646
{ OPTION_6RD | OPTION_LIST | OPTION_REQ, D6_OPT_DNS_SERVERS },
{ OPTION_DNS_STRING | OPTION_LIST | OPTION_REQ, D6_OPT_DOMAIN_LIST },
#endif
#if ENABLE_FEATURE_UDHCPC6_RFC4704
{ OPTION_DNS_STRING, D6_OPT_CLIENT_FQDN },
#endif
#if ENABLE_FEATURE_UDHCPC6_RFC4833
{ OPTION_STRING, D6_OPT_TZ_POSIX },
{ OPTION_STRING, D6_OPT_TZ_NAME },
#endif
{ 0, 0 }
};
/* Must match d6_optflags[] order */
static const char d6_option_strings[] ALIGN1 =
#if ENABLE_FEATURE_UDHCPC6_RFC3646
"dns" "\0" /* D6_OPT_DNS_SERVERS */
"search" "\0" /* D6_OPT_DOMAIN_LIST */
#endif
#if ENABLE_FEATURE_UDHCPC6_RFC4704
"fqdn" "\0" /* D6_OPT_CLIENT_FQDN */
#endif
#if ENABLE_FEATURE_UDHCPC6_RFC4833
"tz" "\0" /* D6_OPT_TZ_POSIX */
"timezone" "\0" /* D6_OPT_TZ_NAME */
#endif
"\0";
#if ENABLE_LONG_OPTS #if ENABLE_LONG_OPTS
static const char udhcpc6_longopts[] ALIGN1 = static const char udhcpc6_longopts[] ALIGN1 =
@ -88,14 +139,7 @@ enum {
IF_FEATURE_UDHCP_PORT( OPT_P = 1 << OPTBIT_P,) IF_FEATURE_UDHCP_PORT( OPT_P = 1 << OPTBIT_P,)
}; };
static const char opt_req[] = { #if ENABLE_FEATURE_UDHCPC6_RFC4704
(D6_OPT_ORO >> 8), (D6_OPT_ORO & 0xff),
0, 6,
(D6_OPT_DNS_SERVERS >> 8), (D6_OPT_DNS_SERVERS & 0xff),
(D6_OPT_DOMAIN_LIST >> 8), (D6_OPT_DOMAIN_LIST & 0xff),
(D6_OPT_CLIENT_FQDN >> 8), (D6_OPT_CLIENT_FQDN & 0xff),
};
static const char opt_fqdn_req[] = { static const char opt_fqdn_req[] = {
(D6_OPT_CLIENT_FQDN >> 8), (D6_OPT_CLIENT_FQDN & 0xff), (D6_OPT_CLIENT_FQDN >> 8), (D6_OPT_CLIENT_FQDN & 0xff),
0, 2, /* optlen */ 0, 2, /* optlen */
@ -105,6 +149,7 @@ static const char opt_fqdn_req[] = {
/* N=0: server SHOULD perform updates (PTR RR only in our case, since S=0) */ /* N=0: server SHOULD perform updates (PTR RR only in our case, since S=0) */
0 /* empty DNS-encoded name */ 0 /* empty DNS-encoded name */
}; };
#endif
/*** Utility functions ***/ /*** Utility functions ***/
@ -152,10 +197,12 @@ static char** new_env(void)
/* put all the parameters into the environment */ /* put all the parameters into the environment */
static void option_to_env(uint8_t *option, uint8_t *option_end) static void option_to_env(uint8_t *option, uint8_t *option_end)
{ {
char *dlist; #if ENABLE_FEATURE_UDHCPC6_RFC3646
int addrs, option_offset;
#endif
/* "length minus 4" */ /* "length minus 4" */
int len_m4 = option_end - option - 4; int len_m4 = option_end - option - 4;
int addrs, option_offset;
while (len_m4 >= 0) { while (len_m4 >= 0) {
uint32_t v32; uint32_t v32;
char ipv6str[sizeof("ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff")]; char ipv6str[sizeof("ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff")];
@ -237,7 +284,10 @@ static void option_to_env(uint8_t *option, uint8_t *option_end)
sprint_nip6(ipv6str, option + 4 + 4 + 1); sprint_nip6(ipv6str, option + 4 + 4 + 1);
*new_env() = xasprintf("ipv6prefix=%s/%u", ipv6str, (unsigned)(option[4 + 4])); *new_env() = xasprintf("ipv6prefix=%s/%u", ipv6str, (unsigned)(option[4 + 4]));
break; break;
case D6_OPT_DNS_SERVERS: #if ENABLE_FEATURE_UDHCPC6_RFC3646
case D6_OPT_DNS_SERVERS: {
char *dlist;
/* Make sure payload-size is a multiple of 16 */ /* Make sure payload-size is a multiple of 16 */
if ((option[3] & 0x0f) != 0) if ((option[3] & 0x0f) != 0)
break; break;
@ -259,13 +309,21 @@ static void option_to_env(uint8_t *option, uint8_t *option_end)
} }
break; break;
case D6_OPT_DOMAIN_LIST: }
case D6_OPT_DOMAIN_LIST: {
char *dlist;
dlist = dname_dec(option + 4, (option[2] << 8) | option[3], "search="); dlist = dname_dec(option + 4, (option[2] << 8) | option[3], "search=");
if (!dlist) if (!dlist)
break; break;
*new_env() = dlist; *new_env() = dlist;
break; break;
case D6_OPT_CLIENT_FQDN: }
#endif
#if ENABLE_FEATURE_UDHCPC6_RFC4704
case D6_OPT_CLIENT_FQDN: {
char *dlist;
if (option[3] == 0) if (option[3] == 0)
break; break;
/* Work around broken ISC DHCPD6. /* Work around broken ISC DHCPD6.
@ -284,6 +342,9 @@ static void option_to_env(uint8_t *option, uint8_t *option_end)
break; break;
*new_env() = dlist; *new_env() = dlist;
break; break;
}
#endif
#if ENABLE_FEATURE_UDHCPC6_RFC4833
/* RFC 4833 Timezones */ /* RFC 4833 Timezones */
case D6_OPT_TZ_POSIX: case D6_OPT_TZ_POSIX:
*new_env() = xasprintf("tz=%.*s", (int)option[3], (char*)option + 4); *new_env() = xasprintf("tz=%.*s", (int)option[3], (char*)option + 4);
@ -291,6 +352,7 @@ static void option_to_env(uint8_t *option, uint8_t *option_end)
case D6_OPT_TZ_NAME: case D6_OPT_TZ_NAME:
*new_env() = xasprintf("tz_name=%.*s", (int)option[3], (char*)option + 4); *new_env() = xasprintf("tz_name=%.*s", (int)option[3], (char*)option + 4);
break; break;
#endif
} }
len_m4 -= 4 + option[3]; len_m4 -= 4 + option[3];
option += 4 + option[3]; option += 4 + option[3];
@ -363,17 +425,33 @@ static uint8_t *init_d6_packet(struct d6_packet *packet, char type, uint32_t xid
static uint8_t *add_d6_client_options(uint8_t *ptr) static uint8_t *add_d6_client_options(uint8_t *ptr)
{ {
return ptr; uint8_t *start = ptr;
//uint8_t c; unsigned option;
//int i, end, len;
/* Add a "param req" option with the list of options we'd like to have ptr += 4;
* from stubborn DHCP servers. Pull the data from the struct in common.c. for (option = 1; option < 256; option++) {
* No bounds checking because it goes towards the head of the packet. */ if (client_config.opt_mask[option >> 3] & (1 << (option & 7))) {
//... ptr[0] = (option >> 8);
ptr[1] = option;
ptr += 2;
}
}
if ((ptr - start - 4) != 0) {
start[0] = (D6_OPT_ORO >> 8);
start[1] = D6_OPT_ORO;
start[2] = ((ptr - start - 4) >> 8);
start[3] = (ptr - start - 4);
} else
ptr = start;
#if ENABLE_FEATURE_UDHCPC6_RFC4704
ptr = mempcpy(ptr, &opt_fqdn_req, sizeof(opt_fqdn_req));
#endif
/* Add -x options if any */ /* Add -x options if any */
//... //...
return ptr;
} }
static int d6_mcast_from_client_config_ifindex(struct d6_packet *packet, uint8_t *end) static int d6_mcast_from_client_config_ifindex(struct d6_packet *packet, uint8_t *end)
@ -497,10 +575,6 @@ static NOINLINE int send_d6_discover(uint32_t xid, struct in6_addr *requested_ip
} }
opt_ptr = mempcpy(opt_ptr, client6_data.ia_na, len); opt_ptr = mempcpy(opt_ptr, client6_data.ia_na, len);
/* Request additional options */
opt_ptr = mempcpy(opt_ptr, &opt_req, sizeof(opt_req));
opt_ptr = mempcpy(opt_ptr, &opt_fqdn_req, sizeof(opt_fqdn_req));
/* Add options: /* Add options:
* "param req" option according to -O, options specified with -x * "param req" option according to -O, options specified with -x
*/ */
@ -554,10 +628,6 @@ static NOINLINE int send_d6_select(uint32_t xid)
/* IA NA (contains requested IP) */ /* IA NA (contains requested IP) */
opt_ptr = mempcpy(opt_ptr, client6_data.ia_na, client6_data.ia_na->len + 2+2); opt_ptr = mempcpy(opt_ptr, client6_data.ia_na, client6_data.ia_na->len + 2+2);
/* Request additional options */
opt_ptr = mempcpy(opt_ptr, &opt_req, sizeof(opt_req));
opt_ptr = mempcpy(opt_ptr, &opt_fqdn_req, sizeof(opt_fqdn_req));
/* Add options: /* Add options:
* "param req" option according to -O, options specified with -x * "param req" option according to -O, options specified with -x
*/ */
@ -1063,20 +1133,18 @@ int udhcpc6_main(int argc UNUSED_PARAM, char **argv)
char *optstr = llist_pop(&list_O); char *optstr = llist_pop(&list_O);
unsigned n = bb_strtou(optstr, NULL, 0); unsigned n = bb_strtou(optstr, NULL, 0);
if (errno || n > 254) { if (errno || n > 254) {
n = udhcp_option_idx(optstr); n = udhcp_option_idx(optstr, d6_option_strings);
n = dhcp_optflags[n].code; n = d6_optflags[n].code;
} }
client_config.opt_mask[n >> 3] |= 1 << (n & 7); client_config.opt_mask[n >> 3] |= 1 << (n & 7);
} }
if (!(opt & OPT_o)) { if (!(opt & OPT_o)) {
/*
unsigned i, n; unsigned i, n;
for (i = 0; (n = dhcp_optflags[i].code) != 0; i++) { for (i = 0; (n = d6_optflags[i].code) != 0; i++) {
if (dhcp_optflags[i].flags & OPTION_REQ) { if (d6_optflags[i].flags & OPTION_REQ) {
client_config.opt_mask[n >> 3] |= 1 << (n & 7); client_config.opt_mask[n >> 3] |= 1 << (n & 7);
} }
} }
*/
} }
while (list_x) { while (list_x) {
char *optstr = llist_pop(&list_x); char *optstr = llist_pop(&list_x);
@ -1085,7 +1153,7 @@ int udhcpc6_main(int argc UNUSED_PARAM, char **argv)
*colon = ' '; *colon = ' ';
/* now it looks similar to udhcpd's config file line: /* now it looks similar to udhcpd's config file line:
* "optname optval", using the common routine: */ * "optname optval", using the common routine: */
udhcp_str2optset(optstr, &client_config.options); udhcp_str2optset(optstr, &client_config.options, d6_optflags, d6_option_strings);
if (colon) if (colon)
*colon = ':'; /* restore it for NOMMU reexec */ *colon = ':'; /* restore it for NOMMU reexec */
} }

View File

@ -1346,7 +1346,7 @@ int udhcpc_main(int argc UNUSED_PARAM, char **argv)
char *optstr = llist_pop(&list_O); char *optstr = llist_pop(&list_O);
unsigned n = bb_strtou(optstr, NULL, 0); unsigned n = bb_strtou(optstr, NULL, 0);
if (errno || n > 254) { if (errno || n > 254) {
n = udhcp_option_idx(optstr); n = udhcp_option_idx(optstr, dhcp_option_strings);
n = dhcp_optflags[n].code; n = dhcp_optflags[n].code;
} }
client_config.opt_mask[n >> 3] |= 1 << (n & 7); client_config.opt_mask[n >> 3] |= 1 << (n & 7);
@ -1366,7 +1366,7 @@ int udhcpc_main(int argc UNUSED_PARAM, char **argv)
*colon = ' '; *colon = ' ';
/* now it looks similar to udhcpd's config file line: /* now it looks similar to udhcpd's config file line:
* "optname optval", using the common routine: */ * "optname optval", using the common routine: */
udhcp_str2optset(optstr, &client_config.options); udhcp_str2optset(optstr, &client_config.options, dhcp_optflags, dhcp_option_strings);
if (colon) if (colon)
*colon = ':'; /* restore it for NOMMU reexec */ *colon = ':'; /* restore it for NOMMU reexec */
} }

View File

@ -12,6 +12,7 @@ struct client_config_t {
IF_FEATURE_UDHCP_PORT(uint16_t port;) IF_FEATURE_UDHCP_PORT(uint16_t port;)
int ifindex; /* Index number of the interface to use */ int ifindex; /* Index number of the interface to use */
uint8_t opt_mask[256 / 8]; /* Bitmask of options to send (-O option) */ uint8_t opt_mask[256 / 8]; /* Bitmask of options to send (-O option) */
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ TODO: DHCPv6 has 16-bit option numbers
const char *interface; /* The name of the interface to use */ const char *interface; /* The name of the interface to use */
char *pidfile; /* Optionally store the process ID */ char *pidfile; /* Optionally store the process ID */
const char *script; /* User script to run at dhcp events */ const char *script; /* User script to run at dhcp events */

View File

@ -361,6 +361,10 @@ static int FAST_FUNC read_staticlease(const char *const_line, void *arg)
return 1; return 1;
} }
static int FAST_FUNC read_optset(const char *line, void *arg) {
return udhcp_str2optset(line, arg, dhcp_optflags, dhcp_option_strings);
}
struct config_keyword { struct config_keyword {
const char *keyword; const char *keyword;
int (*handler)(const char *line, void *var) FAST_FUNC; int (*handler)(const char *line, void *var) FAST_FUNC;
@ -387,8 +391,8 @@ static const struct config_keyword keywords[] = {
{"pidfile" , read_str , OFS(pidfile ), "/var/run/udhcpd.pid"}, {"pidfile" , read_str , OFS(pidfile ), "/var/run/udhcpd.pid"},
{"siaddr" , udhcp_str2nip , OFS(siaddr_nip ), "0.0.0.0"}, {"siaddr" , udhcp_str2nip , OFS(siaddr_nip ), "0.0.0.0"},
/* keywords with no defaults must be last! */ /* keywords with no defaults must be last! */
{"option" , udhcp_str2optset, OFS(options ), ""}, {"option" , read_optset , OFS(options ), ""},
{"opt" , udhcp_str2optset, OFS(options ), ""}, {"opt" , read_optset , OFS(options ), ""},
{"notify_file" , read_str , OFS(notify_file ), NULL}, {"notify_file" , read_str , OFS(notify_file ), NULL},
{"sname" , read_str , OFS(sname ), NULL}, {"sname" , read_str , OFS(sname ), NULL},
{"boot_file" , read_str , OFS(boot_file ), NULL}, {"boot_file" , read_str , OFS(boot_file ), NULL},