/* $OpenBSD: clparse.c,v 1.148 2017/10/23 13:31:35 krw Exp $ */ /* Parser for dhclient config and lease files. */ /* * Copyright (c) 1997 The Internet Software Consortium. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of The Internet Software Consortium nor the names * of its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE INTERNET SOFTWARE CONSORTIUM AND * CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE INTERNET SOFTWARE CONSORTIUM OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * This software has been written for the Internet Software Consortium * by Ted Lemon in cooperation with Vixie * Enterprises. To learn more about the Internet Software Consortium, * see ``http://www.vix.com/isc''. To learn more about Vixie * Enterprises, see ``http://www.vix.com''. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "dhcp.h" #include "dhcpd.h" #include "dhctoken.h" #include "log.h" void parse_client_statement(FILE *, char *, int); int parse_hex_octets(FILE *, unsigned int *, uint8_t **); int parse_option_list(FILE *, int *, uint8_t *); int parse_interface_declaration(FILE *, char *); int parse_client_lease_statement(FILE *, char *, struct client_lease **); void parse_client_lease_declaration(FILE *, struct client_lease *, char *); int parse_option_decl(FILE *, int *, struct option_data *); int parse_reject_statement(FILE *); void add_lease(struct client_lease_tq *, struct client_lease *); void add_lease(struct client_lease_tq *tq, struct client_lease *lease) { struct client_lease *lp, *nlp; if (lease == NULL) return; /* * The new lease will supersede a lease with the same ssid * AND the same Client Identifier AND the same * IP address. */ TAILQ_FOREACH_SAFE(lp, tq, next, nlp) { if (lp->ssid_len != lease->ssid_len) continue; if (memcmp(lp->ssid, lease->ssid, lp->ssid_len) != 0) continue; if ((lease->options[DHO_DHCP_CLIENT_IDENTIFIER].len != 0) && ((lp->options[DHO_DHCP_CLIENT_IDENTIFIER].len != lease->options[DHO_DHCP_CLIENT_IDENTIFIER].len) || memcmp(lp->options[DHO_DHCP_CLIENT_IDENTIFIER].data, lease->options[DHO_DHCP_CLIENT_IDENTIFIER].data, lp->options[DHO_DHCP_CLIENT_IDENTIFIER].len))) continue; if (lp->address.s_addr != lease->address.s_addr) continue; TAILQ_REMOVE(tq, lp, next); free_client_lease(lp); } TAILQ_INSERT_TAIL(tq, lease, next); } /* * client-conf-file :== client-declarations EOF * client-declarations :== * | client-declaration * | client-declarations client-declaration */ void read_client_conf(char *name) { FILE *cfile; int token; new_parse(path_dhclient_conf); TAILQ_INIT(&config->static_leases); TAILQ_INIT(&config->reject_list); /* Set some defaults. */ config->link_timeout = 10; /* secs before going daemon w/o link */ config->timeout = 30; /* secs to wait for an OFFER */ config->select_interval = 0; /* secs to wait for other OFFERs */ config->reboot_timeout = 1; /* secs before giving up on reboot */ config->retry_interval = 1; /* secs before asking for OFFER */ config->backoff_cutoff = 10; /* max secs between packet retries */ config->initial_interval = 1; /* secs before 1st retry */ config->requested_options [config->requested_option_count++] = DHO_SUBNET_MASK; config->requested_options [config->requested_option_count++] = DHO_BROADCAST_ADDRESS; config->requested_options [config->requested_option_count++] = DHO_TIME_OFFSET; /* RFC 3442 says CLASSLESS_STATIC_ROUTES must be before ROUTERS! */ config->requested_options [config->requested_option_count++] = DHO_CLASSLESS_STATIC_ROUTES; config->requested_options [config->requested_option_count++] = DHO_ROUTERS; config->requested_options [config->requested_option_count++] = DHO_DOMAIN_NAME; config->requested_options [config->requested_option_count++] = DHO_DOMAIN_SEARCH; config->requested_options [config->requested_option_count++] = DHO_DOMAIN_NAME_SERVERS; config->requested_options [config->requested_option_count++] = DHO_HOST_NAME; config->requested_options [config->requested_option_count++] = DHO_BOOTFILE_NAME; config->requested_options [config->requested_option_count++] = DHO_TFTP_SERVER; if ((cfile = fopen(path_dhclient_conf, "r")) != NULL) { for (;;) { token = peek_token(NULL, cfile); if (token == EOF) break; parse_client_statement(cfile, name, 0); } fclose(cfile); } } /* * lease-file :== client-lease-statements EOF * client-lease-statements :== * | client-lease-statements LEASE client-lease-statement */ void read_client_leases(char *name, struct client_lease_tq *tq) { struct client_lease *lp; FILE *cfile; int token; TAILQ_INIT(tq); if ((cfile = fopen(path_dhclient_db, "r")) == NULL) return; new_parse(path_dhclient_db); for (;;) { token = next_token(NULL, cfile); if (token == EOF) break; if (token != TOK_LEASE) { log_warnx("%s: expecting lease", log_procname); break; } if (parse_client_lease_statement(cfile, name, &lp) == 1) add_lease(tq, lp); } fclose(cfile); } /* * client-declaration :== * TOK_APPEND option-decl | * TOK_BACKOFF_CUTOFF number | * TOK_DEFAULT option-decl | * TOK_FILENAME string | * TOK_FIXED_ADDR ip-address | * TOK_IGNORE option-list | * TOK_INITIAL_INTERVAL number | * TOK_INTERFACE interface-declaration | * TOK_LEASE client-lease-statement | * TOK_LINK_TIMEOUT number | * TOK_NEXT_SERVER string | * TOK_PREPEND option-decl | * TOK_REBOOT number | * TOK_REJECT reject-statement | * TOK_REQUEST option-list | * TOK_REQUIRE option-list | * TOK_RETRY number | * TOK_SELECT_TIMEOUT number | * TOK_SEND option-decl | * TOK_SERVER_NAME string | * TOK_SUPERSEDE option-decl | * TOK_TIMEOUT number * * If nested == 1 then TOK_INTERFACE and TOK_LEASE are not allowed. */ void parse_client_statement(FILE *cfile, char *name, int nested) { struct client_lease *lp; char *val; int i, token; token = next_token(NULL, cfile); switch (token) { case TOK_APPEND: if (parse_option_decl(cfile, &i, config->defaults) == 1) { config->default_actions[i] = ACTION_APPEND; parse_semi(cfile); } break; case TOK_BACKOFF_CUTOFF: if (parse_lease_time(cfile, &config->backoff_cutoff) == 1) parse_semi(cfile); break; case TOK_DEFAULT: if (parse_option_decl(cfile, &i, config->defaults) == 1) { config->default_actions[i] = ACTION_DEFAULT; parse_semi(cfile); } break; case TOK_FILENAME: if (parse_string(cfile, NULL, &val) == 1) { free(config->filename); config->filename = val; parse_semi(cfile); } break; case TOK_FIXED_ADDR: if (parse_ip_addr(cfile, &config->address) == 1) parse_semi(cfile); break; case TOK_IGNORE: if (parse_option_list(cfile, &config->ignored_option_count, config->ignored_options) == 1) parse_semi(cfile); break; case TOK_INITIAL_INTERVAL: if (parse_lease_time(cfile, &config->initial_interval) == 1) parse_semi(cfile); break; case TOK_INTERFACE: if (nested == 1) { parse_warn("expecting statement."); skip_to_semi(cfile); } else if (parse_interface_declaration(cfile, name) == 1) ; break; case TOK_LEASE: if (nested == 1) { parse_warn("expecting statement."); skip_to_semi(cfile); } else if (parse_client_lease_statement(cfile, name, &lp) == 1) add_lease(&config->static_leases, lp); break; case TOK_LINK_TIMEOUT: if (parse_lease_time(cfile, &config->link_timeout) == 1) parse_semi(cfile); break; case TOK_NEXT_SERVER: if (parse_ip_addr(cfile, &config->next_server) == 1) parse_semi(cfile); break; case TOK_PREPEND: if (parse_option_decl(cfile, &i, config->defaults) == 1) { config->default_actions[i] = ACTION_PREPEND; parse_semi(cfile); } break; case TOK_REBOOT: if (parse_lease_time(cfile, &config->reboot_timeout) == 1) parse_semi(cfile); break; case TOK_REJECT: if (parse_reject_statement(cfile) == 1) parse_semi(cfile); break; case TOK_REQUEST: if (parse_option_list(cfile, &config->requested_option_count, config->requested_options) == 1) parse_semi(cfile); break; case TOK_REQUIRE: if (parse_option_list(cfile, &config->required_option_count, config->required_options) == 1) parse_semi(cfile); break; case TOK_RETRY: if (parse_lease_time(cfile, &config->retry_interval) == 1) parse_semi(cfile); break; case TOK_SELECT_TIMEOUT: if (parse_lease_time(cfile, &config->select_interval) == 1) parse_semi(cfile); break; case TOK_SEND: if (parse_option_decl(cfile, &i, config->send_options) == 1) parse_semi(cfile); break; case TOK_SERVER_NAME: if (parse_string(cfile, NULL, &val) == 1) { free(config->server_name); config->server_name = val; parse_semi(cfile); } break; case TOK_SUPERSEDE: if (parse_option_decl(cfile, &i, config->defaults) == 1) { config->default_actions[i] = ACTION_SUPERSEDE; parse_semi(cfile); } break; case TOK_TIMEOUT: if (parse_lease_time(cfile, &config->timeout) == 1) parse_semi(cfile); break; default: parse_warn("expecting statement."); if (token != ';') skip_to_semi(cfile); break; } } int parse_hex_octets(FILE *cfile, unsigned int *len, uint8_t **buf) { static uint8_t octets[1500]; char *val, *ep; unsigned long ulval; unsigned int i; int token; i = 0; do { token = next_token(&val, cfile); errno = 0; ulval = strtoul(val, &ep, 16); if ((val[0] == '\0' || *ep != '\0') || (errno == ERANGE && ulval == ULONG_MAX) || (ulval > UINT8_MAX)) break; octets[i++] = ulval; if (peek_token(NULL, cfile) == ';') { *buf = malloc(i); if (*buf == NULL) break; memcpy(*buf, octets, i); *len = i; return 1; } if (i == sizeof(octets)) break; token = next_token(NULL, cfile); } while (token == ':'); parse_warn("expecting colon delimited list of hex octets."); if (token != ';') skip_to_semi(cfile); return 0; } /* * option-list :== option_name | * option_list COMMA option_name */ int parse_option_list(FILE *cfile, int *count, uint8_t *optlist) { uint8_t list[DHO_COUNT]; unsigned int ix, j; int i; int token; char *val; /* Empty list of option names is allowed, to re-init optlist. */ if (peek_token(NULL, cfile) == ';') { memset(optlist, DHO_PAD, sizeof(list)); *count = 0; return 1; } memset(list, DHO_PAD, sizeof(list)); ix = 0; do { /* Next token must be an option name. */ token = next_token(&val, cfile); i = name_to_code(val); if (i == DHO_END) break; /* Avoid storing duplicate options in the list. */ for (j = 0; j < ix && list[j] != i; j++) ; if (j == ix) list[ix++] = i; if (peek_token(NULL, cfile) == ';') { memcpy(optlist, list, sizeof(list)); *count = ix; return 1; } token = next_token(NULL, cfile); } while (token == ','); parse_warn("expecting comma delimited list of option names."); if (token != ';') skip_to_semi(cfile); return 0; } /* * interface-declaration :== * INTERFACE string LBRACE client-declarations RBRACE */ int parse_interface_declaration(FILE *cfile, char *name) { char *val; int token; token = next_token(&val, cfile); if (token != TOK_STRING) { parse_warn("expecting string."); if (token != ';') skip_to_semi(cfile); return 0; } if (strcmp(name, val) != 0) { skip_to_semi(cfile); return 1; } token = next_token(&val, cfile); if (token != '{') { parse_warn("expecting '{'."); if (token != ';') skip_to_semi(cfile); return 0; } for (;;) { token = peek_token(&val, cfile); if (token == EOF) { parse_warn("unterminated interface declaration."); return 0; } if (token == '}') { token = next_token(NULL, cfile); return 1; } parse_client_statement(cfile, name, 1); } return 0; } /* * client-lease-statement :== * RBRACE client-lease-declarations LBRACE * * client-lease-declarations :== * | * client-lease-declaration | * client-lease-declarations client-lease-declaration */ int parse_client_lease_statement(FILE *cfile, char *name, struct client_lease **lp) { struct client_lease *lease; int token; token = next_token(NULL, cfile); if (token != '{') { parse_warn("expecting '{'."); if (token != ';') skip_to_semi(cfile); return 0; } lease = calloc(1, sizeof(*lease)); if (lease == NULL) fatal("lease"); for (;;) { token = peek_token(NULL, cfile); if (token == EOF) { parse_warn("unterminated lease declaration."); free_client_lease(lease); break; } if (token == '}') { token = next_token(NULL, cfile); if (lease->interface != NULL && strcmp(name, lease->interface) == 0) *lp = lease; else { *lp = NULL; free_client_lease(lease); } return 1; } parse_client_lease_declaration(cfile, lease, name); } return 0; } /* * client-lease-declaration :== * BOOTP | * EXPIRE time-decl | * FILENAME string | * FIXED_ADDR ip_address | * INTERFACE string | * NEXT_SERVER string | * OPTION option-decl | * REBIND time-decl | * RENEW time-decl | * SERVER_NAME string | * SSID string */ void parse_client_lease_declaration(FILE *cfile, struct client_lease *lease, char *name) { char *val; unsigned int len; int i, token; token = next_token(&val, cfile); switch (token) { case TOK_BOOTP: /* 'bootp' is just a comment. See BOOTP_LEASE(). */ break; case TOK_EXPIRE: if (parse_date(cfile, &lease->expiry) == 0) return; break; case TOK_FILENAME: if (parse_string(cfile, NULL, &val) == 0) return; free(lease->filename); lease->filename = val; break; case TOK_FIXED_ADDR: if (parse_ip_addr(cfile, &lease->address) == 0) return; break; case TOK_INTERFACE: if (parse_string(cfile, NULL, &val) == 0) return; free(lease->interface); lease->interface = val; break; case TOK_NEXT_SERVER: if (parse_ip_addr(cfile, &lease->next_server) == 0) return; break; case TOK_OPTION: if (parse_option_decl(cfile, &i, lease->options) == 0) return; break; case TOK_REBIND: if (parse_date(cfile, &lease->rebind) == 0) return; break; case TOK_RENEW: if (parse_date(cfile, &lease->renewal) == 0) return; break; case TOK_SERVER_NAME: if (parse_string(cfile, NULL, &val) == 0) return; free(lease->server_name); lease->server_name = val; break; case TOK_SSID: if (parse_string(cfile, &len, &val) == 0) return; if (len > sizeof(lease->ssid)) { free(val); parse_warn("ssid > 32 bytes"); skip_to_semi(cfile); return; } memset(lease->ssid, 0, sizeof(lease->ssid)); memcpy(lease->ssid, val, len); free(val); lease->ssid_len = len; break; default: parse_warn("expecting lease declaration."); if (token != ';') skip_to_semi(cfile); return; } parse_semi(cfile); } int parse_option_decl(FILE *cfile, int *code, struct option_data *options) { uint8_t hunkbuf[1024], cidr[5], buf[4]; struct in_addr ip_addr; uint8_t *dp; char *fmt, *val; unsigned int hunkix = 0; int i, freedp, len, token; int nul_term = 0; token = next_token(&val, cfile); i = name_to_code(val); if (i == DHO_END) { parse_warn("expecting option name."); skip_to_semi(cfile); return 0; } /* Parse the option data. */ do { for (fmt = code_to_format(i); *fmt != '\0'; fmt++) { if (*fmt == 'A') break; freedp = 0; switch (*fmt) { case 'X': if (peek_token(NULL, cfile) == TOK_STRING) { if (parse_string(cfile, &len, (char **)&dp) == 0) return 0; } else if (parse_hex_octets(cfile, &len, &dp) == 0) return 0; freedp = 1; break; case 't': /* Text string. */ if (parse_string(cfile, &len, (char **)&dp) == 0) return 0; freedp = 1; break; case 'I': /* IP address. */ if (parse_ip_addr(cfile, &ip_addr) == 0) return 0; len = sizeof(ip_addr); dp = (uint8_t *)&ip_addr; break; case 'l': /* Signed 32-bit integer. */ if (parse_decimal(cfile, buf, 'l') == 0) return 0; len = 4; dp = buf; break; case 'L': /* Unsigned 32-bit integer. */ if (parse_decimal(cfile, buf, 'L') == 0) return 0; len = 4; dp = buf; break; case 'S': /* Unsigned 16-bit integer. */ if (parse_decimal(cfile, buf, 'S') == 0) return 0; len = 2; dp = buf; break; case 'B': /* Unsigned 8-bit integer. */ if (parse_decimal(cfile, buf, 'B') == 0) return 0; len = 1; dp = buf; break; case 'f': /* Boolean flag. */ if (parse_boolean(cfile, buf) == 0) return 0; len = 1; dp = buf; break; case 'C': if (parse_cidr(cfile, cidr) == 0) return 0; len = 1 + (cidr[0] + 7) / 8; dp = cidr; break; default: log_warnx("%s: bad format %c in " "parse_option_param", log_procname, *fmt); skip_to_semi(cfile); return 0; } if (dp != NULL && len > 0) { if (hunkix + len > sizeof(hunkbuf)) { if (freedp == 1) free(dp); parse_warn("option data buffer " "overflow"); skip_to_semi(cfile); return 0; } memcpy(&hunkbuf[hunkix], dp, len); hunkix += len; if (freedp == 1) free(dp); } } token = peek_token(NULL, cfile); if (*fmt == 'A' && token == ',') token = next_token(NULL, cfile); } while (*fmt == 'A' && token == ','); free(options[i].data); options[i].data = malloc(hunkix + nul_term); if (options[i].data == NULL) fatal("option data"); memcpy(options[i].data, hunkbuf, hunkix + nul_term); options[i].len = hunkix; *code = i; return 1; } int parse_reject_statement(FILE *cfile) { struct in_addr addr; struct reject_elem *elem; if (parse_ip_addr(cfile, &addr) == 0) return 0; TAILQ_FOREACH(elem, &config->reject_list, next) { if (elem->addr.s_addr == addr.s_addr) return 1; } elem = malloc(sizeof(*elem)); if (elem == NULL) fatal("reject address"); elem->addr = addr; TAILQ_INSERT_TAIL(&config->reject_list, elem, next); return 1; }