xref: /openbsd-src/usr.sbin/ospf6d/interface.c (revision 5b133f3f277e80f096764111e64f3a1284acb179)
1*5b133f3fSguenther /*	$OpenBSD: interface.c,v 1.30 2023/03/08 04:43:14 guenther Exp $ */
2a1a4e97bSnorby 
3a1a4e97bSnorby /*
4a1a4e97bSnorby  * Copyright (c) 2005 Claudio Jeker <claudio@openbsd.org>
5a1a4e97bSnorby  * Copyright (c) 2004, 2005, 2007 Esben Norby <norby@openbsd.org>
6a1a4e97bSnorby  *
7a1a4e97bSnorby  * Permission to use, copy, modify, and distribute this software for any
8a1a4e97bSnorby  * purpose with or without fee is hereby granted, provided that the above
9a1a4e97bSnorby  * copyright notice and this permission notice appear in all copies.
10a1a4e97bSnorby  *
11a1a4e97bSnorby  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
12a1a4e97bSnorby  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
13a1a4e97bSnorby  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
14a1a4e97bSnorby  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
15a1a4e97bSnorby  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
16a1a4e97bSnorby  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
17a1a4e97bSnorby  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18a1a4e97bSnorby  */
19a1a4e97bSnorby 
20a1a4e97bSnorby #include <sys/types.h>
21a1a4e97bSnorby #include <sys/ioctl.h>
22a1a4e97bSnorby #include <sys/time.h>
23a1a4e97bSnorby #include <sys/socket.h>
24a1a4e97bSnorby #include <netinet/in.h>
25a1a4e97bSnorby #include <arpa/inet.h>
26a1a4e97bSnorby #include <net/if.h>
27a1a4e97bSnorby #include <net/if_types.h>
28a1a4e97bSnorby #include <ctype.h>
29a1a4e97bSnorby #include <err.h>
30a1a4e97bSnorby #include <stdio.h>
31a1a4e97bSnorby #include <stdlib.h>
32a1a4e97bSnorby #include <unistd.h>
33a1a4e97bSnorby #include <string.h>
34a1a4e97bSnorby #include <event.h>
35a1a4e97bSnorby 
36a1a4e97bSnorby #include "ospf6d.h"
37a1a4e97bSnorby #include "ospf6.h"
38a1a4e97bSnorby #include "log.h"
39a1a4e97bSnorby #include "ospfe.h"
40a1a4e97bSnorby 
41a1a4e97bSnorby void		 if_hello_timer(int, short, void *);
42a1a4e97bSnorby void		 if_start_hello_timer(struct iface *);
43a1a4e97bSnorby void		 if_stop_hello_timer(struct iface *);
44a1a4e97bSnorby void		 if_stop_wait_timer(struct iface *);
45a1a4e97bSnorby void		 if_wait_timer(int, short, void *);
46a1a4e97bSnorby void		 if_start_wait_timer(struct iface *);
47a1a4e97bSnorby void		 if_stop_wait_timer(struct iface *);
48a1a4e97bSnorby struct nbr	*if_elect(struct nbr *, struct nbr *);
49a1a4e97bSnorby 
50a1a4e97bSnorby struct {
51a1a4e97bSnorby 	int			state;
52a1a4e97bSnorby 	enum iface_event	event;
53a1a4e97bSnorby 	enum iface_action	action;
54a1a4e97bSnorby 	int			new_state;
55a1a4e97bSnorby } iface_fsm[] = {
56a1a4e97bSnorby     /* current state	event that happened	action to take	resulting state */
57a1a4e97bSnorby     {IF_STA_DOWN,	IF_EVT_UP,		IF_ACT_STRT,	0},
58a1a4e97bSnorby     {IF_STA_WAITING,	IF_EVT_BACKUP_SEEN,	IF_ACT_ELECT,	0},
59a1a4e97bSnorby     {IF_STA_WAITING,	IF_EVT_WTIMER,		IF_ACT_ELECT,	0},
60a1a4e97bSnorby     {IF_STA_ANY,	IF_EVT_WTIMER,		IF_ACT_NOTHING,	0},
61a1a4e97bSnorby     {IF_STA_WAITING,	IF_EVT_NBR_CHNG,	IF_ACT_NOTHING,	0},
62a1a4e97bSnorby     {IF_STA_MULTI,	IF_EVT_NBR_CHNG,	IF_ACT_ELECT,	0},
63a1a4e97bSnorby     {IF_STA_ANY,	IF_EVT_NBR_CHNG,	IF_ACT_NOTHING,	0},
64a1a4e97bSnorby     {IF_STA_ANY,	IF_EVT_DOWN,		IF_ACT_RST,	IF_STA_DOWN},
65a1a4e97bSnorby     {IF_STA_ANY,	IF_EVT_LOOP,		IF_ACT_RST,	IF_STA_LOOPBACK},
66a1a4e97bSnorby     {IF_STA_LOOPBACK,	IF_EVT_UNLOOP,		IF_ACT_NOTHING,	IF_STA_DOWN},
67a1a4e97bSnorby     {-1,		IF_EVT_NOTHING,		IF_ACT_NOTHING,	0},
68a1a4e97bSnorby };
69a1a4e97bSnorby 
7090a39e67Sstsp #if 0
7190a39e67Sstsp /* TODO virtual links */
72a1a4e97bSnorby static int vlink_cnt = 0;
7390a39e67Sstsp #endif
74a1a4e97bSnorby 
756c9e7a5bSclaudio TAILQ_HEAD(, iface)	iflist;
766c9e7a5bSclaudio 
77a1a4e97bSnorby const char * const if_event_names[] = {
78a1a4e97bSnorby 	"NOTHING",
79a1a4e97bSnorby 	"UP",
80a1a4e97bSnorby 	"WAITTIMER",
81a1a4e97bSnorby 	"BACKUPSEEN",
82a1a4e97bSnorby 	"NEIGHBORCHANGE",
83a1a4e97bSnorby 	"LOOP",
84a1a4e97bSnorby 	"UNLOOP",
85a1a4e97bSnorby 	"DOWN"
86a1a4e97bSnorby };
87a1a4e97bSnorby 
88a1a4e97bSnorby const char * const if_action_names[] = {
89a1a4e97bSnorby 	"NOTHING",
90a1a4e97bSnorby 	"START",
91a1a4e97bSnorby 	"ELECT",
92a1a4e97bSnorby 	"RESET"
93a1a4e97bSnorby };
94a1a4e97bSnorby 
95a1a4e97bSnorby int
if_fsm(struct iface * iface,enum iface_event event)96a1a4e97bSnorby if_fsm(struct iface *iface, enum iface_event event)
97a1a4e97bSnorby {
98a1a4e97bSnorby 	int	old_state;
99a1a4e97bSnorby 	int	new_state = 0;
100a1a4e97bSnorby 	int	i, ret = 0;
101a1a4e97bSnorby 
102a1a4e97bSnorby 	old_state = iface->state;
103a1a4e97bSnorby 
104a1a4e97bSnorby 	for (i = 0; iface_fsm[i].state != -1; i++)
105a1a4e97bSnorby 		if ((iface_fsm[i].state & old_state) &&
106a1a4e97bSnorby 		    (iface_fsm[i].event == event)) {
107a1a4e97bSnorby 			new_state = iface_fsm[i].new_state;
108a1a4e97bSnorby 			break;
109a1a4e97bSnorby 		}
110a1a4e97bSnorby 
111a1a4e97bSnorby 	if (iface_fsm[i].state == -1) {
112a1a4e97bSnorby 		/* event outside of the defined fsm, ignore it. */
113a1a4e97bSnorby 		log_debug("if_fsm: interface %s, "
114a1a4e97bSnorby 		    "event %s not expected in state %s", iface->name,
115a1a4e97bSnorby 		    if_event_names[event], if_state_name(old_state));
116a1a4e97bSnorby 		return (0);
117a1a4e97bSnorby 	}
118a1a4e97bSnorby 
119a1a4e97bSnorby 	switch (iface_fsm[i].action) {
120a1a4e97bSnorby 	case IF_ACT_STRT:
121a1a4e97bSnorby 		ret = if_act_start(iface);
122a1a4e97bSnorby 		break;
123a1a4e97bSnorby 	case IF_ACT_ELECT:
124a1a4e97bSnorby 		ret = if_act_elect(iface);
125a1a4e97bSnorby 		break;
126a1a4e97bSnorby 	case IF_ACT_RST:
127a1a4e97bSnorby 		ret = if_act_reset(iface);
128a1a4e97bSnorby 		break;
129a1a4e97bSnorby 	case IF_ACT_NOTHING:
130a1a4e97bSnorby 		/* do nothing */
131a1a4e97bSnorby 		break;
132a1a4e97bSnorby 	}
133a1a4e97bSnorby 
134a1a4e97bSnorby 	if (ret) {
135a1a4e97bSnorby 		log_debug("if_fsm: error changing state for interface %s, "
136a1a4e97bSnorby 		    "event %s, state %s", iface->name, if_event_names[event],
137a1a4e97bSnorby 		    if_state_name(old_state));
138a1a4e97bSnorby 		return (-1);
139a1a4e97bSnorby 	}
140a1a4e97bSnorby 
141a1a4e97bSnorby 	if (new_state != 0)
142a1a4e97bSnorby 		iface->state = new_state;
143a1a4e97bSnorby 
144d913b22eSclaudio 	if (iface->state != old_state) {
14577fbfa19Sdenis 		area_track(iface->area);
146d18517d2Sdenis 		orig_rtr_lsa(iface->area);
147d913b22eSclaudio 		orig_link_lsa(iface);
1482bc51794Sstsp 
14942176c47Sclaudio 		/* state change inform RDE */
15042176c47Sclaudio 		ospfe_imsg_compose_rde(IMSG_IFINFO, iface->self->peerid, 0,
15142176c47Sclaudio 		    &iface->state, sizeof(iface->state));
15242176c47Sclaudio 	}
153a1a4e97bSnorby 
154a1a4e97bSnorby 	if (old_state & (IF_STA_MULTI | IF_STA_POINTTOPOINT) &&
155a1a4e97bSnorby 	    (iface->state & (IF_STA_MULTI | IF_STA_POINTTOPOINT)) == 0)
156a1a4e97bSnorby 		ospfe_demote_iface(iface, 0);
157a1a4e97bSnorby 	if ((old_state & (IF_STA_MULTI | IF_STA_POINTTOPOINT)) == 0 &&
158a1a4e97bSnorby 	    iface->state & (IF_STA_MULTI | IF_STA_POINTTOPOINT))
159a1a4e97bSnorby 		ospfe_demote_iface(iface, 1);
160a1a4e97bSnorby 
161a1a4e97bSnorby 	log_debug("if_fsm: event %s resulted in action %s and changing "
162a1a4e97bSnorby 	    "state for interface %s from %s to %s",
163a1a4e97bSnorby 	    if_event_names[event], if_action_names[iface_fsm[i].action],
164a1a4e97bSnorby 	    iface->name, if_state_name(old_state), if_state_name(iface->state));
165a1a4e97bSnorby 
166a1a4e97bSnorby 	return (ret);
167a1a4e97bSnorby }
168a1a4e97bSnorby 
1696c9e7a5bSclaudio int
if_init(void)1706c9e7a5bSclaudio if_init(void)
1716c9e7a5bSclaudio {
1726c9e7a5bSclaudio 	TAILQ_INIT(&iflist);
1736c9e7a5bSclaudio 
1746c9e7a5bSclaudio 	return (fetchifs(0));
1756c9e7a5bSclaudio }
1766c9e7a5bSclaudio 
1776c9e7a5bSclaudio /* XXX using a linked list should be OK for now */
178a1a4e97bSnorby struct iface *
if_find(unsigned int ifindex)1796c9e7a5bSclaudio if_find(unsigned int ifindex)
1806c9e7a5bSclaudio {
1816c9e7a5bSclaudio 	struct iface	*iface;
1826c9e7a5bSclaudio 
1836c9e7a5bSclaudio 	TAILQ_FOREACH(iface, &iflist, list) {
1846c9e7a5bSclaudio 		if (ifindex == iface->ifindex)
1856c9e7a5bSclaudio 			return (iface);
1866c9e7a5bSclaudio 	}
1876c9e7a5bSclaudio 	return (NULL);
1886c9e7a5bSclaudio }
1896c9e7a5bSclaudio 
1906c9e7a5bSclaudio struct iface *
if_findname(char * name)1916c9e7a5bSclaudio if_findname(char *name)
1926c9e7a5bSclaudio {
1936c9e7a5bSclaudio 	struct iface	*iface;
1946c9e7a5bSclaudio 
1956c9e7a5bSclaudio 	TAILQ_FOREACH(iface, &iflist, list) {
1966c9e7a5bSclaudio 		if (!strcmp(name, iface->name))
1976c9e7a5bSclaudio 			return (iface);
1986c9e7a5bSclaudio 	}
1996c9e7a5bSclaudio 	return (NULL);
2006c9e7a5bSclaudio }
2016c9e7a5bSclaudio 
2026c9e7a5bSclaudio struct iface *
if_new(u_short ifindex,char * ifname)2036c9e7a5bSclaudio if_new(u_short ifindex, char *ifname)
204a1a4e97bSnorby {
205a1a4e97bSnorby 	struct iface		*iface;
206a1a4e97bSnorby 
207a1a4e97bSnorby 	if ((iface = calloc(1, sizeof(*iface))) == NULL)
208a1a4e97bSnorby 		err(1, "if_new: calloc");
209a1a4e97bSnorby 
210a1a4e97bSnorby 	iface->state = IF_STA_DOWN;
211a1a4e97bSnorby 
212a1a4e97bSnorby 	LIST_INIT(&iface->nbr_list);
2136c9e7a5bSclaudio 	TAILQ_INIT(&iface->ifa_list);
214a1a4e97bSnorby 	TAILQ_INIT(&iface->ls_ack_list);
2154139c605Sclaudio 	RB_INIT(&iface->lsa_tree);
216a1a4e97bSnorby 
21790a39e67Sstsp #if 0
21890a39e67Sstsp 	/* TODO */
21990a39e67Sstsp 	if (virtual) {
220a1a4e97bSnorby 		iface->type = IF_TYPE_VIRTUALLINK;
221a1a4e97bSnorby 		snprintf(iface->name, sizeof(iface->name), "vlink%d",
222a1a4e97bSnorby 		    vlink_cnt++);
223a1a4e97bSnorby 		iface->flags |= IFF_UP;
224a1a4e97bSnorby 		iface->mtu = IP_MSS;
225a1a4e97bSnorby 		return (iface);
226a1a4e97bSnorby 	}
22790a39e67Sstsp #endif
2286c9e7a5bSclaudio 	strlcpy(iface->name, ifname, sizeof(iface->name));
2296c9e7a5bSclaudio 	iface->ifindex = ifindex;
230a1a4e97bSnorby 
2316c9e7a5bSclaudio 	TAILQ_INSERT_TAIL(&iflist, iface, list);
2326c9e7a5bSclaudio 
2336c9e7a5bSclaudio 	return (iface);
2346c9e7a5bSclaudio }
2356c9e7a5bSclaudio 
2366c9e7a5bSclaudio void
if_update(struct iface * iface,int mtu,int flags,u_int8_t type,u_int8_t state,u_int64_t rate,u_int32_t rdomain)2376c9e7a5bSclaudio if_update(struct iface *iface, int mtu, int flags, u_int8_t type,
2385d393f89Sremi     u_int8_t state, u_int64_t rate, u_int32_t rdomain)
2396c9e7a5bSclaudio {
2406c9e7a5bSclaudio 	iface->mtu = mtu;
2416c9e7a5bSclaudio 	iface->flags = flags;
24218ffdd94Sstsp 	iface->if_type = type;
2436c9e7a5bSclaudio 	iface->linkstate = state;
2446c9e7a5bSclaudio 	iface->baudrate = rate;
2455d393f89Sremi 	iface->rdomain = rdomain;
2466c9e7a5bSclaudio 
2476c9e7a5bSclaudio 	/* set type */
2486c9e7a5bSclaudio 	if (flags & IFF_POINTOPOINT)
249a1a4e97bSnorby 		iface->type = IF_TYPE_POINTOPOINT;
2506c9e7a5bSclaudio 	if (flags & IFF_BROADCAST && flags & IFF_MULTICAST)
251a1a4e97bSnorby 		iface->type = IF_TYPE_BROADCAST;
2526c9e7a5bSclaudio 	if (flags & IFF_LOOPBACK) {
253a1a4e97bSnorby 		iface->type = IF_TYPE_POINTOPOINT;
2540dc8d668Sclaudio 		iface->cflags |= F_IFACE_PASSIVE;
255a1a4e97bSnorby 	}
256a1a4e97bSnorby }
257a1a4e97bSnorby 
258a1a4e97bSnorby void
if_del(struct iface * iface)259a1a4e97bSnorby if_del(struct iface *iface)
260a1a4e97bSnorby {
261a1a4e97bSnorby 	struct nbr	*nbr = NULL;
262a1a4e97bSnorby 
263a1a4e97bSnorby 	log_debug("if_del: interface %s", iface->name);
264a1a4e97bSnorby 
265a1a4e97bSnorby 	/* revert the demotion when the interface is deleted */
266a1a4e97bSnorby 	if ((iface->state & (IF_STA_MULTI | IF_STA_POINTTOPOINT)) == 0)
267a1a4e97bSnorby 		ospfe_demote_iface(iface, 1);
268a1a4e97bSnorby 
269a1a4e97bSnorby 	/* clear lists etc */
270a1a4e97bSnorby 	while ((nbr = LIST_FIRST(&iface->nbr_list)) != NULL)
271a1a4e97bSnorby 		nbr_del(nbr);
272a1a4e97bSnorby 
273a1a4e97bSnorby 	if (evtimer_pending(&iface->hello_timer, NULL))
274a1a4e97bSnorby 		evtimer_del(&iface->hello_timer);
275a1a4e97bSnorby 	if (evtimer_pending(&iface->wait_timer, NULL))
276a1a4e97bSnorby 		evtimer_del(&iface->wait_timer);
277a1a4e97bSnorby 	if (evtimer_pending(&iface->lsack_tx_timer, NULL))
278a1a4e97bSnorby 		evtimer_del(&iface->lsack_tx_timer);
279a1a4e97bSnorby 
280a1a4e97bSnorby 	ls_ack_list_clr(iface);
2816c9e7a5bSclaudio 	TAILQ_REMOVE(&iflist, iface, list);
282a1a4e97bSnorby 	free(iface);
283a1a4e97bSnorby }
284a1a4e97bSnorby 
285a1a4e97bSnorby void
if_start(struct ospfd_conf * xconf,struct iface * iface)2866c9e7a5bSclaudio if_start(struct ospfd_conf *xconf, struct iface *iface)
287a1a4e97bSnorby {
288a1a4e97bSnorby 	/* init the dummy local neighbor */
289d2afa435Sstsp 	iface->self = nbr_new(ospfe_router_id(), iface, iface->ifindex, 1,
290d2afa435Sstsp 			NULL);
291a1a4e97bSnorby 
292a1a4e97bSnorby 	/* set event handlers for interface */
293a1a4e97bSnorby 	evtimer_set(&iface->lsack_tx_timer, ls_ack_tx_timer, iface);
294a1a4e97bSnorby 	evtimer_set(&iface->hello_timer, if_hello_timer, iface);
295a1a4e97bSnorby 	evtimer_set(&iface->wait_timer, if_wait_timer, iface);
296a1a4e97bSnorby 
297a1a4e97bSnorby 	iface->fd = xconf->ospf_socket;
298a1a4e97bSnorby 
299a1a4e97bSnorby 	ospfe_demote_iface(iface, 0);
3006c9e7a5bSclaudio 
3016c9e7a5bSclaudio 	if (if_fsm(iface, IF_EVT_UP))
3026c9e7a5bSclaudio 		log_debug("error starting interface %s", iface->name);
303a1a4e97bSnorby }
304a1a4e97bSnorby 
305a1a4e97bSnorby /* timers */
306a1a4e97bSnorby void
if_hello_timer(int fd,short event,void * arg)307a1a4e97bSnorby if_hello_timer(int fd, short event, void *arg)
308a1a4e97bSnorby {
309a1a4e97bSnorby 	struct iface *iface = arg;
310a1a4e97bSnorby 	struct timeval tv;
311a1a4e97bSnorby 
312a1a4e97bSnorby 	send_hello(iface);
313a1a4e97bSnorby 
314a1a4e97bSnorby 	/* reschedule hello_timer */
315a1a4e97bSnorby 	timerclear(&tv);
316a1a4e97bSnorby 	tv.tv_sec = iface->hello_interval;
317a1a4e97bSnorby 	if (evtimer_add(&iface->hello_timer, &tv) == -1)
318a1a4e97bSnorby 		fatal("if_hello_timer");
319a1a4e97bSnorby }
320a1a4e97bSnorby 
321a1a4e97bSnorby void
if_start_hello_timer(struct iface * iface)322a1a4e97bSnorby if_start_hello_timer(struct iface *iface)
323a1a4e97bSnorby {
324a1a4e97bSnorby 	struct timeval tv;
325a1a4e97bSnorby 
326a1a4e97bSnorby 	timerclear(&tv);
327a1a4e97bSnorby 	if (evtimer_add(&iface->hello_timer, &tv) == -1)
328a1a4e97bSnorby 		fatal("if_start_hello_timer");
329a1a4e97bSnorby }
330a1a4e97bSnorby 
331a1a4e97bSnorby void
if_stop_hello_timer(struct iface * iface)332a1a4e97bSnorby if_stop_hello_timer(struct iface *iface)
333a1a4e97bSnorby {
334a1a4e97bSnorby 	if (evtimer_del(&iface->hello_timer) == -1)
335a1a4e97bSnorby 		fatal("if_stop_hello_timer");
336a1a4e97bSnorby }
337a1a4e97bSnorby 
338a1a4e97bSnorby void
if_wait_timer(int fd,short event,void * arg)339a1a4e97bSnorby if_wait_timer(int fd, short event, void *arg)
340a1a4e97bSnorby {
341a1a4e97bSnorby 	struct iface *iface = arg;
342a1a4e97bSnorby 
343a1a4e97bSnorby 	if_fsm(iface, IF_EVT_WTIMER);
344a1a4e97bSnorby }
345a1a4e97bSnorby 
346a1a4e97bSnorby void
if_start_wait_timer(struct iface * iface)347a1a4e97bSnorby if_start_wait_timer(struct iface *iface)
348a1a4e97bSnorby {
349a1a4e97bSnorby 	struct timeval	tv;
350a1a4e97bSnorby 
351a1a4e97bSnorby 	timerclear(&tv);
352a1a4e97bSnorby 	tv.tv_sec = iface->dead_interval;
353a1a4e97bSnorby 	if (evtimer_add(&iface->wait_timer, &tv) == -1)
354a1a4e97bSnorby 		fatal("if_start_wait_timer");
355a1a4e97bSnorby }
356a1a4e97bSnorby 
357a1a4e97bSnorby void
if_stop_wait_timer(struct iface * iface)358a1a4e97bSnorby if_stop_wait_timer(struct iface *iface)
359a1a4e97bSnorby {
360a1a4e97bSnorby 	if (evtimer_del(&iface->wait_timer) == -1)
361a1a4e97bSnorby 		fatal("if_stop_wait_timer");
362a1a4e97bSnorby }
363a1a4e97bSnorby 
364a1a4e97bSnorby /* actions */
365a1a4e97bSnorby int
if_act_start(struct iface * iface)366a1a4e97bSnorby if_act_start(struct iface *iface)
367a1a4e97bSnorby {
368a1a4e97bSnorby 	struct in6_addr		 addr;
369a1a4e97bSnorby 	struct timeval		 now;
370a1a4e97bSnorby 
371a1a4e97bSnorby 	if (!((iface->flags & IFF_UP) &&
3729a2e0324Sclaudio 	    LINK_STATE_IS_UP(iface->linkstate))) {
373a1a4e97bSnorby 		log_debug("if_act_start: interface %s link down",
374a1a4e97bSnorby 		    iface->name);
375a1a4e97bSnorby 		return (0);
376a1a4e97bSnorby 	}
377a1a4e97bSnorby 
37818ffdd94Sstsp 	if (iface->if_type == IFT_CARP &&
3796c9e7a5bSclaudio 	    !(iface->cflags & F_IFACE_PASSIVE)) {
380a1a4e97bSnorby 		/* force passive mode on carp interfaces */
381a1a4e97bSnorby 		log_warnx("if_act_start: forcing interface %s to passive",
382a1a4e97bSnorby 		    iface->name);
3836c9e7a5bSclaudio 		iface->cflags |= F_IFACE_PASSIVE;
384a1a4e97bSnorby 	}
385a1a4e97bSnorby 
3860dc8d668Sclaudio 	gettimeofday(&now, NULL);
3870dc8d668Sclaudio 	iface->uptime = now.tv_sec;
3880dc8d668Sclaudio 
3890dc8d668Sclaudio 	/* loopback interfaces have a special state */
3900dc8d668Sclaudio 	if (iface->flags & IFF_LOOPBACK)
3910dc8d668Sclaudio 		iface->state = IF_STA_LOOPBACK;
3920dc8d668Sclaudio 
3936c9e7a5bSclaudio 	if (iface->cflags & F_IFACE_PASSIVE) {
394a1a4e97bSnorby 		/* for an update of stub network entries */
395d18517d2Sdenis 		orig_rtr_lsa(iface->area);
396a1a4e97bSnorby 		return (0);
397a1a4e97bSnorby 	}
398a1a4e97bSnorby 
399a1a4e97bSnorby 	switch (iface->type) {
400a1a4e97bSnorby 	case IF_TYPE_POINTOPOINT:
401a1a4e97bSnorby 		inet_pton(AF_INET6, AllSPFRouters, &addr);
402a1a4e97bSnorby 
403a1a4e97bSnorby 		if (if_join_group(iface, &addr))
404a1a4e97bSnorby 			return (-1);
405a1a4e97bSnorby 		iface->state = IF_STA_POINTTOPOINT;
406a1a4e97bSnorby 		break;
407a1a4e97bSnorby 	case IF_TYPE_VIRTUALLINK:
408a1a4e97bSnorby 		iface->state = IF_STA_POINTTOPOINT;
409a1a4e97bSnorby 		break;
410a1a4e97bSnorby 	case IF_TYPE_POINTOMULTIPOINT:
411a1a4e97bSnorby 	case IF_TYPE_NBMA:
412a1a4e97bSnorby 		log_debug("if_act_start: type %s not supported, interface %s",
413a1a4e97bSnorby 		    if_type_name(iface->type), iface->name);
414a1a4e97bSnorby 		return (-1);
415a1a4e97bSnorby 	case IF_TYPE_BROADCAST:
416a1a4e97bSnorby 		inet_pton(AF_INET6, AllSPFRouters, &addr);
417a1a4e97bSnorby 
418a1a4e97bSnorby 		if (if_join_group(iface, &addr))
419a1a4e97bSnorby 			return (-1);
420a1a4e97bSnorby 		if (iface->priority == 0) {
421a1a4e97bSnorby 			iface->state = IF_STA_DROTHER;
422a1a4e97bSnorby 		} else {
423a1a4e97bSnorby 			iface->state = IF_STA_WAITING;
424a1a4e97bSnorby 			if_start_wait_timer(iface);
425a1a4e97bSnorby 		}
426a1a4e97bSnorby 		break;
427a1a4e97bSnorby 	default:
428a1a4e97bSnorby 		fatalx("if_act_start: unknown interface type");
429a1a4e97bSnorby 	}
430a1a4e97bSnorby 
431a1a4e97bSnorby 	/* hello timer needs to be started in any case */
432a1a4e97bSnorby 	if_start_hello_timer(iface);
433a1a4e97bSnorby 	return (0);
434a1a4e97bSnorby }
435a1a4e97bSnorby 
436a1a4e97bSnorby struct nbr *
if_elect(struct nbr * a,struct nbr * b)437a1a4e97bSnorby if_elect(struct nbr *a, struct nbr *b)
438a1a4e97bSnorby {
439a1a4e97bSnorby 	if (a->priority > b->priority)
440a1a4e97bSnorby 		return (a);
441a1a4e97bSnorby 	if (a->priority < b->priority)
442a1a4e97bSnorby 		return (b);
443a1a4e97bSnorby 	if (ntohl(a->id.s_addr) > ntohl(b->id.s_addr))
444a1a4e97bSnorby 		return (a);
445a1a4e97bSnorby 	return (b);
446a1a4e97bSnorby }
447a1a4e97bSnorby 
448a1a4e97bSnorby int
if_act_elect(struct iface * iface)449a1a4e97bSnorby if_act_elect(struct iface *iface)
450a1a4e97bSnorby {
451a1a4e97bSnorby 	struct in6_addr	 addr;
452a1a4e97bSnorby 	struct nbr	*nbr, *bdr = NULL, *dr = NULL;
453a1a4e97bSnorby 	int		 round = 0;
454a1a4e97bSnorby 	int		 changed = 0;
455a1a4e97bSnorby 	int		 old_state;
456a1a4e97bSnorby 	char		 b1[16], b2[16], b3[16], b4[16];
457a1a4e97bSnorby 
458a1a4e97bSnorby start:
459a1a4e97bSnorby 	/* elect backup designated router */
460a1a4e97bSnorby 	LIST_FOREACH(nbr, &iface->nbr_list, entry) {
461a1a4e97bSnorby 		if (nbr->priority == 0 || nbr == dr ||	/* not electable */
462a1a4e97bSnorby 		    nbr->state & NBR_STA_PRELIM ||	/* not available */
463a1a4e97bSnorby 		    nbr->dr.s_addr == nbr->id.s_addr)	/* don't elect DR */
464a1a4e97bSnorby 			continue;
465a1a4e97bSnorby 		if (bdr != NULL) {
466a1a4e97bSnorby 			/*
467a1a4e97bSnorby 			 * routers announcing themselves as BDR have higher
468a1a4e97bSnorby 			 * precedence over those routers announcing a
469a1a4e97bSnorby 			 * different BDR.
470a1a4e97bSnorby 			 */
471a1a4e97bSnorby 			if (nbr->bdr.s_addr == nbr->id.s_addr) {
472a1a4e97bSnorby 				if (bdr->bdr.s_addr == bdr->id.s_addr)
473a1a4e97bSnorby 					bdr = if_elect(bdr, nbr);
474a1a4e97bSnorby 				else
475a1a4e97bSnorby 					bdr = nbr;
476a1a4e97bSnorby 			} else if (bdr->bdr.s_addr != bdr->id.s_addr)
477a1a4e97bSnorby 					bdr = if_elect(bdr, nbr);
478a1a4e97bSnorby 		} else
479a1a4e97bSnorby 			bdr = nbr;
480a1a4e97bSnorby 	}
481a1a4e97bSnorby 
482a1a4e97bSnorby 	/* elect designated router */
483a1a4e97bSnorby 	LIST_FOREACH(nbr, &iface->nbr_list, entry) {
484a1a4e97bSnorby 		if (nbr->priority == 0 || nbr->state & NBR_STA_PRELIM ||
485a1a4e97bSnorby 		    (nbr != dr && nbr->dr.s_addr != nbr->id.s_addr))
486a1a4e97bSnorby 			/* only DR may be elected check priority too */
487a1a4e97bSnorby 			continue;
488a1a4e97bSnorby 		if (dr == NULL)
489a1a4e97bSnorby 			dr = nbr;
490a1a4e97bSnorby 		else
491a1a4e97bSnorby 			dr = if_elect(dr, nbr);
492a1a4e97bSnorby 	}
493a1a4e97bSnorby 
494a1a4e97bSnorby 	if (dr == NULL) {
495a1a4e97bSnorby 		/* no designate router found use backup DR */
496a1a4e97bSnorby 		dr = bdr;
497a1a4e97bSnorby 		bdr = NULL;
498a1a4e97bSnorby 	}
499a1a4e97bSnorby 
500a1a4e97bSnorby 	/*
501a1a4e97bSnorby 	 * if we are involved in the election (e.g. new DR or no
502a1a4e97bSnorby 	 * longer BDR) redo the election
503a1a4e97bSnorby 	 */
504a1a4e97bSnorby 	if (round == 0 &&
505a1a4e97bSnorby 	    ((iface->self == dr && iface->self != iface->dr) ||
506a1a4e97bSnorby 	    (iface->self != dr && iface->self == iface->dr) ||
507a1a4e97bSnorby 	    (iface->self == bdr && iface->self != iface->bdr) ||
508a1a4e97bSnorby 	    (iface->self != bdr && iface->self == iface->bdr))) {
509a1a4e97bSnorby 		/*
510a1a4e97bSnorby 		 * Reset announced DR/BDR to calculated one, so
511a1a4e97bSnorby 		 * that we may get elected in the second round.
512a1a4e97bSnorby 		 * This is needed to drop from a DR to a BDR.
513a1a4e97bSnorby 		 */
514a1a4e97bSnorby 		iface->self->dr.s_addr = dr->id.s_addr;
515a1a4e97bSnorby 		if (bdr)
516a1a4e97bSnorby 			iface->self->bdr.s_addr = bdr->id.s_addr;
517a1a4e97bSnorby 		round = 1;
518a1a4e97bSnorby 		goto start;
519a1a4e97bSnorby 	}
520a1a4e97bSnorby 
521a1a4e97bSnorby 	log_debug("if_act_elect: interface %s old dr %s new dr %s, "
522a1a4e97bSnorby 	    "old bdr %s new bdr %s", iface->name,
5233897a748Sclaudio 	    iface->dr ? inet_ntop(AF_INET, &iface->dr->id, b1, sizeof(b1)) :
5243897a748Sclaudio 	    "none", dr ? inet_ntop(AF_INET, &dr->id, b2, sizeof(b2)) : "none",
5253897a748Sclaudio 	    iface->bdr ? inet_ntop(AF_INET, &iface->bdr->id, b3, sizeof(b3)) :
5263897a748Sclaudio 	    "none", bdr ? inet_ntop(AF_INET, &bdr->id, b4, sizeof(b4)) :
527a1a4e97bSnorby 	    "none");
528a1a4e97bSnorby 
529a1a4e97bSnorby 	/*
530a1a4e97bSnorby 	 * After the second round still DR or BDR change state to DR or BDR,
531a1a4e97bSnorby 	 * etc.
532a1a4e97bSnorby 	 */
533a1a4e97bSnorby 	old_state = iface->state;
534a1a4e97bSnorby 	if (dr == iface->self)
535a1a4e97bSnorby 		iface->state = IF_STA_DR;
536a1a4e97bSnorby 	else if (bdr == iface->self)
537a1a4e97bSnorby 		iface->state = IF_STA_BACKUP;
538a1a4e97bSnorby 	else
539a1a4e97bSnorby 		iface->state = IF_STA_DROTHER;
540a1a4e97bSnorby 
541a1a4e97bSnorby 	/* TODO if iface is NBMA send all non eligible neighbors event Start */
542a1a4e97bSnorby 
543a1a4e97bSnorby 	/*
544a1a4e97bSnorby 	 * if DR or BDR changed issue a AdjOK? event for all neighbors > 2-Way
545a1a4e97bSnorby 	 */
546a1a4e97bSnorby 	if (iface->dr != dr || iface->bdr != bdr)
547a1a4e97bSnorby 		changed = 1;
548a1a4e97bSnorby 
549a1a4e97bSnorby 	iface->dr = dr;
550a1a4e97bSnorby 	iface->bdr = bdr;
551a1a4e97bSnorby 
552a1a4e97bSnorby 	if (changed) {
553a1a4e97bSnorby 		inet_pton(AF_INET6, AllDRouters, &addr);
554a1a4e97bSnorby 		if (old_state & IF_STA_DRORBDR &&
555a1a4e97bSnorby 		    (iface->state & IF_STA_DRORBDR) == 0) {
556a1a4e97bSnorby 			if (if_leave_group(iface, &addr))
557a1a4e97bSnorby 				return (-1);
558a1a4e97bSnorby 		} else if ((old_state & IF_STA_DRORBDR) == 0 &&
559a1a4e97bSnorby 		    iface->state & IF_STA_DRORBDR) {
560a1a4e97bSnorby 			if (if_join_group(iface, &addr))
561a1a4e97bSnorby 				return (-1);
562a1a4e97bSnorby 		}
563a1a4e97bSnorby 
564a1a4e97bSnorby 		LIST_FOREACH(nbr, &iface->nbr_list, entry) {
565a1a4e97bSnorby 			if (nbr->state & NBR_STA_BIDIR)
566a1a4e97bSnorby 				nbr_fsm(nbr, NBR_EVT_ADJ_OK);
567a1a4e97bSnorby 		}
568a1a4e97bSnorby 
569d18517d2Sdenis 		orig_rtr_lsa(iface->area);
570a1a4e97bSnorby 		if (iface->state & IF_STA_DR || old_state & IF_STA_DR)
571a1a4e97bSnorby 			orig_net_lsa(iface);
572a1a4e97bSnorby 	}
573a1a4e97bSnorby 
574a1a4e97bSnorby 	if_start_hello_timer(iface);
575a1a4e97bSnorby 	return (0);
576a1a4e97bSnorby }
577a1a4e97bSnorby 
578a1a4e97bSnorby int
if_act_reset(struct iface * iface)579a1a4e97bSnorby if_act_reset(struct iface *iface)
580a1a4e97bSnorby {
581a1a4e97bSnorby 	struct nbr		*nbr = NULL;
582a1a4e97bSnorby 	struct in6_addr		 addr;
583a1a4e97bSnorby 
5846c9e7a5bSclaudio 	if (iface->cflags & F_IFACE_PASSIVE) {
585a1a4e97bSnorby 		/* for an update of stub network entries */
586d18517d2Sdenis 		orig_rtr_lsa(iface->area);
587a1a4e97bSnorby 		return (0);
588a1a4e97bSnorby 	}
589a1a4e97bSnorby 
590a1a4e97bSnorby 	switch (iface->type) {
591a1a4e97bSnorby 	case IF_TYPE_POINTOPOINT:
592a1a4e97bSnorby 	case IF_TYPE_BROADCAST:
593a1a4e97bSnorby 		inet_pton(AF_INET6, AllSPFRouters, &addr);
594a1a4e97bSnorby 		if (if_leave_group(iface, &addr)) {
595a1a4e97bSnorby 			log_warnx("if_act_reset: error leaving group %s, "
596a1a4e97bSnorby 			    "interface %s", log_in6addr(&addr), iface->name);
597a1a4e97bSnorby 		}
598a1a4e97bSnorby 		if (iface->state & IF_STA_DRORBDR) {
599a1a4e97bSnorby 			inet_pton(AF_INET6, AllDRouters, &addr);
600a1a4e97bSnorby 			if (if_leave_group(iface, &addr)) {
601a1a4e97bSnorby 				log_warnx("if_act_reset: "
602a1a4e97bSnorby 				    "error leaving group %s, interface %s",
603a1a4e97bSnorby 				    log_in6addr(&addr), iface->name);
604a1a4e97bSnorby 			}
605a1a4e97bSnorby 		}
606a1a4e97bSnorby 		break;
607a1a4e97bSnorby 	case IF_TYPE_VIRTUALLINK:
608a1a4e97bSnorby 		/* nothing */
609a1a4e97bSnorby 		break;
610a1a4e97bSnorby 	case IF_TYPE_NBMA:
611a1a4e97bSnorby 	case IF_TYPE_POINTOMULTIPOINT:
612a1a4e97bSnorby 		log_debug("if_act_reset: type %s not supported, interface %s",
613a1a4e97bSnorby 		    if_type_name(iface->type), iface->name);
614a1a4e97bSnorby 		return (-1);
615a1a4e97bSnorby 	default:
616a1a4e97bSnorby 		fatalx("if_act_reset: unknown interface type");
617a1a4e97bSnorby 	}
618a1a4e97bSnorby 
619a1a4e97bSnorby 	LIST_FOREACH(nbr, &iface->nbr_list, entry) {
620a1a4e97bSnorby 		if (nbr_fsm(nbr, NBR_EVT_KILL_NBR)) {
621a1a4e97bSnorby 			log_debug("if_act_reset: error killing neighbor %s",
622a1a4e97bSnorby 			    inet_ntoa(nbr->id));
623a1a4e97bSnorby 		}
624a1a4e97bSnorby 	}
625a1a4e97bSnorby 
626a1a4e97bSnorby 	iface->dr = NULL;
627a1a4e97bSnorby 	iface->bdr = NULL;
628a1a4e97bSnorby 
629a1a4e97bSnorby 	ls_ack_list_clr(iface);
630a1a4e97bSnorby 	stop_ls_ack_tx_timer(iface);
631a1a4e97bSnorby 	if_stop_hello_timer(iface);
632a1a4e97bSnorby 	if_stop_wait_timer(iface);
633a1a4e97bSnorby 
634a1a4e97bSnorby 	/* send empty hello to tell everybody that we are going down */
635a1a4e97bSnorby 	send_hello(iface);
636a1a4e97bSnorby 
637a1a4e97bSnorby 	return (0);
638a1a4e97bSnorby }
639a1a4e97bSnorby 
640a1a4e97bSnorby struct ctl_iface *
if_to_ctl(struct iface * iface)641a1a4e97bSnorby if_to_ctl(struct iface *iface)
642a1a4e97bSnorby {
643a1a4e97bSnorby 	static struct ctl_iface	 ictl;
644a1a4e97bSnorby 	struct timeval		 tv, now, res;
645a1a4e97bSnorby 	struct nbr		*nbr;
646a1a4e97bSnorby 
647a1a4e97bSnorby 	memcpy(ictl.name, iface->name, sizeof(ictl.name));
648a1a4e97bSnorby 	memcpy(&ictl.addr, &iface->addr, sizeof(ictl.addr));
649a1a4e97bSnorby 	ictl.rtr_id.s_addr = ospfe_router_id();
65077fbfa19Sdenis 	memcpy(&ictl.area, &iface->area->id, sizeof(ictl.area));
651a1a4e97bSnorby 	if (iface->dr) {
652a1a4e97bSnorby 		memcpy(&ictl.dr_id, &iface->dr->id, sizeof(ictl.dr_id));
653a1a4e97bSnorby 		memcpy(&ictl.dr_addr, &iface->dr->addr, sizeof(ictl.dr_addr));
654a1a4e97bSnorby 	} else {
655a1a4e97bSnorby 		bzero(&ictl.dr_id, sizeof(ictl.dr_id));
656a1a4e97bSnorby 		bzero(&ictl.dr_addr, sizeof(ictl.dr_addr));
657a1a4e97bSnorby 	}
658a1a4e97bSnorby 	if (iface->bdr) {
659a1a4e97bSnorby 		memcpy(&ictl.bdr_id, &iface->bdr->id, sizeof(ictl.bdr_id));
660a1a4e97bSnorby 		memcpy(&ictl.bdr_addr, &iface->bdr->addr,
661a1a4e97bSnorby 		    sizeof(ictl.bdr_addr));
662a1a4e97bSnorby 	} else {
663a1a4e97bSnorby 		bzero(&ictl.bdr_id, sizeof(ictl.bdr_id));
664a1a4e97bSnorby 		bzero(&ictl.bdr_addr, sizeof(ictl.bdr_addr));
665a1a4e97bSnorby 	}
666a1a4e97bSnorby 	ictl.ifindex = iface->ifindex;
667a1a4e97bSnorby 	ictl.state = iface->state;
668a1a4e97bSnorby 	ictl.mtu = iface->mtu;
669a1a4e97bSnorby 	ictl.nbr_cnt = 0;
670a1a4e97bSnorby 	ictl.adj_cnt = 0;
671a1a4e97bSnorby 	ictl.baudrate = iface->baudrate;
672a1a4e97bSnorby 	ictl.dead_interval = iface->dead_interval;
673a1a4e97bSnorby 	ictl.transmit_delay = iface->transmit_delay;
674a1a4e97bSnorby 	ictl.hello_interval = iface->hello_interval;
675a1a4e97bSnorby 	ictl.flags = iface->flags;
676a1a4e97bSnorby 	ictl.metric = iface->metric;
677a1a4e97bSnorby 	ictl.rxmt_interval = iface->rxmt_interval;
678a1a4e97bSnorby 	ictl.type = iface->type;
679a1a4e97bSnorby 	ictl.linkstate = iface->linkstate;
68018ffdd94Sstsp 	ictl.if_type = iface->if_type;
681a1a4e97bSnorby 	ictl.priority = iface->priority;
6826c9e7a5bSclaudio 	ictl.passive = (iface->cflags & F_IFACE_PASSIVE) == F_IFACE_PASSIVE;
683a1a4e97bSnorby 
684a1a4e97bSnorby 	gettimeofday(&now, NULL);
685a1a4e97bSnorby 	if (evtimer_pending(&iface->hello_timer, &tv)) {
686a1a4e97bSnorby 		timersub(&tv, &now, &res);
687a1a4e97bSnorby 		ictl.hello_timer = res.tv_sec;
688a1a4e97bSnorby 	} else
689a1a4e97bSnorby 		ictl.hello_timer = -1;
690a1a4e97bSnorby 
691a1a4e97bSnorby 	if (iface->state != IF_STA_DOWN) {
692a1a4e97bSnorby 		ictl.uptime = now.tv_sec - iface->uptime;
693a1a4e97bSnorby 	} else
694a1a4e97bSnorby 		ictl.uptime = 0;
695a1a4e97bSnorby 
696a1a4e97bSnorby 	LIST_FOREACH(nbr, &iface->nbr_list, entry) {
697a1a4e97bSnorby 		if (nbr == iface->self)
698a1a4e97bSnorby 			continue;
699a1a4e97bSnorby 		ictl.nbr_cnt++;
700a1a4e97bSnorby 		if (nbr->state & NBR_STA_ADJFORM)
701a1a4e97bSnorby 			ictl.adj_cnt++;
702a1a4e97bSnorby 	}
703a1a4e97bSnorby 
704a1a4e97bSnorby 	return (&ictl);
705a1a4e97bSnorby }
706a1a4e97bSnorby 
707a1a4e97bSnorby /* misc */
708a1a4e97bSnorby void
if_set_sockbuf(int fd)709678bb30aSdenis if_set_sockbuf(int fd)
710a1a4e97bSnorby {
711a1a4e97bSnorby 	int	bsize;
712a1a4e97bSnorby 
7139aaef93cSclaudio 	bsize = 256 * 1024;
714a1a4e97bSnorby 	while (setsockopt(fd, SOL_SOCKET, SO_RCVBUF, &bsize,
715a1a4e97bSnorby 	    sizeof(bsize)) == -1)
716a1a4e97bSnorby 		bsize /= 2;
7179aaef93cSclaudio 
7189aaef93cSclaudio 	if (bsize != 256 * 1024)
719678bb30aSdenis 		log_warnx("if_set_sockbuf: recvbuf size only %d", bsize);
720678bb30aSdenis 
721678bb30aSdenis 	bsize = 64 * 1024;
722678bb30aSdenis 	while (setsockopt(fd, SOL_SOCKET, SO_SNDBUF, &bsize,
723678bb30aSdenis 	    sizeof(bsize)) == -1)
724678bb30aSdenis 		bsize /= 2;
725678bb30aSdenis 
726678bb30aSdenis 	if (bsize != 64 * 1024)
727678bb30aSdenis 		log_warnx("if_set_sockbuf: sendbuf size only %d", bsize);
728a1a4e97bSnorby }
729a1a4e97bSnorby 
730a1a4e97bSnorby int
if_join_group(struct iface * iface,struct in6_addr * addr)731a1a4e97bSnorby if_join_group(struct iface *iface, struct in6_addr *addr)
732a1a4e97bSnorby {
733a1a4e97bSnorby 	struct ipv6_mreq	 mreq;
734a1a4e97bSnorby 
735a1a4e97bSnorby 	switch (iface->type) {
736a1a4e97bSnorby 	case IF_TYPE_POINTOPOINT:
737a1a4e97bSnorby 	case IF_TYPE_BROADCAST:
738a1a4e97bSnorby 		log_debug("if_join_group: interface %s addr %s",
739a1a4e97bSnorby 		    iface->name, log_in6addr(addr));
740a1a4e97bSnorby 		mreq.ipv6mr_multiaddr = *addr;
741a1a4e97bSnorby 		mreq.ipv6mr_interface = iface->ifindex;
742a1a4e97bSnorby 
743a1a4e97bSnorby 		if (setsockopt(iface->fd, IPPROTO_IPV6, IPV6_JOIN_GROUP,
744df69c215Sderaadt 		    &mreq, sizeof(mreq)) == -1) {
745a1a4e97bSnorby 			log_warn("if_join_group: error IPV6_JOIN_GROUP, "
746a1a4e97bSnorby 			    "interface %s address %s", iface->name,
747a1a4e97bSnorby 			    log_in6addr(addr));
748a1a4e97bSnorby 			return (-1);
749a1a4e97bSnorby 		}
750a1a4e97bSnorby 		break;
751a1a4e97bSnorby 	case IF_TYPE_POINTOMULTIPOINT:
752a1a4e97bSnorby 	case IF_TYPE_VIRTUALLINK:
753a1a4e97bSnorby 	case IF_TYPE_NBMA:
754a1a4e97bSnorby 		log_debug("if_join_group: type %s not supported, interface %s",
755a1a4e97bSnorby 		    if_type_name(iface->type), iface->name);
756a1a4e97bSnorby 		return (-1);
757a1a4e97bSnorby 	default:
758a1a4e97bSnorby 		fatalx("if_join_group: unknown interface type");
759a1a4e97bSnorby 	}
760a1a4e97bSnorby 
761a1a4e97bSnorby 	return (0);
762a1a4e97bSnorby }
763a1a4e97bSnorby 
764a1a4e97bSnorby int
if_leave_group(struct iface * iface,struct in6_addr * addr)765a1a4e97bSnorby if_leave_group(struct iface *iface, struct in6_addr *addr)
766a1a4e97bSnorby {
767a1a4e97bSnorby 	struct ipv6_mreq	 mreq;
768a1a4e97bSnorby 
769a1a4e97bSnorby 	switch (iface->type) {
770a1a4e97bSnorby 	case IF_TYPE_POINTOPOINT:
771a1a4e97bSnorby 	case IF_TYPE_BROADCAST:
772a1a4e97bSnorby 		log_debug("if_leave_group: interface %s addr %s",
773a1a4e97bSnorby 		    iface->name, log_in6addr(addr));
774a1a4e97bSnorby 		mreq.ipv6mr_multiaddr = *addr;
775a1a4e97bSnorby 		mreq.ipv6mr_interface = iface->ifindex;
776a1a4e97bSnorby 
777a1a4e97bSnorby 		if (setsockopt(iface->fd, IPPROTO_IPV6, IPV6_LEAVE_GROUP,
778df69c215Sderaadt 		    (void *)&mreq, sizeof(mreq)) == -1) {
779a1a4e97bSnorby 			log_warn("if_leave_group: error IPV6_LEAVE_GROUP, "
780a1a4e97bSnorby 			    "interface %s address %s", iface->name,
781a1a4e97bSnorby 			    log_in6addr(addr));
782a1a4e97bSnorby 			return (-1);
783a1a4e97bSnorby 		}
784a1a4e97bSnorby 		break;
785a1a4e97bSnorby 	case IF_TYPE_POINTOMULTIPOINT:
786a1a4e97bSnorby 	case IF_TYPE_VIRTUALLINK:
787a1a4e97bSnorby 	case IF_TYPE_NBMA:
788a1a4e97bSnorby 		log_debug("if_leave_group: type %s not supported, interface %s",
789a1a4e97bSnorby 		    if_type_name(iface->type), iface->name);
790a1a4e97bSnorby 		return (-1);
791a1a4e97bSnorby 	default:
792a1a4e97bSnorby 		fatalx("if_leave_group: unknown interface type");
793a1a4e97bSnorby 	}
794a1a4e97bSnorby 	return (0);
795a1a4e97bSnorby }
796a1a4e97bSnorby 
797a1a4e97bSnorby int
if_set_mcast(struct iface * iface)798a1a4e97bSnorby if_set_mcast(struct iface *iface)
799a1a4e97bSnorby {
800a1a4e97bSnorby 	switch (iface->type) {
801a1a4e97bSnorby 	case IF_TYPE_POINTOPOINT:
802a1a4e97bSnorby 	case IF_TYPE_BROADCAST:
803a1a4e97bSnorby 		if (setsockopt(iface->fd, IPPROTO_IPV6, IPV6_MULTICAST_IF,
804df69c215Sderaadt 		    &iface->ifindex, sizeof(iface->ifindex)) == -1) {
805a1a4e97bSnorby 			log_debug("if_set_mcast: error setting "
806a1a4e97bSnorby 			    "IP_MULTICAST_IF, interface %s", iface->name);
807a1a4e97bSnorby 			return (-1);
808a1a4e97bSnorby 		}
809a1a4e97bSnorby 		break;
810a1a4e97bSnorby 	case IF_TYPE_POINTOMULTIPOINT:
811a1a4e97bSnorby 	case IF_TYPE_VIRTUALLINK:
812a1a4e97bSnorby 	case IF_TYPE_NBMA:
813a1a4e97bSnorby 		log_debug("if_set_mcast: type %s not supported, interface %s",
814a1a4e97bSnorby 		    if_type_name(iface->type), iface->name);
815a1a4e97bSnorby 		return (-1);
816a1a4e97bSnorby 	default:
817a1a4e97bSnorby 		fatalx("if_set_mcast: unknown interface type");
818a1a4e97bSnorby 	}
819a1a4e97bSnorby 
820a1a4e97bSnorby 	return (0);
821a1a4e97bSnorby }
822a1a4e97bSnorby 
823a1a4e97bSnorby int
if_set_mcast_loop(int fd)824a1a4e97bSnorby if_set_mcast_loop(int fd)
825a1a4e97bSnorby {
826a1a4e97bSnorby 	u_int	loop = 0;
827a1a4e97bSnorby 
828a1a4e97bSnorby 	if (setsockopt(fd, IPPROTO_IPV6, IPV6_MULTICAST_LOOP,
829df69c215Sderaadt 	    (u_int *)&loop, sizeof(loop)) == -1) {
830a1a4e97bSnorby 		log_warn("if_set_mcast_loop: error setting "
831a1a4e97bSnorby 		    "IPV6_MULTICAST_LOOP");
832a1a4e97bSnorby 		return (-1);
833a1a4e97bSnorby 	}
834a1a4e97bSnorby 
835a1a4e97bSnorby 	return (0);
836a1a4e97bSnorby }
837a1a4e97bSnorby 
838a1a4e97bSnorby int
if_set_ipv6_pktinfo(int fd,int enable)83977b36df3Sclaudio if_set_ipv6_pktinfo(int fd, int enable)
840a1a4e97bSnorby {
84177b36df3Sclaudio 	if (setsockopt(fd, IPPROTO_IPV6, IPV6_RECVPKTINFO, &enable,
842df69c215Sderaadt 	    sizeof(enable)) == -1) {
84377b36df3Sclaudio 		log_warn("if_set_ipv6_pktinfo: error setting IPV6_PKTINFO");
844a1a4e97bSnorby 		return (-1);
845a1a4e97bSnorby 	}
84677b36df3Sclaudio 
84777b36df3Sclaudio 	return (0);
84877b36df3Sclaudio }
84977b36df3Sclaudio 
85077b36df3Sclaudio int
if_set_ipv6_checksum(int fd)85177b36df3Sclaudio if_set_ipv6_checksum(int fd)
85277b36df3Sclaudio {
85377b36df3Sclaudio 	int	offset = offsetof(struct ospf_hdr, chksum);
85477b36df3Sclaudio 
855ab4d8988Sderaadt 	log_debug("if_set_ipv6_checksum setting cksum offset to %d", offset);
85677b36df3Sclaudio 	if (setsockopt(fd, IPPROTO_IPV6, IPV6_CHECKSUM, &offset,
857df69c215Sderaadt 	     sizeof(offset)) == -1) {
85877b36df3Sclaudio 		log_warn("if_set_ipv6_checksum: error setting IPV6_CHECKSUM");
85977b36df3Sclaudio 		return (-1);
86077b36df3Sclaudio 	}
861a1a4e97bSnorby 	return (0);
862a1a4e97bSnorby }
863