2 * Copyright (C) 2012 Steven Barth <steven@midlink.org>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License v2 as published by
6 * the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
17 #include <netinet/in.h>
19 #define _unused __attribute__((unused))
20 #define _packed __attribute__((packed))
22 #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
25 #define SOL_NETLINK 270
28 #define ND_OPT_RECURSIVE_DNS 25
29 #define ND_OPT_DNSSL 31
32 DHCPV6_OPT_CLIENTID = 1,
33 DHCPV6_OPT_SERVERID = 2,
35 DHCPV6_OPT_IA_ADDR = 5,
38 DHCPV6_OPT_ELAPSED = 8,
39 DHCPV6_OPT_RELAY_MSG = 9,
41 DHCPV6_OPT_STATUS = 13,
42 DHCPV6_OPT_RAPID_COMMIT = 14,
43 DHCPV6_OPT_RECONF_MESSAGE = 19,
44 DHCPV6_OPT_RECONF_ACCEPT = 20,
45 DHCPV6_OPT_DNS_SERVERS = 23,
46 DHCPV6_OPT_DNS_DOMAIN = 24,
47 DHCPV6_OPT_IA_PD = 25,
48 DHCPV6_OPT_IA_PREFIX = 26,
49 DHCPV6_OPT_INFO_REFRESH = 32,
51 DHCPV6_OPT_NTP_SERVER = 56,
52 DHCPV6_OPT_SIP_SERVER_D = 21,
53 DHCPV6_OPT_SIP_SERVER_A = 22,
63 DHCPV6_MSG_UNKNOWN = 0,
64 DHCPV6_MSG_SOLICIT = 1,
65 DHCPV6_MSG_ADVERT = 2,
66 DHCPV6_MSG_REQUEST = 3,
68 DHCPV6_MSG_REBIND = 6,
70 DHCPV6_MSG_RELEASE = 8,
71 DHCPV6_MSG_DECLINE = 9,
72 DHCPV6_MSG_RECONF = 10,
73 DHCPV6_MSG_INFO_REQ = 11,
78 DHCPV6_NoAddrsAvail = 2,
79 DHCPV6_NoPrefixAvail = 6,
82 typedef int(reply_handler)(enum dhcpv6_msg orig,
83 const void *opt, const void *end);
85 // retransmission strategy
91 reply_handler *handler_reply;
92 int(*handler_finish)(void);
96 // DHCPv6 Protocol Headers
97 struct dhcpv6_header {
100 } __attribute__((packed));
102 struct dhcpv6_ia_hdr {
110 struct dhcpv6_ia_addr {
113 struct in6_addr addr;
118 struct dhcpv6_ia_prefix {
124 struct in6_addr addr;
135 #define dhcpv6_for_each_option(start, end, otype, olen, odata)\
136 for (uint8_t *_o = (uint8_t*)(start); _o + 4 <= (uint8_t*)(end) &&\
137 ((otype) = _o[0] << 8 | _o[1]) && ((odata) = (void*)&_o[4]) &&\
138 ((olen) = _o[2] << 8 | _o[3]) + (odata) <= (uint8_t*)(end); \
139 _o += 4 + (_o[2] << 8 | _o[3]))
142 struct dhcpv6_server_cand {
143 bool has_noaddravail;
144 bool wants_reconfigure;
186 enum odhcp6c_ia_mode {
193 struct odhcp6c_entry {
194 struct in6_addr router;
197 struct in6_addr target;
203 int init_dhcpv6(const char *ifname, int request_pd);
204 void dhcpv6_set_ia_na_mode(enum odhcp6c_ia_mode mode);
205 int dhcpv6_request(enum dhcpv6_msg type);
206 int dhcpv6_poll_reconfigure(void);
208 int init_rtnetlink(void);
209 int set_rtnetlink_addr(int ifindex, const struct in6_addr *addr,
210 uint32_t pref, uint32_t valid);
212 int script_init(const char *path, const char *ifname);
213 ssize_t script_unhexlify(uint8_t *dst, size_t len, const char *src);
214 void script_call(const char *status);
216 bool odhcp6c_signal_process(void);
217 uint64_t odhcp6c_get_milli_time(void);
218 void odhcp6c_random(void *buf, size_t len);
220 // State manipulation
221 void odhcp6c_clear_state(enum odhcp6c_state state);
222 void odhcp6c_add_state(enum odhcp6c_state state, const void *data, size_t len);
223 size_t odhcp6c_remove_state(enum odhcp6c_state state, size_t offset, size_t len);
224 void* odhcp6c_get_state(enum odhcp6c_state state, size_t *len);
226 // Entry manipulation
227 struct odhcp6c_entry* odhcp6c_find_entry(enum odhcp6c_state state, const struct odhcp6c_entry *new);
228 void odhcp6c_update_entry(enum odhcp6c_state state, struct odhcp6c_entry *new);
229 void odhcp6c_update_entry_safe(enum odhcp6c_state state, struct odhcp6c_entry *new, uint32_t safe);
231 void odhcp6c_expire(void);