xref: /netbsd-src/usr.sbin/mrouted/config.c (revision ee70b5d6af9f4f51fb9c7f3884f534cc3fcf2dba)
1*ee70b5d6Sdsl /*	$NetBSD: config.c,v 1.13 2003/05/16 22:59:50 dsl Exp $	*/
2c60d41a9Swiz 
3c60d41a9Swiz /*
4c60d41a9Swiz  * The mrouted program is covered by the license in the accompanying file
5c60d41a9Swiz  * named "LICENSE".  Use of the mrouted program represents acceptance of
6c60d41a9Swiz  * the terms and conditions listed in that file.
7c60d41a9Swiz  *
8c60d41a9Swiz  * The mrouted program is COPYRIGHT 1989 by The Board of Trustees of
9c60d41a9Swiz  * Leland Stanford Junior University.
10c60d41a9Swiz  */
11c60d41a9Swiz 
12c60d41a9Swiz #include "defs.h"
13c60d41a9Swiz #include <net/if.h>
14c60d41a9Swiz #include <ifaddrs.h>
15c60d41a9Swiz 
16c60d41a9Swiz 
17c60d41a9Swiz /*
18c60d41a9Swiz  * Query the kernel to find network interfaces that are multicast-capable
19c60d41a9Swiz  * and install them in the uvifs array.
20c60d41a9Swiz  */
21c60d41a9Swiz void
config_vifs_from_kernel(void)22c60d41a9Swiz config_vifs_from_kernel(void)
23c60d41a9Swiz {
24c60d41a9Swiz     struct ifaddrs *ifa, *ifap;
25c60d41a9Swiz     struct uvif *v;
26c60d41a9Swiz     vifi_t vifi;
27c60d41a9Swiz     u_int32_t addr, mask, subnet;
28c60d41a9Swiz     short flags;
29c60d41a9Swiz 
30c60d41a9Swiz     if (getifaddrs(&ifap) < 0)
31c60d41a9Swiz 	logit(LOG_ERR, errno, "getifaddrs");
32c60d41a9Swiz 
33c60d41a9Swiz     for (ifa = ifap; ifa; ifa = ifa->ifa_next) {
34c60d41a9Swiz 	/*
35c60d41a9Swiz 	 * Ignore any interface for an address family other than IP.
36c60d41a9Swiz 	 */
37c60d41a9Swiz 	if (ifa->ifa_addr->sa_family != AF_INET)
38c60d41a9Swiz 	    continue;
39c60d41a9Swiz 
40c60d41a9Swiz 	addr = ((struct sockaddr_in *)ifa->ifa_addr)->sin_addr.s_addr;
41c60d41a9Swiz 
42c60d41a9Swiz 	/*
43c60d41a9Swiz 	 * Ignore loopback interfaces and interfaces that do not support
44c60d41a9Swiz 	 * multicast.
45c60d41a9Swiz 	 */
46c60d41a9Swiz 	flags = ifa->ifa_flags;
47c60d41a9Swiz 	if ((flags & (IFF_LOOPBACK|IFF_MULTICAST)) != IFF_MULTICAST)
48c60d41a9Swiz 	    continue;
49c60d41a9Swiz 
50c60d41a9Swiz 	/*
51c60d41a9Swiz 	 * Ignore any interface whose address and mask do not define a
52c60d41a9Swiz 	 * valid subnet number, or whose address is of the form {subnet,0}
53c60d41a9Swiz 	 * or {subnet,-1}.
54c60d41a9Swiz 	 */
55c60d41a9Swiz 	mask = ((struct sockaddr_in *)ifa->ifa_netmask)->sin_addr.s_addr;
56c60d41a9Swiz 	subnet = addr & mask;
57c60d41a9Swiz 	if (!inet_valid_subnet(subnet, mask) ||
58c60d41a9Swiz 	    addr == subnet ||
59c60d41a9Swiz 	    addr == (subnet | ~mask)) {
60c60d41a9Swiz 	    logit(LOG_WARNING, 0,
61c60d41a9Swiz 		"ignoring %s, has invalid address (%s) and/or mask (%s)",
62*ee70b5d6Sdsl 		ifa->ifa_name, inet_fmt(addr),
63*ee70b5d6Sdsl 		inet_fmt(mask));
64c60d41a9Swiz 	    continue;
65c60d41a9Swiz 	}
66c60d41a9Swiz 
67c60d41a9Swiz 	/*
68c60d41a9Swiz 	 * Ignore any interface that is connected to the same subnet as
69c60d41a9Swiz 	 * one already installed in the uvifs array.
70c60d41a9Swiz 	 */
71c60d41a9Swiz 	for (vifi = 0, v = uvifs; vifi < numvifs; ++vifi, ++v) {
72c60d41a9Swiz 	    if ((addr & v->uv_subnetmask) == v->uv_subnet ||
73c60d41a9Swiz 		(v->uv_subnet & mask) == subnet) {
74c60d41a9Swiz 		logit(LOG_WARNING, 0, "ignoring %s, same subnet as %s",
75c60d41a9Swiz 		    ifa->ifa_name, v->uv_name);
76c60d41a9Swiz 		break;
77c60d41a9Swiz 	    }
78c60d41a9Swiz 	}
79c60d41a9Swiz 	if (vifi != numvifs)
80c60d41a9Swiz 	    continue;
81c60d41a9Swiz 
82c60d41a9Swiz 	/*
83c60d41a9Swiz 	 * If there is room in the uvifs array, install this interface.
84c60d41a9Swiz 	 */
85c60d41a9Swiz 	if (numvifs == MAXVIFS) {
86c60d41a9Swiz 	    logit(LOG_WARNING, 0, "too many vifs, ignoring %s", ifa->ifa_name);
87c60d41a9Swiz 	    continue;
88c60d41a9Swiz 	}
89c60d41a9Swiz 	v  = &uvifs[numvifs];
90c60d41a9Swiz 	v->uv_flags       = 0;
91c60d41a9Swiz 	v->uv_metric      = DEFAULT_METRIC;
92c60d41a9Swiz 	v->uv_rate_limit  = DEFAULT_PHY_RATE_LIMIT;
93c60d41a9Swiz 	v->uv_threshold   = DEFAULT_THRESHOLD;
94c60d41a9Swiz 	v->uv_lcl_addr    = addr;
95c60d41a9Swiz 	v->uv_rmt_addr    = 0;
96c60d41a9Swiz 	v->uv_subnet      = subnet;
97c60d41a9Swiz 	v->uv_subnetmask  = mask;
98c60d41a9Swiz 	v->uv_subnetbcast = subnet | ~mask;
99c60d41a9Swiz 	strlcpy(v->uv_name, ifa->ifa_name, sizeof(v->uv_name));
100c60d41a9Swiz 	v->uv_groups      = NULL;
101c60d41a9Swiz 	v->uv_neighbors   = NULL;
102c60d41a9Swiz 	v->uv_acl         = NULL;
103c60d41a9Swiz 	v->uv_addrs	  = NULL;
104c60d41a9Swiz 
105c60d41a9Swiz 	logit(LOG_INFO,0,"installing %s (%s on subnet %s) as vif #%u - rate=%d",
106*ee70b5d6Sdsl 	    v->uv_name, inet_fmt(addr),
107*ee70b5d6Sdsl 	    inet_fmts(subnet, mask),
108c60d41a9Swiz 	    numvifs, v->uv_rate_limit);
109c60d41a9Swiz 
110c60d41a9Swiz 	++numvifs;
111c60d41a9Swiz 
112c60d41a9Swiz 	/*
113c60d41a9Swiz 	 * If the interface is not yet up, set the vifs_down flag to
114c60d41a9Swiz 	 * remind us to check again later.
115c60d41a9Swiz 	 */
116c60d41a9Swiz 	if (!(flags & IFF_UP)) {
117c60d41a9Swiz 	    v->uv_flags |= VIFF_DOWN;
118c60d41a9Swiz 	    vifs_down = TRUE;
119c60d41a9Swiz 	}
120c60d41a9Swiz     }
121c60d41a9Swiz 
122c60d41a9Swiz     freeifaddrs(ifap);
123c60d41a9Swiz }
124