2022-02-06 20:05:29 -05:00
|
|
|
// Copyright 2004-2018 Nicholas J. Kain <njkain at gmail dot com>
|
|
|
|
// SPDX-License-Identifier: MIT
|
2010-11-12 04:02:18 -05:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2011-09-02 02:12:51 -04:00
|
|
|
#include <arpa/inet.h>
|
2014-03-30 17:02:48 -04:00
|
|
|
#include "nk/log.h"
|
2018-02-09 02:39:46 -05:00
|
|
|
#include <limits.h>
|
2010-11-12 04:02:18 -05:00
|
|
|
|
|
|
|
#include "options.h"
|
2011-03-30 07:05:00 -04:00
|
|
|
|
2022-08-10 11:32:30 -04:00
|
|
|
static int do_overload_value(const uint8_t *buf, size_t blen, int overload)
|
2010-11-12 04:02:18 -05:00
|
|
|
{
|
2022-08-10 11:32:30 -04:00
|
|
|
size_t i = 0;
|
2011-07-27 07:39:45 -04:00
|
|
|
while (i < blen) {
|
|
|
|
if (buf[i] == DCODE_PADDING) {
|
|
|
|
++i;
|
2011-03-30 16:35:23 -04:00
|
|
|
continue;
|
|
|
|
}
|
2011-07-27 07:39:45 -04:00
|
|
|
if (buf[i] == DCODE_END)
|
|
|
|
break;
|
2022-08-10 11:32:30 -04:00
|
|
|
if (i + 2 >= blen)
|
2011-07-27 07:39:45 -04:00
|
|
|
break;
|
|
|
|
if (buf[i] == DCODE_OVERLOAD) {
|
|
|
|
if (buf[i+1] == 1) {
|
|
|
|
overload |= buf[i+2];
|
|
|
|
i += 3;
|
|
|
|
continue;
|
|
|
|
}
|
2011-03-30 16:35:23 -04:00
|
|
|
}
|
2011-07-27 07:39:45 -04:00
|
|
|
i += buf[i+1] + 2;
|
|
|
|
}
|
|
|
|
return overload;
|
|
|
|
}
|
2011-03-30 16:35:23 -04:00
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
static int overload_value(const struct dhcpmsg * const packet)
|
2011-07-27 07:39:45 -04:00
|
|
|
{
|
|
|
|
int ol = do_overload_value(packet->options, sizeof packet->options, 0);
|
|
|
|
if (ol & 1 && ol & 2)
|
|
|
|
return ol;
|
|
|
|
if (ol & 1) {
|
|
|
|
ol |= do_overload_value(packet->file, sizeof packet->file, ol);
|
|
|
|
return ol;
|
|
|
|
}
|
|
|
|
if (ol & 2) {
|
|
|
|
ol |= do_overload_value(packet->sname, sizeof packet->sname, ol);
|
|
|
|
return ol;
|
|
|
|
}
|
|
|
|
return ol; // ol == 0
|
|
|
|
}
|
2011-03-30 16:35:23 -04:00
|
|
|
|
2018-02-09 02:39:46 -05:00
|
|
|
static void do_get_dhcp_opt(const uint8_t *sbuf, size_t slen, uint8_t code,
|
|
|
|
uint8_t *dbuf, size_t dlen, size_t *didx)
|
2011-07-27 07:39:45 -04:00
|
|
|
{
|
2018-02-09 02:39:46 -05:00
|
|
|
size_t i = 0;
|
2011-07-27 07:39:45 -04:00
|
|
|
while (i < slen) {
|
|
|
|
if (sbuf[i] == DCODE_PADDING) {
|
|
|
|
++i;
|
|
|
|
continue;
|
2011-03-30 16:35:23 -04:00
|
|
|
}
|
2011-07-27 07:39:45 -04:00
|
|
|
if (sbuf[i] == DCODE_END)
|
|
|
|
break;
|
|
|
|
if (i >= slen - 2)
|
|
|
|
break;
|
2018-02-09 02:39:46 -05:00
|
|
|
size_t soptsiz = sbuf[i+1];
|
2011-07-27 07:39:45 -04:00
|
|
|
if (sbuf[i] == code) {
|
2018-02-09 02:39:46 -05:00
|
|
|
if (dlen < soptsiz + *didx)
|
2014-05-10 21:32:15 -04:00
|
|
|
return;
|
2018-02-09 02:39:46 -05:00
|
|
|
if (slen < soptsiz + i + 2)
|
2014-05-10 21:32:15 -04:00
|
|
|
return;
|
|
|
|
memcpy(dbuf + *didx, sbuf+i+2, soptsiz);
|
|
|
|
*didx += soptsiz;
|
2011-03-30 16:35:23 -04:00
|
|
|
}
|
2014-03-12 13:30:55 -04:00
|
|
|
i += soptsiz + 2;
|
2011-03-30 16:35:23 -04:00
|
|
|
}
|
2010-11-12 04:02:18 -05:00
|
|
|
}
|
|
|
|
|
2018-02-09 02:39:46 -05:00
|
|
|
size_t get_dhcp_opt(const struct dhcpmsg * const packet, uint8_t code,
|
|
|
|
uint8_t *dbuf, size_t dlen)
|
2011-03-30 09:35:17 -04:00
|
|
|
{
|
2011-07-27 07:39:45 -04:00
|
|
|
int ol = overload_value(packet);
|
2018-02-09 02:39:46 -05:00
|
|
|
size_t didx = 0;
|
2014-05-10 21:32:15 -04:00
|
|
|
do_get_dhcp_opt(packet->options, sizeof packet->options, code,
|
|
|
|
dbuf, dlen, &didx);
|
2011-07-27 07:39:45 -04:00
|
|
|
if (ol & 1)
|
2014-05-10 21:32:15 -04:00
|
|
|
do_get_dhcp_opt(packet->file, sizeof packet->file, code,
|
|
|
|
dbuf, dlen, &didx);
|
2011-07-27 07:39:45 -04:00
|
|
|
if (ol & 2)
|
2014-05-10 21:32:15 -04:00
|
|
|
do_get_dhcp_opt(packet->sname, sizeof packet->sname, code,
|
|
|
|
dbuf, dlen, &didx);
|
2011-07-27 07:39:45 -04:00
|
|
|
return didx;
|
2011-03-30 09:35:17 -04:00
|
|
|
}
|
|
|
|
|
2011-06-27 15:19:54 -04:00
|
|
|
// return the position of the 'end' option
|
2015-02-14 02:14:30 -05:00
|
|
|
ssize_t get_end_option_idx(const struct dhcpmsg * const packet)
|
2010-11-12 04:02:18 -05:00
|
|
|
{
|
2011-06-27 15:19:54 -04:00
|
|
|
for (size_t i = 0; i < sizeof packet->options; ++i) {
|
2011-07-25 03:11:47 -04:00
|
|
|
if (packet->options[i] == DCODE_END)
|
2018-02-09 02:39:46 -05:00
|
|
|
return (ssize_t)i;
|
2011-07-25 03:11:47 -04:00
|
|
|
if (packet->options[i] == DCODE_PADDING)
|
2011-03-30 16:35:23 -04:00
|
|
|
continue;
|
2014-03-18 01:38:58 -04:00
|
|
|
if (i + 1 >= sizeof packet->options)
|
|
|
|
break;
|
2018-02-09 02:39:46 -05:00
|
|
|
i += (size_t)packet->options[i+1] + 1;
|
2011-03-30 16:35:23 -04:00
|
|
|
}
|
2020-11-24 21:02:51 -05:00
|
|
|
log_line("get_end_option_idx: Did not find DCODE_END marker.");
|
2011-03-30 16:35:23 -04:00
|
|
|
return -1;
|
2010-11-12 04:02:18 -05:00
|
|
|
}
|
|
|
|
|
2011-07-26 06:43:12 -04:00
|
|
|
static inline size_t sizeof_option_str(uint8_t code, size_t datalen)
|
2011-07-26 06:24:30 -04:00
|
|
|
{
|
|
|
|
if (code == DCODE_PADDING || code == DCODE_END)
|
|
|
|
return 1;
|
|
|
|
return 2 + datalen;
|
|
|
|
}
|
|
|
|
|
2011-07-26 06:43:12 -04:00
|
|
|
// Add a raw data string to the options. It will take a binary string suitable
|
|
|
|
// for use with eg memcpy() and will append it to the options[] field of
|
|
|
|
// a dhcp packet with the requested option code and proper length specifier.
|
2015-02-14 02:14:30 -05:00
|
|
|
size_t add_option_string(struct dhcpmsg *packet, uint8_t code,
|
|
|
|
const char * const str, size_t slen)
|
2010-11-12 04:02:18 -05:00
|
|
|
{
|
2011-07-26 06:43:12 -04:00
|
|
|
size_t len = sizeof_option_str(code, slen);
|
2011-07-02 03:48:08 -04:00
|
|
|
if (slen > 255 || len != slen + 2) {
|
2020-11-24 21:02:51 -05:00
|
|
|
log_line("add_option_string: Length checks failed.");
|
2011-07-02 03:48:08 -04:00
|
|
|
return 0;
|
|
|
|
}
|
2011-03-30 16:35:23 -04:00
|
|
|
|
2011-07-02 04:45:11 -04:00
|
|
|
ssize_t end = get_end_option_idx(packet);
|
2014-03-10 19:00:08 -04:00
|
|
|
if (end < 0) {
|
2020-11-24 21:02:51 -05:00
|
|
|
log_line("add_option_string: Buffer has no DCODE_END marker.");
|
2011-03-30 16:35:23 -04:00
|
|
|
return 0;
|
|
|
|
}
|
2018-02-09 02:39:46 -05:00
|
|
|
if ((size_t)end + len >= sizeof packet->options) {
|
2020-11-24 21:02:51 -05:00
|
|
|
log_line("add_option_string: No space for option 0x%02x.", code);
|
2011-03-30 16:35:23 -04:00
|
|
|
return 0;
|
|
|
|
}
|
2011-07-02 03:48:08 -04:00
|
|
|
packet->options[end] = code;
|
|
|
|
packet->options[end+1] = slen;
|
|
|
|
memcpy(packet->options + end + 2, str, slen);
|
2018-02-09 02:39:46 -05:00
|
|
|
packet->options[(size_t)end+len] = DCODE_END;
|
2011-07-02 03:48:08 -04:00
|
|
|
return len;
|
2010-11-12 04:02:18 -05:00
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
static ssize_t add_option_check(struct dhcpmsg *packet, uint8_t code,
|
2011-07-02 04:45:11 -04:00
|
|
|
uint8_t rlen)
|
2010-11-12 04:02:18 -05:00
|
|
|
{
|
2011-07-02 04:45:11 -04:00
|
|
|
ssize_t end = get_end_option_idx(packet);
|
2014-03-10 19:00:08 -04:00
|
|
|
if (end < 0) {
|
2020-11-24 21:02:51 -05:00
|
|
|
log_line("add_u%01u_option: Buffer has no DCODE_END marker.", rlen*8);
|
2011-07-02 04:45:11 -04:00
|
|
|
return -1;
|
2011-03-30 16:35:23 -04:00
|
|
|
}
|
2014-03-10 19:00:08 -04:00
|
|
|
if ((size_t)end + 2 + rlen >= sizeof packet->options) {
|
2020-11-24 21:02:51 -05:00
|
|
|
log_line("add_u%01u_option: No space for option 0x%02x.",
|
|
|
|
rlen*8, code);
|
2011-07-02 04:45:11 -04:00
|
|
|
return -1;
|
2011-07-02 03:48:08 -04:00
|
|
|
}
|
2011-07-02 04:45:11 -04:00
|
|
|
return end;
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
static size_t add_u8_option(struct dhcpmsg *packet, uint8_t code, uint8_t data)
|
2011-07-02 04:45:11 -04:00
|
|
|
{
|
|
|
|
ssize_t end = add_option_check(packet, code, 1);
|
|
|
|
if (end < 0)
|
|
|
|
return 0;
|
|
|
|
packet->options[end] = code;
|
|
|
|
packet->options[end+1] = 1;
|
|
|
|
packet->options[end+2] = data;
|
2011-07-25 03:11:47 -04:00
|
|
|
packet->options[end+3] = DCODE_END;
|
2011-07-02 04:45:11 -04:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
2014-03-18 01:51:58 -04:00
|
|
|
#ifndef NDHS_BUILD
|
2011-07-02 04:45:11 -04:00
|
|
|
// Data should be in network byte order.
|
2015-02-14 02:14:30 -05:00
|
|
|
static size_t add_u16_option(struct dhcpmsg *packet, uint8_t code,
|
2011-07-25 23:34:32 -04:00
|
|
|
uint16_t data)
|
2011-07-02 04:45:11 -04:00
|
|
|
{
|
|
|
|
ssize_t end = add_option_check(packet, code, 2);
|
|
|
|
if (end < 0)
|
|
|
|
return 0;
|
|
|
|
uint8_t *dp = (uint8_t *)&data;
|
|
|
|
packet->options[end] = code;
|
|
|
|
packet->options[end+1] = 2;
|
|
|
|
packet->options[end+2] = dp[0];
|
|
|
|
packet->options[end+3] = dp[1];
|
2011-07-25 03:11:47 -04:00
|
|
|
packet->options[end+4] = DCODE_END;
|
2011-07-02 04:45:11 -04:00
|
|
|
return 4;
|
|
|
|
}
|
2014-03-18 01:51:58 -04:00
|
|
|
#endif
|
2011-07-02 04:45:11 -04:00
|
|
|
|
|
|
|
// Data should be in network byte order.
|
2015-02-14 02:14:30 -05:00
|
|
|
size_t add_u32_option(struct dhcpmsg *packet, uint8_t code, uint32_t data)
|
2011-07-02 04:45:11 -04:00
|
|
|
{
|
|
|
|
ssize_t end = add_option_check(packet, code, 4);
|
|
|
|
if (end < 0)
|
|
|
|
return 0;
|
|
|
|
uint8_t *dp = (uint8_t *)&data;
|
2011-07-02 03:48:08 -04:00
|
|
|
packet->options[end] = code;
|
2011-07-02 04:45:11 -04:00
|
|
|
packet->options[end+1] = 4;
|
|
|
|
packet->options[end+2] = dp[0];
|
|
|
|
packet->options[end+3] = dp[1];
|
|
|
|
packet->options[end+4] = dp[2];
|
|
|
|
packet->options[end+5] = dp[3];
|
2011-07-25 03:11:47 -04:00
|
|
|
packet->options[end+6] = DCODE_END;
|
2011-07-02 04:45:11 -04:00
|
|
|
return 6;
|
2010-11-12 04:02:18 -05:00
|
|
|
}
|
|
|
|
|
2011-07-26 06:43:12 -04:00
|
|
|
// Add a parameter request list for stubborn DHCP servers
|
2015-02-14 02:14:30 -05:00
|
|
|
size_t add_option_request_list(struct dhcpmsg *packet)
|
2011-03-30 07:26:42 -04:00
|
|
|
{
|
2018-10-26 13:07:19 -04:00
|
|
|
static const char reqdata[] = {
|
2011-07-26 01:58:05 -04:00
|
|
|
DCODE_SUBNET, DCODE_ROUTER, DCODE_DNS, DCODE_HOSTNAME, DCODE_DOMAIN,
|
|
|
|
DCODE_BROADCAST,
|
|
|
|
};
|
|
|
|
return add_option_string(packet, DCODE_PARAM_REQ,
|
2018-10-26 13:07:19 -04:00
|
|
|
reqdata, sizeof reqdata);
|
2011-07-01 11:37:13 -04:00
|
|
|
}
|
|
|
|
|
2014-03-19 06:00:45 -04:00
|
|
|
#ifdef NDHS_BUILD
|
2015-02-14 02:14:30 -05:00
|
|
|
size_t add_option_domain_name(struct dhcpmsg *packet, const char * const dom,
|
|
|
|
size_t domlen)
|
2011-09-02 02:12:51 -04:00
|
|
|
{
|
|
|
|
return add_option_string(packet, DCODE_DOMAIN, dom, domlen);
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_subnet_mask(struct dhcpmsg *packet, uint32_t subnet)
|
2011-09-02 02:12:51 -04:00
|
|
|
{
|
|
|
|
add_u32_option(packet, DCODE_SUBNET, subnet);
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_broadcast(struct dhcpmsg *packet, uint32_t bc)
|
2011-09-02 02:12:51 -04:00
|
|
|
{
|
|
|
|
add_u32_option(packet, DCODE_BROADCAST, bc);
|
|
|
|
}
|
2014-03-19 06:00:45 -04:00
|
|
|
#endif
|
2011-09-02 02:12:51 -04:00
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_msgtype(struct dhcpmsg *packet, uint8_t type)
|
2011-07-25 23:34:32 -04:00
|
|
|
{
|
|
|
|
add_u8_option(packet, DCODE_MSGTYPE, type);
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_reqip(struct dhcpmsg *packet, uint32_t ip)
|
2011-07-25 23:34:32 -04:00
|
|
|
{
|
|
|
|
add_u32_option(packet, DCODE_REQIP, ip);
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_serverid(struct dhcpmsg *packet, uint32_t sid)
|
2011-07-25 23:34:32 -04:00
|
|
|
{
|
2011-09-02 02:12:51 -04:00
|
|
|
add_u32_option(packet, DCODE_SERVER_ID, sid);
|
2011-07-25 23:34:32 -04:00
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_clientid(struct dhcpmsg *packet, const char * const clientid,
|
2014-03-18 03:38:32 -04:00
|
|
|
size_t clen)
|
2014-03-18 03:13:51 -04:00
|
|
|
{
|
|
|
|
add_option_string(packet, DCODE_CLIENT_ID, clientid, clen);
|
|
|
|
}
|
|
|
|
|
2011-09-02 02:12:51 -04:00
|
|
|
#ifndef NDHS_BUILD
|
2015-02-14 02:14:30 -05:00
|
|
|
void add_option_maxsize(struct dhcpmsg *packet)
|
2011-07-25 23:34:32 -04:00
|
|
|
{
|
2011-09-02 02:12:51 -04:00
|
|
|
add_u16_option(packet, DCODE_MAX_SIZE,
|
|
|
|
htons(sizeof(struct ip_udp_dhcp_packet)));
|
2011-07-25 23:34:32 -04:00
|
|
|
}
|
|
|
|
|
2015-02-20 03:58:25 -05:00
|
|
|
void add_option_vendor(struct dhcpmsg *packet, const char * const vendor,
|
|
|
|
size_t vsize)
|
2011-07-25 23:48:35 -04:00
|
|
|
{
|
2015-02-20 03:58:25 -05:00
|
|
|
if (vsize)
|
|
|
|
add_option_string(packet, DCODE_VENDOR, vendor, vsize);
|
2011-07-25 23:48:35 -04:00
|
|
|
}
|
|
|
|
|
2015-02-20 03:58:25 -05:00
|
|
|
void add_option_hostname(struct dhcpmsg *packet, const char * const hostname,
|
|
|
|
size_t hsize)
|
2011-07-25 23:48:35 -04:00
|
|
|
{
|
2015-02-20 03:58:25 -05:00
|
|
|
if (hsize)
|
|
|
|
add_option_string(packet, DCODE_HOSTNAME, hostname, hsize);
|
2011-07-25 23:48:35 -04:00
|
|
|
}
|
2011-09-02 02:12:51 -04:00
|
|
|
#endif
|
2011-07-25 23:48:35 -04:00
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
uint32_t get_option_router(const struct dhcpmsg * const packet)
|
2011-07-26 01:04:59 -04:00
|
|
|
{
|
|
|
|
uint32_t ret = 0;
|
2011-07-27 07:39:45 -04:00
|
|
|
uint8_t buf[MAX_DOPT_SIZE];
|
2018-02-09 02:39:46 -05:00
|
|
|
const size_t ol = get_dhcp_opt(packet, DCODE_ROUTER, buf, sizeof buf);
|
2011-07-26 01:04:59 -04:00
|
|
|
if (ol == sizeof ret)
|
2011-07-27 07:39:45 -04:00
|
|
|
memcpy(&ret, buf, sizeof ret);
|
2011-07-26 01:04:59 -04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
uint8_t get_option_msgtype(const struct dhcpmsg * const packet)
|
2011-07-26 01:04:59 -04:00
|
|
|
{
|
|
|
|
uint8_t ret = 0;
|
2011-07-27 07:39:45 -04:00
|
|
|
uint8_t buf[MAX_DOPT_SIZE];
|
2018-02-09 02:39:46 -05:00
|
|
|
const size_t ol = get_dhcp_opt(packet, DCODE_MSGTYPE, buf, sizeof buf);
|
2011-07-27 07:39:45 -04:00
|
|
|
if (ol == sizeof ret)
|
|
|
|
ret = buf[0];
|
2011-07-26 01:04:59 -04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
uint32_t get_option_serverid(const struct dhcpmsg *const packet, int *found)
|
2011-07-26 01:04:59 -04:00
|
|
|
{
|
|
|
|
uint32_t ret = 0;
|
2011-07-27 07:39:45 -04:00
|
|
|
uint8_t buf[MAX_DOPT_SIZE];
|
2011-07-26 01:04:59 -04:00
|
|
|
*found = 0;
|
2018-02-09 02:39:46 -05:00
|
|
|
const size_t ol = get_dhcp_opt(packet, DCODE_SERVER_ID, buf, sizeof buf);
|
2011-07-26 01:04:59 -04:00
|
|
|
if (ol == sizeof ret) {
|
|
|
|
*found = 1;
|
2011-07-27 07:39:45 -04:00
|
|
|
memcpy(&ret, buf, sizeof ret);
|
2011-07-26 01:04:59 -04:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-02-14 02:14:30 -05:00
|
|
|
uint32_t get_option_leasetime(const struct dhcpmsg * const packet)
|
2011-07-26 01:04:59 -04:00
|
|
|
{
|
|
|
|
uint32_t ret = 0;
|
2011-07-27 07:39:45 -04:00
|
|
|
uint8_t buf[MAX_DOPT_SIZE];
|
2018-02-09 02:39:46 -05:00
|
|
|
const size_t ol = get_dhcp_opt(packet, DCODE_LEASET, buf, sizeof buf);
|
2011-07-26 01:04:59 -04:00
|
|
|
if (ol == sizeof ret) {
|
2011-07-27 07:39:45 -04:00
|
|
|
memcpy(&ret, buf, sizeof ret);
|
2011-07-26 01:04:59 -04:00
|
|
|
ret = ntohl(ret);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2014-03-18 03:13:51 -04:00
|
|
|
|
|
|
|
// Returned buffer is not nul-terminated.
|
2015-02-14 02:14:30 -05:00
|
|
|
size_t get_option_clientid(const struct dhcpmsg * const packet, char *cbuf,
|
|
|
|
size_t clen)
|
2014-03-18 03:13:51 -04:00
|
|
|
{
|
|
|
|
if (clen < 1)
|
|
|
|
return 0;
|
2018-02-09 02:39:46 -05:00
|
|
|
return get_dhcp_opt(packet, DCODE_CLIENT_ID, (uint8_t *)cbuf, clen);
|
2014-03-18 03:13:51 -04:00
|
|
|
}
|
2015-02-14 02:14:30 -05:00
|
|
|
|