udhcpc: fix wrong options in decline and release packets

(Jonas Danielsson <jonas.danielsson at axis.com>)
This commit is contained in:
Denis Vlasenko 2008-01-25 19:27:08 +00:00
parent a5549c9617
commit ca9635b19d
3 changed files with 15 additions and 13 deletions

View File

@ -48,14 +48,15 @@ static void init_packet(struct dhcpMessage *packet, char type)
add_option_string(packet->options, client_config.hostname); add_option_string(packet->options, client_config.hostname);
if (client_config.fqdn) if (client_config.fqdn)
add_option_string(packet->options, client_config.fqdn); add_option_string(packet->options, client_config.fqdn);
add_option_string(packet->options, client_config.vendorclass); if ((type != DHCPDECLINE) && (type != DHCPRELEASE))
add_option_string(packet->options, client_config.vendorclass);
} }
/* Add a parameter request list for stubborn DHCP servers. Pull the data /* Add a parameter request list for stubborn DHCP servers. Pull the data
* from the struct in options.c. Don't do bounds checking here because it * from the struct in options.c. Don't do bounds checking here because it
* goes towards the head of the packet. */ * goes towards the head of the packet. */
static void add_requests(struct dhcpMessage *packet) static void add_param_req_option(struct dhcpMessage *packet)
{ {
uint8_t c; uint8_t c;
int end = end_option(packet->options); int end = end_option(packet->options);
@ -63,26 +64,28 @@ static void add_requests(struct dhcpMessage *packet)
packet->options[end + OPT_CODE] = DHCP_PARAM_REQ; packet->options[end + OPT_CODE] = DHCP_PARAM_REQ;
for (i = 0; (c = dhcp_options[i].code) != 0; i++) { for (i = 0; (c = dhcp_options[i].code) != 0; i++) {
if (dhcp_options[i].flags & OPTION_REQ if ((dhcp_options[i].flags & OPTION_REQ)
|| (client_config.opt_mask[c >> 3] & (1 << (c & 7))) || (client_config.opt_mask[c >> 3] & (1 << (c & 7)))
) { ) {
packet->options[end + OPT_DATA + len++] = c; packet->options[end + OPT_DATA + len] = c;
len++;
} }
} }
packet->options[end + OPT_LEN] = len; packet->options[end + OPT_LEN] = len;
packet->options[end + OPT_DATA + len] = DHCP_END; packet->options[end + OPT_DATA + len] = DHCP_END;
} }
#if ENABLE_FEATURE_UDHCPC_ARPING #if ENABLE_FEATURE_UDHCPC_ARPING
/* Unicast a DHCP decline message */ /* Unicast a DHCP decline message */
int send_decline(uint32_t xid, uint32_t server) int send_decline(uint32_t xid, uint32_t server, uint32_t requested)
{ {
struct dhcpMessage packet; struct dhcpMessage packet;
init_packet(&packet, DHCPDECLINE); init_packet(&packet, DHCPDECLINE);
packet.xid = xid; packet.xid = xid;
add_requests(&packet); add_simple_option(packet.options, DHCP_REQUESTED_IP, requested);
add_simple_option(packet.options, DHCP_SERVER_ID, server);
bb_info_msg("Sending decline..."); bb_info_msg("Sending decline...");
@ -104,7 +107,7 @@ int send_discover(uint32_t xid, uint32_t requested)
/* Explicitly saying that we want RFC-compliant packets helps /* Explicitly saying that we want RFC-compliant packets helps
* some buggy DHCP servers to NOT send bigger packets */ * some buggy DHCP servers to NOT send bigger packets */
add_simple_option(packet.options, DHCP_MAX_SIZE, htons(576)); add_simple_option(packet.options, DHCP_MAX_SIZE, htons(576));
add_requests(&packet); add_param_req_option(&packet);
bb_info_msg("Sending discover..."); bb_info_msg("Sending discover...");
return udhcp_send_raw_packet(&packet, INADDR_ANY, CLIENT_PORT, INADDR_BROADCAST, return udhcp_send_raw_packet(&packet, INADDR_ANY, CLIENT_PORT, INADDR_BROADCAST,
SERVER_PORT, MAC_BCAST_ADDR, client_config.ifindex); SERVER_PORT, MAC_BCAST_ADDR, client_config.ifindex);
@ -123,7 +126,7 @@ int send_selecting(uint32_t xid, uint32_t server, uint32_t requested)
add_simple_option(packet.options, DHCP_REQUESTED_IP, requested); add_simple_option(packet.options, DHCP_REQUESTED_IP, requested);
add_simple_option(packet.options, DHCP_SERVER_ID, server); add_simple_option(packet.options, DHCP_SERVER_ID, server);
add_requests(&packet); add_param_req_option(&packet);
addr.s_addr = requested; addr.s_addr = requested;
bb_info_msg("Sending select for %s...", inet_ntoa(addr)); bb_info_msg("Sending select for %s...", inet_ntoa(addr));
return udhcp_send_raw_packet(&packet, INADDR_ANY, CLIENT_PORT, INADDR_BROADCAST, return udhcp_send_raw_packet(&packet, INADDR_ANY, CLIENT_PORT, INADDR_BROADCAST,
@ -140,7 +143,7 @@ int send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr)
packet.xid = xid; packet.xid = xid;
packet.ciaddr = ciaddr; packet.ciaddr = ciaddr;
add_requests(&packet); add_param_req_option(&packet);
bb_info_msg("Sending renew..."); bb_info_msg("Sending renew...");
if (server) if (server)
return udhcp_send_kernel_packet(&packet, ciaddr, CLIENT_PORT, server, SERVER_PORT); return udhcp_send_kernel_packet(&packet, ciaddr, CLIENT_PORT, server, SERVER_PORT);
@ -159,7 +162,6 @@ int send_release(uint32_t server, uint32_t ciaddr)
packet.xid = random_xid(); packet.xid = random_xid();
packet.ciaddr = ciaddr; packet.ciaddr = ciaddr;
add_simple_option(packet.options, DHCP_REQUESTED_IP, ciaddr);
add_simple_option(packet.options, DHCP_SERVER_ID, server); add_simple_option(packet.options, DHCP_SERVER_ID, server);
bb_info_msg("Sending release..."); bb_info_msg("Sending release...");

View File

@ -523,7 +523,7 @@ int udhcpc_main(int argc, char **argv)
) { ) {
bb_info_msg("offered address is in use " bb_info_msg("offered address is in use "
"(got ARP reply), declining"); "(got ARP reply), declining");
send_decline(xid, server_addr); send_decline(xid, server_addr, packet.yiaddr);
if (state != REQUESTING) if (state != REQUESTING)
udhcp_run_script(NULL, "deconfig"); udhcp_run_script(NULL, "deconfig");

View File

@ -42,7 +42,7 @@ uint32_t random_xid(void);
int send_discover(uint32_t xid, uint32_t requested); int send_discover(uint32_t xid, uint32_t requested);
int send_selecting(uint32_t xid, uint32_t server, uint32_t requested); int send_selecting(uint32_t xid, uint32_t server, uint32_t requested);
#if ENABLE_FEATURE_UDHCPC_ARPING #if ENABLE_FEATURE_UDHCPC_ARPING
int send_decline(uint32_t xid, uint32_t server); int send_decline(uint32_t xid, uint32_t server, uint32_t requested);
#endif #endif
int send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr); int send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr);
int send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr); int send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr);