#include <stdio.h> #include <string.h> #include <core.h> #include <sys/cpu.h> #include <lwip/opt.h> /* DNS_MAX_SERVERS */ #include <dprintf.h> #include "pxe.h" char LocalDomain[256]; int over_load; uint8_t uuid_type; uint8_t uuid[16]; static void subnet_mask(const void *data, int opt_len) { if (opt_len != 4) return; IPInfo.netmask = *(const uint32_t *)data; } static void router(const void *data, int opt_len) { if (opt_len != 4) return; IPInfo.gateway = *(const uint32_t *)data; } static void dns_servers(const void *data, int opt_len) { const uint32_t *dp = data; int num = 0; while (num < DNS_MAX_SERVERS) { uint32_t ip; if (opt_len < 4) break; opt_len -= 4; ip = *dp++; if (ip_ok(ip)) dns_server[num++] = ip; } while (num < DNS_MAX_SERVERS) dns_server[num++] = 0; } static void local_domain(const void *data, int opt_len) { memcpy(LocalDomain, data, opt_len); LocalDomain[opt_len] = 0; } static void vendor_encaps(const void *data, int opt_len) { /* Only recognize PXELINUX options */ parse_dhcp_options(data, opt_len, 208); } static void option_overload(const void *data, int opt_len) { if (opt_len != 1) return; over_load = *(uint8_t *)data; } static void server(const void *data, int opt_len) { uint32_t ip; if (opt_len != 4) return; if (IPInfo.serverip) return; ip = *(uint32_t *)data; if (ip_ok(ip)) IPInfo.serverip = ip; } static void client_identifier(const void *data, int opt_len) { if (opt_len > MAC_MAX || opt_len < 2 || MAC_len != (opt_len >> 8) || *(uint8_t *)data != MAC_type) return; opt_len --; MAC_len = opt_len & 0xff; memcpy(MAC, data+1, opt_len); MAC[opt_len] = 0; } static void bootfile_name(const void *data, int opt_len) { memcpy(boot_file, data, opt_len); boot_file[opt_len] = 0; } static void uuid_client_identifier(const void *data, int opt_len) { int type = *(const uint8_t *)data; if (opt_len != 17 || type != 0 || have_uuid) return; have_uuid = true; uuid_type = type; memcpy(uuid, data+1, 16); } static void pxelinux_configfile(const void *data, int opt_len) { DHCPMagic |= 2; memcpy(ConfigName, data, opt_len); ConfigName[opt_len] = 0; } static void pxelinux_pathprefix(const void *data, int opt_len) { DHCPMagic |= 4; memcpy(path_prefix, data, opt_len); path_prefix[opt_len] = 0; } static void pxelinux_reboottime(const void *data, int opt_len) { if (opt_len != 4) return; RebootTime = ntohl(*(const uint32_t *)data); DHCPMagic |= 8; /* Got reboot time */ } struct dhcp_options { int opt_num; void (*fun)(const void *, int); }; static const struct dhcp_options dhcp_opts[] = { {1, subnet_mask}, {3, router}, {6, dns_servers}, {15, local_domain}, {43, vendor_encaps}, {52, option_overload}, {54, server}, {61, client_identifier}, {67, bootfile_name}, {97, uuid_client_identifier}, {209, pxelinux_configfile}, {210, pxelinux_pathprefix}, {211, pxelinux_reboottime} }; /* * Parse a sequence of DHCP options, pointed to by _option_; * -- some DHCP servers leave option fields unterminated * in violation of the spec. * * filter contains the minimum value for the option to recognize * -- this is used to restrict parsing to PXELINUX-specific options only. */ void parse_dhcp_options(const void *option, int size, uint8_t opt_filter) { int opt_num; int opt_len; const int opt_entries = sizeof(dhcp_opts) / sizeof(dhcp_opts[0]); int i = 0; const uint8_t *p = option; const struct dhcp_options *opt; /* The only 1-byte options are 00 and FF, neither of which matter */ while (size >= 2) { opt_num = *p++; size--; if (opt_num == 0) continue; if (opt_num == 0xff) break; /* Anything else will have a length field */ opt_len = *p++; /* c <- option lenght */ size -= opt_len + 1; if (size < 0) break; dprintf("DHCP: option %d, len %d\n", opt_num, opt_len); if (opt_num >= opt_filter) { opt = dhcp_opts; for (i = 0; i < opt_entries; i++) { if (opt_num == opt->opt_num) { opt->fun(p, opt_len); break; } opt++; } } /* parse next */ p += opt_len; } } /* * parse_dhcp * * Parse a DHCP packet. This includes dealing with "overloaded" * option fields (see RFC 2132, section 9.3) * * This should fill in the following global variables, if the * information is present: * * MyIP - client IP address * server_ip - boot server IP address * net_mask - network mask * gate_way - default gateway router IP * boot_file - boot file name * DNSServers - DNS server IPs * LocalDomain - Local domain name * MAC_len, MAC - Client identifier, if MAC_len == 0 * */ void parse_dhcp(const void *pkt, size_t pkt_len) { const struct bootp_t *dhcp = (const struct bootp_t *)pkt; int opt_len; IPInfo.ipver = 4; /* This is IPv4 only for now... */ over_load = 0; if (ip_ok(dhcp->yip)) IPInfo.myip = dhcp->yip; if (ip_ok(dhcp->sip)) IPInfo.serverip = dhcp->sip; opt_len = (char *)dhcp + pkt_len - (char *)&dhcp->options; if (opt_len && (dhcp->option_magic == BOOTP_OPTION_MAGIC)) parse_dhcp_options(&dhcp->options, opt_len, 0); if (over_load & 1) parse_dhcp_options(&dhcp->bootfile, 128, 0); else if (dhcp->bootfile[0]) strcpy(boot_file, dhcp->bootfile); if (over_load & 2) parse_dhcp_options(dhcp->sname, 64, 0); }