1*9f20bfa6SDavid van Moolenbroek #include <sys/cdefs.h>
2*9f20bfa6SDavid van Moolenbroek __RCSID("$NetBSD: ipv4ll.c,v 1.12 2015/08/21 10:39:00 roy Exp $");
3*9f20bfa6SDavid van Moolenbroek
4*9f20bfa6SDavid van Moolenbroek /*
5*9f20bfa6SDavid van Moolenbroek * dhcpcd - DHCP client daemon
6*9f20bfa6SDavid van Moolenbroek * Copyright (c) 2006-2015 Roy Marples <roy@marples.name>
7*9f20bfa6SDavid van Moolenbroek * All rights reserved
8*9f20bfa6SDavid van Moolenbroek
9*9f20bfa6SDavid van Moolenbroek * Redistribution and use in source and binary forms, with or without
10*9f20bfa6SDavid van Moolenbroek * modification, are permitted provided that the following conditions
11*9f20bfa6SDavid van Moolenbroek * are met:
12*9f20bfa6SDavid van Moolenbroek * 1. Redistributions of source code must retain the above copyright
13*9f20bfa6SDavid van Moolenbroek * notice, this list of conditions and the following disclaimer.
14*9f20bfa6SDavid van Moolenbroek * 2. Redistributions in binary form must reproduce the above copyright
15*9f20bfa6SDavid van Moolenbroek * notice, this list of conditions and the following disclaimer in the
16*9f20bfa6SDavid van Moolenbroek * documentation and/or other materials provided with the distribution.
17*9f20bfa6SDavid van Moolenbroek *
18*9f20bfa6SDavid van Moolenbroek * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19*9f20bfa6SDavid van Moolenbroek * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20*9f20bfa6SDavid van Moolenbroek * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21*9f20bfa6SDavid van Moolenbroek * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
22*9f20bfa6SDavid van Moolenbroek * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23*9f20bfa6SDavid van Moolenbroek * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24*9f20bfa6SDavid van Moolenbroek * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25*9f20bfa6SDavid van Moolenbroek * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26*9f20bfa6SDavid van Moolenbroek * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27*9f20bfa6SDavid van Moolenbroek * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28*9f20bfa6SDavid van Moolenbroek * SUCH DAMAGE.
29*9f20bfa6SDavid van Moolenbroek */
30*9f20bfa6SDavid van Moolenbroek
31*9f20bfa6SDavid van Moolenbroek #include <arpa/inet.h>
32*9f20bfa6SDavid van Moolenbroek
33*9f20bfa6SDavid van Moolenbroek #include <assert.h>
34*9f20bfa6SDavid van Moolenbroek #include <errno.h>
35*9f20bfa6SDavid van Moolenbroek #include <signal.h>
36*9f20bfa6SDavid van Moolenbroek #include <stdlib.h>
37*9f20bfa6SDavid van Moolenbroek #include <string.h>
38*9f20bfa6SDavid van Moolenbroek #include <unistd.h>
39*9f20bfa6SDavid van Moolenbroek
40*9f20bfa6SDavid van Moolenbroek #define ELOOP_QUEUE 6
41*9f20bfa6SDavid van Moolenbroek #include "config.h"
42*9f20bfa6SDavid van Moolenbroek #include "arp.h"
43*9f20bfa6SDavid van Moolenbroek #include "common.h"
44*9f20bfa6SDavid van Moolenbroek #include "eloop.h"
45*9f20bfa6SDavid van Moolenbroek #include "if.h"
46*9f20bfa6SDavid van Moolenbroek #include "if-options.h"
47*9f20bfa6SDavid van Moolenbroek #include "ipv4.h"
48*9f20bfa6SDavid van Moolenbroek #include "ipv4ll.h"
49*9f20bfa6SDavid van Moolenbroek #include "script.h"
50*9f20bfa6SDavid van Moolenbroek
51*9f20bfa6SDavid van Moolenbroek const struct in_addr inaddr_llmask = { HTONL(LINKLOCAL_MASK) };
52*9f20bfa6SDavid van Moolenbroek const struct in_addr inaddr_llbcast = { HTONL(LINKLOCAL_BRDC) };
53*9f20bfa6SDavid van Moolenbroek
54*9f20bfa6SDavid van Moolenbroek static in_addr_t
ipv4ll_pick_addr(const struct arp_state * astate)55*9f20bfa6SDavid van Moolenbroek ipv4ll_pick_addr(const struct arp_state *astate)
56*9f20bfa6SDavid van Moolenbroek {
57*9f20bfa6SDavid van Moolenbroek struct in_addr addr;
58*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *istate;
59*9f20bfa6SDavid van Moolenbroek
60*9f20bfa6SDavid van Moolenbroek istate = IPV4LL_STATE(astate->iface);
61*9f20bfa6SDavid van Moolenbroek setstate(istate->randomstate);
62*9f20bfa6SDavid van Moolenbroek
63*9f20bfa6SDavid van Moolenbroek do {
64*9f20bfa6SDavid van Moolenbroek /* RFC 3927 Section 2.1 states that the first 256 and
65*9f20bfa6SDavid van Moolenbroek * last 256 addresses are reserved for future use.
66*9f20bfa6SDavid van Moolenbroek * See ipv4ll_start for why we don't use arc4_random. */
67*9f20bfa6SDavid van Moolenbroek addr.s_addr = ntohl(LINKLOCAL_ADDR |
68*9f20bfa6SDavid van Moolenbroek ((uint32_t)(random() % 0xFD00) + 0x0100));
69*9f20bfa6SDavid van Moolenbroek
70*9f20bfa6SDavid van Moolenbroek /* No point using a failed address */
71*9f20bfa6SDavid van Moolenbroek if (addr.s_addr == astate->failed.s_addr)
72*9f20bfa6SDavid van Moolenbroek continue;
73*9f20bfa6SDavid van Moolenbroek /* Ensure we don't have the address on another interface */
74*9f20bfa6SDavid van Moolenbroek } while (ipv4_findaddr(astate->iface->ctx, &addr) != NULL);
75*9f20bfa6SDavid van Moolenbroek
76*9f20bfa6SDavid van Moolenbroek /* Restore the original random state */
77*9f20bfa6SDavid van Moolenbroek setstate(astate->iface->ctx->randomstate);
78*9f20bfa6SDavid van Moolenbroek
79*9f20bfa6SDavid van Moolenbroek return addr.s_addr;
80*9f20bfa6SDavid van Moolenbroek }
81*9f20bfa6SDavid van Moolenbroek
82*9f20bfa6SDavid van Moolenbroek struct rt *
ipv4ll_subnet_route(const struct interface * ifp)83*9f20bfa6SDavid van Moolenbroek ipv4ll_subnet_route(const struct interface *ifp)
84*9f20bfa6SDavid van Moolenbroek {
85*9f20bfa6SDavid van Moolenbroek const struct ipv4ll_state *state;
86*9f20bfa6SDavid van Moolenbroek struct rt *rt;
87*9f20bfa6SDavid van Moolenbroek
88*9f20bfa6SDavid van Moolenbroek assert(ifp != NULL);
89*9f20bfa6SDavid van Moolenbroek if ((state = IPV4LL_CSTATE(ifp)) == NULL ||
90*9f20bfa6SDavid van Moolenbroek state->addr.s_addr == INADDR_ANY)
91*9f20bfa6SDavid van Moolenbroek return NULL;
92*9f20bfa6SDavid van Moolenbroek
93*9f20bfa6SDavid van Moolenbroek if ((rt = calloc(1, sizeof(*rt))) == NULL) {
94*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_ERR, "%s: malloc: %m", __func__);
95*9f20bfa6SDavid van Moolenbroek return NULL;
96*9f20bfa6SDavid van Moolenbroek }
97*9f20bfa6SDavid van Moolenbroek rt->iface = ifp;
98*9f20bfa6SDavid van Moolenbroek rt->dest.s_addr = state->addr.s_addr & inaddr_llmask.s_addr;
99*9f20bfa6SDavid van Moolenbroek rt->net = inaddr_llmask;
100*9f20bfa6SDavid van Moolenbroek rt->gate.s_addr = INADDR_ANY;
101*9f20bfa6SDavid van Moolenbroek rt->src = state->addr;
102*9f20bfa6SDavid van Moolenbroek return rt;
103*9f20bfa6SDavid van Moolenbroek }
104*9f20bfa6SDavid van Moolenbroek
105*9f20bfa6SDavid van Moolenbroek struct rt *
ipv4ll_default_route(const struct interface * ifp)106*9f20bfa6SDavid van Moolenbroek ipv4ll_default_route(const struct interface *ifp)
107*9f20bfa6SDavid van Moolenbroek {
108*9f20bfa6SDavid van Moolenbroek const struct ipv4ll_state *state;
109*9f20bfa6SDavid van Moolenbroek struct rt *rt;
110*9f20bfa6SDavid van Moolenbroek
111*9f20bfa6SDavid van Moolenbroek assert(ifp != NULL);
112*9f20bfa6SDavid van Moolenbroek if ((state = IPV4LL_CSTATE(ifp)) == NULL ||
113*9f20bfa6SDavid van Moolenbroek state->addr.s_addr == INADDR_ANY)
114*9f20bfa6SDavid van Moolenbroek return NULL;
115*9f20bfa6SDavid van Moolenbroek
116*9f20bfa6SDavid van Moolenbroek if ((rt = calloc(1, sizeof(*rt))) == NULL) {
117*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_ERR, "%s: malloc: %m", __func__);
118*9f20bfa6SDavid van Moolenbroek return NULL;
119*9f20bfa6SDavid van Moolenbroek }
120*9f20bfa6SDavid van Moolenbroek rt->iface = ifp;
121*9f20bfa6SDavid van Moolenbroek rt->dest.s_addr = INADDR_ANY;
122*9f20bfa6SDavid van Moolenbroek rt->net.s_addr = INADDR_ANY;
123*9f20bfa6SDavid van Moolenbroek rt->gate.s_addr = INADDR_ANY;
124*9f20bfa6SDavid van Moolenbroek rt->src = state->addr;
125*9f20bfa6SDavid van Moolenbroek return rt;
126*9f20bfa6SDavid van Moolenbroek }
127*9f20bfa6SDavid van Moolenbroek
128*9f20bfa6SDavid van Moolenbroek ssize_t
ipv4ll_env(char ** env,const char * prefix,const struct interface * ifp)129*9f20bfa6SDavid van Moolenbroek ipv4ll_env(char **env, const char *prefix, const struct interface *ifp)
130*9f20bfa6SDavid van Moolenbroek {
131*9f20bfa6SDavid van Moolenbroek const struct ipv4ll_state *state;
132*9f20bfa6SDavid van Moolenbroek const char *pf = prefix == NULL ? "" : "_";
133*9f20bfa6SDavid van Moolenbroek struct in_addr netnum;
134*9f20bfa6SDavid van Moolenbroek
135*9f20bfa6SDavid van Moolenbroek assert(ifp != NULL);
136*9f20bfa6SDavid van Moolenbroek if ((state = IPV4LL_CSTATE(ifp)) == NULL)
137*9f20bfa6SDavid van Moolenbroek return 0;
138*9f20bfa6SDavid van Moolenbroek
139*9f20bfa6SDavid van Moolenbroek if (env == NULL)
140*9f20bfa6SDavid van Moolenbroek return 5;
141*9f20bfa6SDavid van Moolenbroek
142*9f20bfa6SDavid van Moolenbroek /* Emulate a DHCP environment */
143*9f20bfa6SDavid van Moolenbroek if (asprintf(&env[0], "%s%sip_address=%s",
144*9f20bfa6SDavid van Moolenbroek prefix, pf, inet_ntoa(state->addr)) == -1)
145*9f20bfa6SDavid van Moolenbroek return -1;
146*9f20bfa6SDavid van Moolenbroek if (asprintf(&env[1], "%s%ssubnet_mask=%s",
147*9f20bfa6SDavid van Moolenbroek prefix, pf, inet_ntoa(inaddr_llmask)) == -1)
148*9f20bfa6SDavid van Moolenbroek return -1;
149*9f20bfa6SDavid van Moolenbroek if (asprintf(&env[2], "%s%ssubnet_cidr=%d",
150*9f20bfa6SDavid van Moolenbroek prefix, pf, inet_ntocidr(inaddr_llmask)) == -1)
151*9f20bfa6SDavid van Moolenbroek return -1;
152*9f20bfa6SDavid van Moolenbroek if (asprintf(&env[3], "%s%sbroadcast_address=%s",
153*9f20bfa6SDavid van Moolenbroek prefix, pf, inet_ntoa(inaddr_llbcast)) == -1)
154*9f20bfa6SDavid van Moolenbroek return -1;
155*9f20bfa6SDavid van Moolenbroek netnum.s_addr = state->addr.s_addr & inaddr_llmask.s_addr;
156*9f20bfa6SDavid van Moolenbroek if (asprintf(&env[4], "%s%snetwork_number=%s",
157*9f20bfa6SDavid van Moolenbroek prefix, pf, inet_ntoa(netnum)) == -1)
158*9f20bfa6SDavid van Moolenbroek return -1;
159*9f20bfa6SDavid van Moolenbroek return 5;
160*9f20bfa6SDavid van Moolenbroek }
161*9f20bfa6SDavid van Moolenbroek
162*9f20bfa6SDavid van Moolenbroek static void
ipv4ll_probed(struct arp_state * astate)163*9f20bfa6SDavid van Moolenbroek ipv4ll_probed(struct arp_state *astate)
164*9f20bfa6SDavid van Moolenbroek {
165*9f20bfa6SDavid van Moolenbroek struct interface *ifp;
166*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *state;
167*9f20bfa6SDavid van Moolenbroek struct ipv4_addr *ia;
168*9f20bfa6SDavid van Moolenbroek
169*9f20bfa6SDavid van Moolenbroek assert(astate != NULL);
170*9f20bfa6SDavid van Moolenbroek assert(astate->iface != NULL);
171*9f20bfa6SDavid van Moolenbroek
172*9f20bfa6SDavid van Moolenbroek ifp = astate->iface;
173*9f20bfa6SDavid van Moolenbroek state = IPV4LL_STATE(ifp);
174*9f20bfa6SDavid van Moolenbroek assert(state != NULL);
175*9f20bfa6SDavid van Moolenbroek
176*9f20bfa6SDavid van Moolenbroek ia = ipv4_iffindaddr(ifp, &astate->addr, &inaddr_llmask);
177*9f20bfa6SDavid van Moolenbroek #ifdef IN_IFF_NOTREADY
178*9f20bfa6SDavid van Moolenbroek if (ia == NULL || ia->addr_flags & IN_IFF_NOTREADY)
179*9f20bfa6SDavid van Moolenbroek #endif
180*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_INFO, "%s: using IPv4LL address %s",
181*9f20bfa6SDavid van Moolenbroek ifp->name, inet_ntoa(astate->addr));
182*9f20bfa6SDavid van Moolenbroek if (ia == NULL)
183*9f20bfa6SDavid van Moolenbroek ia = ipv4_addaddr(ifp, &astate->addr,
184*9f20bfa6SDavid van Moolenbroek &inaddr_llmask, &inaddr_llbcast);
185*9f20bfa6SDavid van Moolenbroek if (ia == NULL)
186*9f20bfa6SDavid van Moolenbroek return;
187*9f20bfa6SDavid van Moolenbroek #ifdef IN_IFF_NOTREADY
188*9f20bfa6SDavid van Moolenbroek if (ia->addr_flags & IN_IFF_NOTREADY)
189*9f20bfa6SDavid van Moolenbroek return;
190*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_DEBUG, "%s: DAD completed for %s",
191*9f20bfa6SDavid van Moolenbroek ifp->name, inet_ntoa(astate->addr));
192*9f20bfa6SDavid van Moolenbroek #endif
193*9f20bfa6SDavid van Moolenbroek state->addr = astate->addr;
194*9f20bfa6SDavid van Moolenbroek timespecclear(&state->defend);
195*9f20bfa6SDavid van Moolenbroek if_initrt(ifp);
196*9f20bfa6SDavid van Moolenbroek ipv4_buildroutes(ifp->ctx);
197*9f20bfa6SDavid van Moolenbroek arp_announce(astate);
198*9f20bfa6SDavid van Moolenbroek script_runreason(ifp, "IPV4LL");
199*9f20bfa6SDavid van Moolenbroek dhcpcd_daemonise(ifp->ctx);
200*9f20bfa6SDavid van Moolenbroek }
201*9f20bfa6SDavid van Moolenbroek
202*9f20bfa6SDavid van Moolenbroek static void
ipv4ll_announced(struct arp_state * astate)203*9f20bfa6SDavid van Moolenbroek ipv4ll_announced(struct arp_state *astate)
204*9f20bfa6SDavid van Moolenbroek {
205*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *state = IPV4LL_STATE(astate->iface);
206*9f20bfa6SDavid van Moolenbroek
207*9f20bfa6SDavid van Moolenbroek state->conflicts = 0;
208*9f20bfa6SDavid van Moolenbroek /* Need to keep the arp state so we can defend our IP. */
209*9f20bfa6SDavid van Moolenbroek }
210*9f20bfa6SDavid van Moolenbroek
211*9f20bfa6SDavid van Moolenbroek static void
ipv4ll_probe(void * arg)212*9f20bfa6SDavid van Moolenbroek ipv4ll_probe(void *arg)
213*9f20bfa6SDavid van Moolenbroek {
214*9f20bfa6SDavid van Moolenbroek
215*9f20bfa6SDavid van Moolenbroek #ifdef IN_IFF_TENTATIVE
216*9f20bfa6SDavid van Moolenbroek ipv4ll_probed(arg);
217*9f20bfa6SDavid van Moolenbroek #else
218*9f20bfa6SDavid van Moolenbroek arp_probe(arg);
219*9f20bfa6SDavid van Moolenbroek #endif
220*9f20bfa6SDavid van Moolenbroek }
221*9f20bfa6SDavid van Moolenbroek
222*9f20bfa6SDavid van Moolenbroek static void
ipv4ll_conflicted(struct arp_state * astate,const struct arp_msg * amsg)223*9f20bfa6SDavid van Moolenbroek ipv4ll_conflicted(struct arp_state *astate, const struct arp_msg *amsg)
224*9f20bfa6SDavid van Moolenbroek {
225*9f20bfa6SDavid van Moolenbroek struct interface *ifp;
226*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *state;
227*9f20bfa6SDavid van Moolenbroek in_addr_t fail;
228*9f20bfa6SDavid van Moolenbroek
229*9f20bfa6SDavid van Moolenbroek assert(astate != NULL);
230*9f20bfa6SDavid van Moolenbroek assert(astate->iface != NULL);
231*9f20bfa6SDavid van Moolenbroek ifp = astate->iface;
232*9f20bfa6SDavid van Moolenbroek state = IPV4LL_STATE(ifp);
233*9f20bfa6SDavid van Moolenbroek assert(state != NULL);
234*9f20bfa6SDavid van Moolenbroek
235*9f20bfa6SDavid van Moolenbroek fail = 0;
236*9f20bfa6SDavid van Moolenbroek /* RFC 3927 2.2.1, Probe Conflict Detection */
237*9f20bfa6SDavid van Moolenbroek if (amsg == NULL ||
238*9f20bfa6SDavid van Moolenbroek (amsg->sip.s_addr == astate->addr.s_addr ||
239*9f20bfa6SDavid van Moolenbroek (amsg->sip.s_addr == 0 && amsg->tip.s_addr == astate->addr.s_addr)))
240*9f20bfa6SDavid van Moolenbroek fail = astate->addr.s_addr;
241*9f20bfa6SDavid van Moolenbroek
242*9f20bfa6SDavid van Moolenbroek /* RFC 3927 2.5, Conflict Defense */
243*9f20bfa6SDavid van Moolenbroek if (IN_LINKLOCAL(ntohl(state->addr.s_addr)) &&
244*9f20bfa6SDavid van Moolenbroek amsg && amsg->sip.s_addr == state->addr.s_addr)
245*9f20bfa6SDavid van Moolenbroek fail = state->addr.s_addr;
246*9f20bfa6SDavid van Moolenbroek
247*9f20bfa6SDavid van Moolenbroek if (fail == 0)
248*9f20bfa6SDavid van Moolenbroek return;
249*9f20bfa6SDavid van Moolenbroek
250*9f20bfa6SDavid van Moolenbroek astate->failed.s_addr = fail;
251*9f20bfa6SDavid van Moolenbroek arp_report_conflicted(astate, amsg);
252*9f20bfa6SDavid van Moolenbroek
253*9f20bfa6SDavid van Moolenbroek if (astate->failed.s_addr == state->addr.s_addr) {
254*9f20bfa6SDavid van Moolenbroek struct timespec now, defend;
255*9f20bfa6SDavid van Moolenbroek
256*9f20bfa6SDavid van Moolenbroek /* RFC 3927 Section 2.5 */
257*9f20bfa6SDavid van Moolenbroek defend.tv_sec = state->defend.tv_sec + DEFEND_INTERVAL;
258*9f20bfa6SDavid van Moolenbroek defend.tv_nsec = state->defend.tv_nsec;
259*9f20bfa6SDavid van Moolenbroek clock_gettime(CLOCK_MONOTONIC, &now);
260*9f20bfa6SDavid van Moolenbroek if (timespeccmp(&defend, &now, >)) {
261*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_WARNING,
262*9f20bfa6SDavid van Moolenbroek "%s: IPv4LL %d second defence failed for %s",
263*9f20bfa6SDavid van Moolenbroek ifp->name, DEFEND_INTERVAL,
264*9f20bfa6SDavid van Moolenbroek inet_ntoa(state->addr));
265*9f20bfa6SDavid van Moolenbroek ipv4_deladdr(ifp, &state->addr, &inaddr_llmask, 1);
266*9f20bfa6SDavid van Moolenbroek state->down = 1;
267*9f20bfa6SDavid van Moolenbroek script_runreason(ifp, "IPV4LL");
268*9f20bfa6SDavid van Moolenbroek state->addr.s_addr = INADDR_ANY;
269*9f20bfa6SDavid van Moolenbroek } else {
270*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_DEBUG,
271*9f20bfa6SDavid van Moolenbroek "%s: defended IPv4LL address %s",
272*9f20bfa6SDavid van Moolenbroek ifp->name, inet_ntoa(state->addr));
273*9f20bfa6SDavid van Moolenbroek state->defend = now;
274*9f20bfa6SDavid van Moolenbroek return;
275*9f20bfa6SDavid van Moolenbroek }
276*9f20bfa6SDavid van Moolenbroek }
277*9f20bfa6SDavid van Moolenbroek
278*9f20bfa6SDavid van Moolenbroek arp_cancel(astate);
279*9f20bfa6SDavid van Moolenbroek if (++state->conflicts == MAX_CONFLICTS)
280*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_ERR,
281*9f20bfa6SDavid van Moolenbroek "%s: failed to acquire an IPv4LL address",
282*9f20bfa6SDavid van Moolenbroek ifp->name);
283*9f20bfa6SDavid van Moolenbroek astate->addr.s_addr = ipv4ll_pick_addr(astate);
284*9f20bfa6SDavid van Moolenbroek eloop_timeout_add_sec(ifp->ctx->eloop,
285*9f20bfa6SDavid van Moolenbroek state->conflicts >= MAX_CONFLICTS ?
286*9f20bfa6SDavid van Moolenbroek RATE_LIMIT_INTERVAL : PROBE_WAIT,
287*9f20bfa6SDavid van Moolenbroek ipv4ll_probe, astate);
288*9f20bfa6SDavid van Moolenbroek }
289*9f20bfa6SDavid van Moolenbroek
290*9f20bfa6SDavid van Moolenbroek static void
ipv4ll_arpfree(struct arp_state * astate)291*9f20bfa6SDavid van Moolenbroek ipv4ll_arpfree(struct arp_state *astate)
292*9f20bfa6SDavid van Moolenbroek {
293*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *state;
294*9f20bfa6SDavid van Moolenbroek
295*9f20bfa6SDavid van Moolenbroek state = IPV4LL_STATE(astate->iface);
296*9f20bfa6SDavid van Moolenbroek if (state->arp == astate)
297*9f20bfa6SDavid van Moolenbroek state->arp = NULL;
298*9f20bfa6SDavid van Moolenbroek }
299*9f20bfa6SDavid van Moolenbroek
300*9f20bfa6SDavid van Moolenbroek void
ipv4ll_start(void * arg)301*9f20bfa6SDavid van Moolenbroek ipv4ll_start(void *arg)
302*9f20bfa6SDavid van Moolenbroek {
303*9f20bfa6SDavid van Moolenbroek struct interface *ifp;
304*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *state;
305*9f20bfa6SDavid van Moolenbroek struct arp_state *astate;
306*9f20bfa6SDavid van Moolenbroek struct ipv4_addr *ia;
307*9f20bfa6SDavid van Moolenbroek
308*9f20bfa6SDavid van Moolenbroek assert(arg != NULL);
309*9f20bfa6SDavid van Moolenbroek ifp = arg;
310*9f20bfa6SDavid van Moolenbroek if ((state = IPV4LL_STATE(ifp)) == NULL) {
311*9f20bfa6SDavid van Moolenbroek ifp->if_data[IF_DATA_IPV4LL] = calloc(1, sizeof(*state));
312*9f20bfa6SDavid van Moolenbroek if ((state = IPV4LL_STATE(ifp)) == NULL) {
313*9f20bfa6SDavid van Moolenbroek syslog(LOG_ERR, "%s: calloc %m", __func__);
314*9f20bfa6SDavid van Moolenbroek return;
315*9f20bfa6SDavid van Moolenbroek }
316*9f20bfa6SDavid van Moolenbroek
317*9f20bfa6SDavid van Moolenbroek state->addr.s_addr = INADDR_ANY;
318*9f20bfa6SDavid van Moolenbroek }
319*9f20bfa6SDavid van Moolenbroek
320*9f20bfa6SDavid van Moolenbroek if (state->arp != NULL)
321*9f20bfa6SDavid van Moolenbroek return;
322*9f20bfa6SDavid van Moolenbroek
323*9f20bfa6SDavid van Moolenbroek /* RFC 3927 Section 2.1 states that the random number generator
324*9f20bfa6SDavid van Moolenbroek * SHOULD be seeded with a value derived from persistent information
325*9f20bfa6SDavid van Moolenbroek * such as the IEEE 802 MAC address so that it usually picks
326*9f20bfa6SDavid van Moolenbroek * the same address without persistent storage. */
327*9f20bfa6SDavid van Moolenbroek if (state->conflicts == 0) {
328*9f20bfa6SDavid van Moolenbroek unsigned int seed;
329*9f20bfa6SDavid van Moolenbroek char *orig;
330*9f20bfa6SDavid van Moolenbroek
331*9f20bfa6SDavid van Moolenbroek if (sizeof(seed) > ifp->hwlen) {
332*9f20bfa6SDavid van Moolenbroek seed = 0;
333*9f20bfa6SDavid van Moolenbroek memcpy(&seed, ifp->hwaddr, ifp->hwlen);
334*9f20bfa6SDavid van Moolenbroek } else
335*9f20bfa6SDavid van Moolenbroek memcpy(&seed, ifp->hwaddr + ifp->hwlen - sizeof(seed),
336*9f20bfa6SDavid van Moolenbroek sizeof(seed));
337*9f20bfa6SDavid van Moolenbroek orig = initstate(seed,
338*9f20bfa6SDavid van Moolenbroek state->randomstate, sizeof(state->randomstate));
339*9f20bfa6SDavid van Moolenbroek
340*9f20bfa6SDavid van Moolenbroek /* Save the original state. */
341*9f20bfa6SDavid van Moolenbroek if (ifp->ctx->randomstate == NULL)
342*9f20bfa6SDavid van Moolenbroek ifp->ctx->randomstate = orig;
343*9f20bfa6SDavid van Moolenbroek
344*9f20bfa6SDavid van Moolenbroek /* Set back the original state until we need the seeded one. */
345*9f20bfa6SDavid van Moolenbroek setstate(ifp->ctx->randomstate);
346*9f20bfa6SDavid van Moolenbroek }
347*9f20bfa6SDavid van Moolenbroek
348*9f20bfa6SDavid van Moolenbroek if ((astate = arp_new(ifp, NULL)) == NULL)
349*9f20bfa6SDavid van Moolenbroek return;
350*9f20bfa6SDavid van Moolenbroek
351*9f20bfa6SDavid van Moolenbroek state->arp = astate;
352*9f20bfa6SDavid van Moolenbroek astate->probed_cb = ipv4ll_probed;
353*9f20bfa6SDavid van Moolenbroek astate->announced_cb = ipv4ll_announced;
354*9f20bfa6SDavid van Moolenbroek astate->conflicted_cb = ipv4ll_conflicted;
355*9f20bfa6SDavid van Moolenbroek astate->free_cb = ipv4ll_arpfree;
356*9f20bfa6SDavid van Moolenbroek
357*9f20bfa6SDavid van Moolenbroek /* Find an existing IPv4LL address and ensure we can work with it. */
358*9f20bfa6SDavid van Moolenbroek ia = ipv4_iffindlladdr(ifp);
359*9f20bfa6SDavid van Moolenbroek #ifdef IN_IFF_TENTATIVE
360*9f20bfa6SDavid van Moolenbroek if (ia != NULL && ia->addr_flags & IN_IFF_DUPLICATED) {
361*9f20bfa6SDavid van Moolenbroek ipv4_deladdr(ifp, &ia->addr, &ia->net, 0);
362*9f20bfa6SDavid van Moolenbroek ia = NULL;
363*9f20bfa6SDavid van Moolenbroek }
364*9f20bfa6SDavid van Moolenbroek #endif
365*9f20bfa6SDavid van Moolenbroek if (ia != NULL) {
366*9f20bfa6SDavid van Moolenbroek astate->addr = ia->addr;
367*9f20bfa6SDavid van Moolenbroek #ifdef IN_IFF_TENTATIVE
368*9f20bfa6SDavid van Moolenbroek if (ia->addr_flags & (IN_IFF_TENTATIVE | IN_IFF_DETACHED)) {
369*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_INFO,
370*9f20bfa6SDavid van Moolenbroek "%s: waiting for DAD to complete on %s",
371*9f20bfa6SDavid van Moolenbroek ifp->name, inet_ntoa(ia->addr));
372*9f20bfa6SDavid van Moolenbroek return;
373*9f20bfa6SDavid van Moolenbroek }
374*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_INFO, "%s: using IPv4LL address %s",
375*9f20bfa6SDavid van Moolenbroek ifp->name, inet_ntoa(astate->addr));
376*9f20bfa6SDavid van Moolenbroek #endif
377*9f20bfa6SDavid van Moolenbroek ipv4ll_probed(astate);
378*9f20bfa6SDavid van Moolenbroek return;
379*9f20bfa6SDavid van Moolenbroek }
380*9f20bfa6SDavid van Moolenbroek
381*9f20bfa6SDavid van Moolenbroek logger(ifp->ctx, LOG_INFO, "%s: probing for an IPv4LL address",
382*9f20bfa6SDavid van Moolenbroek ifp->name);
383*9f20bfa6SDavid van Moolenbroek astate->addr.s_addr = ipv4ll_pick_addr(astate);
384*9f20bfa6SDavid van Moolenbroek #ifdef IN_IFF_TENTATIVE
385*9f20bfa6SDavid van Moolenbroek ipv4ll_probed(astate);
386*9f20bfa6SDavid van Moolenbroek #else
387*9f20bfa6SDavid van Moolenbroek arp_probe(astate);
388*9f20bfa6SDavid van Moolenbroek #endif
389*9f20bfa6SDavid van Moolenbroek }
390*9f20bfa6SDavid van Moolenbroek
391*9f20bfa6SDavid van Moolenbroek void
ipv4ll_freedrop(struct interface * ifp,int drop)392*9f20bfa6SDavid van Moolenbroek ipv4ll_freedrop(struct interface *ifp, int drop)
393*9f20bfa6SDavid van Moolenbroek {
394*9f20bfa6SDavid van Moolenbroek struct ipv4ll_state *state;
395*9f20bfa6SDavid van Moolenbroek int dropped;
396*9f20bfa6SDavid van Moolenbroek
397*9f20bfa6SDavid van Moolenbroek assert(ifp != NULL);
398*9f20bfa6SDavid van Moolenbroek state = IPV4LL_STATE(ifp);
399*9f20bfa6SDavid van Moolenbroek dropped = 0;
400*9f20bfa6SDavid van Moolenbroek
401*9f20bfa6SDavid van Moolenbroek /* Free ARP state first because ipv4_deladdr might also ... */
402*9f20bfa6SDavid van Moolenbroek if (state && state->arp) {
403*9f20bfa6SDavid van Moolenbroek eloop_timeout_delete(ifp->ctx->eloop, NULL, state->arp);
404*9f20bfa6SDavid van Moolenbroek arp_free(state->arp);
405*9f20bfa6SDavid van Moolenbroek state->arp = NULL;
406*9f20bfa6SDavid van Moolenbroek }
407*9f20bfa6SDavid van Moolenbroek
408*9f20bfa6SDavid van Moolenbroek if (drop && (ifp->options->options & DHCPCD_NODROP) != DHCPCD_NODROP) {
409*9f20bfa6SDavid van Moolenbroek struct ipv4_state *istate;
410*9f20bfa6SDavid van Moolenbroek
411*9f20bfa6SDavid van Moolenbroek if (state && state->addr.s_addr != INADDR_ANY) {
412*9f20bfa6SDavid van Moolenbroek ipv4_deladdr(ifp, &state->addr, &inaddr_llmask, 1);
413*9f20bfa6SDavid van Moolenbroek state->addr.s_addr = INADDR_ANY;
414*9f20bfa6SDavid van Moolenbroek dropped = 1;
415*9f20bfa6SDavid van Moolenbroek }
416*9f20bfa6SDavid van Moolenbroek
417*9f20bfa6SDavid van Moolenbroek /* Free any other link local addresses that might exist. */
418*9f20bfa6SDavid van Moolenbroek if ((istate = IPV4_STATE(ifp)) != NULL) {
419*9f20bfa6SDavid van Moolenbroek struct ipv4_addr *ia, *ian;
420*9f20bfa6SDavid van Moolenbroek
421*9f20bfa6SDavid van Moolenbroek TAILQ_FOREACH_SAFE(ia, &istate->addrs, next, ian) {
422*9f20bfa6SDavid van Moolenbroek if (IN_LINKLOCAL(ntohl(ia->addr.s_addr))) {
423*9f20bfa6SDavid van Moolenbroek ipv4_deladdr(ifp, &ia->addr,
424*9f20bfa6SDavid van Moolenbroek &ia->net, 0);
425*9f20bfa6SDavid van Moolenbroek dropped = 1;
426*9f20bfa6SDavid van Moolenbroek }
427*9f20bfa6SDavid van Moolenbroek }
428*9f20bfa6SDavid van Moolenbroek }
429*9f20bfa6SDavid van Moolenbroek }
430*9f20bfa6SDavid van Moolenbroek
431*9f20bfa6SDavid van Moolenbroek if (state) {
432*9f20bfa6SDavid van Moolenbroek free(state);
433*9f20bfa6SDavid van Moolenbroek ifp->if_data[IF_DATA_IPV4LL] = NULL;
434*9f20bfa6SDavid van Moolenbroek
435*9f20bfa6SDavid van Moolenbroek if (dropped) {
436*9f20bfa6SDavid van Moolenbroek ipv4_buildroutes(ifp->ctx);
437*9f20bfa6SDavid van Moolenbroek script_runreason(ifp, "IPV4LL");
438*9f20bfa6SDavid van Moolenbroek }
439*9f20bfa6SDavid van Moolenbroek }
440*9f20bfa6SDavid van Moolenbroek }
441