X-Git-Url: https://git.decadent.org.uk/gitweb/?a=blobdiff_plain;f=src%2Fdhcpv6.c;h=c9915c260aa74ff0d6d6a3a9fd685ce59beba867;hb=b4a0f3d74b04074eae21e88fbf5b8f2ddc150f86;hp=973ca8f03502a8830158316a063ae73ff221683f;hpb=b146f9adc80cc2c2cdf2b04bfeec4c861a2a0e23;p=odhcp6c.git diff --git a/src/dhcpv6.c b/src/dhcpv6.c index 973ca8f..c9915c2 100644 --- a/src/dhcpv6.c +++ b/src/dhcpv6.c @@ -1,5 +1,5 @@ /** - * Copyright (C) 2012-2013 Steven Barth + * Copyright (C) 2012-2014 Steven Barth * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License v2 as published by @@ -105,8 +105,13 @@ static bool accept_reconfig = false; // Reconfigure key static uint8_t reconf_key[16]; -int init_dhcpv6(const char *ifname, bool strict_options, int sol_timeout) +// client options +static unsigned int client_options = 0; + + +int init_dhcpv6(const char *ifname, unsigned int options, int sol_timeout) { + client_options = options; dhcpv6_retx[DHCPV6_MSG_SOLICIT].max_timeo = sol_timeout; sock = socket(AF_INET6, SOCK_DGRAM | SOCK_CLOEXEC, IPPROTO_UDP); @@ -156,7 +161,7 @@ int init_dhcpv6(const char *ifname, bool strict_options, int sol_timeout) } // Create ORO - if (!strict_options) { + if (!(client_options & DHCPV6_STRICT_OPTIONS)) { uint16_t oro[] = { htons(DHCPV6_OPT_SIP_SERVER_D), htons(DHCPV6_OPT_SIP_SERVER_A), @@ -363,6 +368,15 @@ static void dhcpv6_send(enum dhcpv6_msg type, uint8_t trid[3], uint32_t ecs) // Request Information Refresh uint16_t oro_refresh = htons(DHCPV6_OPT_INFO_REFRESH); + // Build vendor-class option + size_t vendor_class_len; + struct dhcpv6_vendorclass *vendor_class = odhcp6c_get_state(STATE_VENDORCLASS, &vendor_class_len); + + struct { + uint16_t type; + uint16_t length; + } vendor_class_hdr = {htons(DHCPV6_OPT_VENDOR_CLASS), htons(vendor_class_len)}; + // Prepare Header size_t oro_len; void *oro = odhcp6c_get_state(STATE_ORO, &oro_len); @@ -387,6 +401,8 @@ static void dhcpv6_send(enum dhcpv6_msg type, uint8_t trid[3], uint32_t ecs) {&oro_refresh, 0}, {cl_id, cl_id_len}, {srv_id, srv_id_len}, + {&vendor_class_hdr, vendor_class_len ? sizeof(vendor_class_hdr) : 0}, + {vendor_class, vendor_class_len}, {&reconf_accept, sizeof(reconf_accept)}, {&fqdn, fqdn_len}, {&hdr_ia_na, sizeof(hdr_ia_na)}, @@ -396,23 +412,29 @@ static void dhcpv6_send(enum dhcpv6_msg type, uint8_t trid[3], uint32_t ecs) size_t cnt = ARRAY_SIZE(iov); if (type == DHCPV6_MSG_INFO_REQ) { - cnt = 5; + cnt = 7; iov[2].iov_len = sizeof(oro_refresh); hdr.oro_len = htons(oro_len + sizeof(oro_refresh)); } else if (!request_prefix) { - cnt = 9; + cnt = 11; } // Disable IAs if not used if (type != DHCPV6_MSG_SOLICIT) { - iov[5].iov_len = 0; + iov[7].iov_len = 0; if (ia_na_len == 0) - iov[7].iov_len = 0; + iov[9].iov_len = 0; } if (na_mode == IA_MODE_NONE) + iov[9].iov_len = 0; + + if (!(client_options & DHCPV6_ACCEPT_RECONFIGURE)) iov[7].iov_len = 0; + if (!(client_options & DHCPV6_CLIENT_FQDN)) + iov[8].iov_len = 0; + struct sockaddr_in6 srv = {AF_INET6, htons(DHCPV6_SERVER_PORT), 0, ALL_DHCPV6_RELAYS, ifindex}; struct msghdr msg = {&srv, sizeof(srv), iov, cnt, NULL, 0, 0};