xref: /freebsd/contrib/bsnmp/snmp_mibII/mibII_route.c (revision 8e9b3e707151d136ec95e7f1d37556e39c1e228c)
1f06ca4afSHartmut Brandt /*
2f06ca4afSHartmut Brandt  * Copyright (c) 2001-2003
3f06ca4afSHartmut Brandt  *	Fraunhofer Institute for Open Communication Systems (FhG Fokus).
4f06ca4afSHartmut Brandt  *	All rights reserved.
5f06ca4afSHartmut Brandt  *
6f06ca4afSHartmut Brandt  * Author: Harti Brandt <harti@freebsd.org>
7f06ca4afSHartmut Brandt  *
8896052c1SHartmut Brandt  * Redistribution and use in source and binary forms, with or without
9896052c1SHartmut Brandt  * modification, are permitted provided that the following conditions
10896052c1SHartmut Brandt  * are met:
11896052c1SHartmut Brandt  * 1. Redistributions of source code must retain the above copyright
12896052c1SHartmut Brandt  *    notice, this list of conditions and the following disclaimer.
13f06ca4afSHartmut Brandt  * 2. Redistributions in binary form must reproduce the above copyright
14f06ca4afSHartmut Brandt  *    notice, this list of conditions and the following disclaimer in the
15f06ca4afSHartmut Brandt  *    documentation and/or other materials provided with the distribution.
16f06ca4afSHartmut Brandt  *
17896052c1SHartmut Brandt  * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18896052c1SHartmut Brandt  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19896052c1SHartmut Brandt  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20896052c1SHartmut Brandt  * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21896052c1SHartmut Brandt  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22896052c1SHartmut Brandt  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23896052c1SHartmut Brandt  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24896052c1SHartmut Brandt  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25896052c1SHartmut Brandt  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26896052c1SHartmut Brandt  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27896052c1SHartmut Brandt  * SUCH DAMAGE.
28f06ca4afSHartmut Brandt  *
29748b5b1eSHartmut Brandt  * $Begemot: bsnmp/snmp_mibII/mibII_route.c,v 1.9 2005/10/06 07:15:00 brandt_h Exp $
30f06ca4afSHartmut Brandt  *
31f06ca4afSHartmut Brandt  * Routing table
32f06ca4afSHartmut Brandt  */
33748b5b1eSHartmut Brandt 
34748b5b1eSHartmut Brandt #ifdef HAVE_SYS_TREE_H
35165c5d31SHartmut Brandt #include <sys/tree.h>
36748b5b1eSHartmut Brandt #else
37748b5b1eSHartmut Brandt #include "tree.h"
38748b5b1eSHartmut Brandt #endif
39748b5b1eSHartmut Brandt 
40f06ca4afSHartmut Brandt #include "mibII.h"
41f06ca4afSHartmut Brandt #include "mibII_oid.h"
42f06ca4afSHartmut Brandt 
43f06ca4afSHartmut Brandt struct sroute {
44165c5d31SHartmut Brandt 	RB_ENTRY(sroute) link;
45165c5d31SHartmut Brandt 	uint32_t	ifindex;
46165c5d31SHartmut Brandt 	uint8_t		index[13];
47165c5d31SHartmut Brandt 	uint8_t		type;
48165c5d31SHartmut Brandt 	uint8_t		proto;
49f06ca4afSHartmut Brandt };
50*8e9b3e70SHartmut Brandt static RB_HEAD(sroutes, sroute) sroutes = RB_INITIALIZER(&sroutes);
51f06ca4afSHartmut Brandt 
52165c5d31SHartmut Brandt RB_PROTOTYPE(sroutes, sroute, link, sroute_compare);
53165c5d31SHartmut Brandt 
54165c5d31SHartmut Brandt #define	ROUTE_UPDATE_INTERVAL	(100 * 60 * 10)	/* 10 min */
5569292cedSHartmut Brandt static uint64_t route_tick;
56f06ca4afSHartmut Brandt static u_int route_total;
57f06ca4afSHartmut Brandt 
58165c5d31SHartmut Brandt /*
59165c5d31SHartmut Brandt  * Compare two routes
60165c5d31SHartmut Brandt  */
61f06ca4afSHartmut Brandt static int
sroute_compare(struct sroute * s1,struct sroute * s2)62165c5d31SHartmut Brandt sroute_compare(struct sroute *s1, struct sroute *s2)
63165c5d31SHartmut Brandt {
64165c5d31SHartmut Brandt 
65165c5d31SHartmut Brandt 	return (memcmp(s1->index, s2->index, 13));
66165c5d31SHartmut Brandt }
67165c5d31SHartmut Brandt 
68165c5d31SHartmut Brandt static void
sroute_index_append(struct asn_oid * oid,u_int sub,const struct sroute * s)69165c5d31SHartmut Brandt sroute_index_append(struct asn_oid *oid, u_int sub, const struct sroute *s)
70165c5d31SHartmut Brandt {
71165c5d31SHartmut Brandt 	int i;
72165c5d31SHartmut Brandt 
73165c5d31SHartmut Brandt 	oid->len = sub + 13;
74165c5d31SHartmut Brandt 	for (i = 0; i < 13; i++)
75165c5d31SHartmut Brandt 		oid->subs[sub + i] = s->index[i];
76165c5d31SHartmut Brandt }
77165c5d31SHartmut Brandt 
78165c5d31SHartmut Brandt #if 0
79165c5d31SHartmut Brandt static void
80165c5d31SHartmut Brandt sroute_print(const struct sroute *r)
81165c5d31SHartmut Brandt {
82165c5d31SHartmut Brandt 	u_int i;
83165c5d31SHartmut Brandt 
84165c5d31SHartmut Brandt 	for (i = 0; i < 13 - 1; i++)
85165c5d31SHartmut Brandt 		printf("%u.", r->index[i]);
86165c5d31SHartmut Brandt 	printf("%u proto=%u type=%u", r->index[i], r->proto, r->type);
87165c5d31SHartmut Brandt }
88165c5d31SHartmut Brandt #endif
89165c5d31SHartmut Brandt 
90165c5d31SHartmut Brandt /*
91165c5d31SHartmut Brandt  * process routing message
92165c5d31SHartmut Brandt  */
93165c5d31SHartmut Brandt void
mib_sroute_process(struct rt_msghdr * rtm,struct sockaddr * gw,struct sockaddr * dst,struct sockaddr * mask)94165c5d31SHartmut Brandt mib_sroute_process(struct rt_msghdr *rtm, struct sockaddr *gw,
95165c5d31SHartmut Brandt     struct sockaddr *dst, struct sockaddr *mask)
96165c5d31SHartmut Brandt {
97165c5d31SHartmut Brandt 	struct sockaddr_in *in_dst, *in_gw;
98165c5d31SHartmut Brandt 	struct in_addr in_mask;
99165c5d31SHartmut Brandt 	struct mibif *ifp;
100165c5d31SHartmut Brandt 	struct sroute key;
101165c5d31SHartmut Brandt 	struct sroute *r, *r1;
102165c5d31SHartmut Brandt 	in_addr_t ha;
103165c5d31SHartmut Brandt 
104165c5d31SHartmut Brandt 	if (dst == NULL || gw == NULL || dst->sa_family != AF_INET ||
105165c5d31SHartmut Brandt 	    gw->sa_family != AF_INET)
106165c5d31SHartmut Brandt 		return;
107165c5d31SHartmut Brandt 
108165c5d31SHartmut Brandt 	in_dst = (struct sockaddr_in *)(void *)dst;
109165c5d31SHartmut Brandt 	in_gw = (struct sockaddr_in *)(void *)gw;
110165c5d31SHartmut Brandt 
111165c5d31SHartmut Brandt 	if (rtm->rtm_flags & RTF_HOST)
112165c5d31SHartmut Brandt 		in_mask.s_addr = 0xffffffff;
113165c5d31SHartmut Brandt 	else if (mask == NULL || mask->sa_len == 0)
114165c5d31SHartmut Brandt 		in_mask.s_addr = 0;
115165c5d31SHartmut Brandt 	else
116165c5d31SHartmut Brandt 		in_mask = ((struct sockaddr_in *)(void *)mask)->sin_addr;
117165c5d31SHartmut Brandt 
118165c5d31SHartmut Brandt 	/* build the index */
119165c5d31SHartmut Brandt 	ha = ntohl(in_dst->sin_addr.s_addr);
120165c5d31SHartmut Brandt 	key.index[0] = (ha >> 24) & 0xff;
121165c5d31SHartmut Brandt 	key.index[1] = (ha >> 16) & 0xff;
122165c5d31SHartmut Brandt 	key.index[2] = (ha >>  8) & 0xff;
123165c5d31SHartmut Brandt 	key.index[3] = (ha >>  0) & 0xff;
124165c5d31SHartmut Brandt 
125165c5d31SHartmut Brandt 	ha = ntohl(in_mask.s_addr);
126165c5d31SHartmut Brandt 	key.index[4] = (ha >> 24) & 0xff;
127165c5d31SHartmut Brandt 	key.index[5] = (ha >> 16) & 0xff;
128165c5d31SHartmut Brandt 	key.index[6] = (ha >>  8) & 0xff;
129165c5d31SHartmut Brandt 	key.index[7] = (ha >>  0) & 0xff;
130165c5d31SHartmut Brandt 
131165c5d31SHartmut Brandt 	/* ToS */
132165c5d31SHartmut Brandt 	key.index[8] = 0;
133165c5d31SHartmut Brandt 
134165c5d31SHartmut Brandt 	ha = ntohl(in_gw->sin_addr.s_addr);
135165c5d31SHartmut Brandt 	key.index[9] = (ha >> 24) & 0xff;
136165c5d31SHartmut Brandt 	key.index[10] = (ha >> 16) & 0xff;
137165c5d31SHartmut Brandt 	key.index[11] = (ha >>  8) & 0xff;
138165c5d31SHartmut Brandt 	key.index[12] = (ha >>  0) & 0xff;
139165c5d31SHartmut Brandt 
140165c5d31SHartmut Brandt 	if (rtm->rtm_type == RTM_DELETE) {
141165c5d31SHartmut Brandt 		r = RB_FIND(sroutes, &sroutes, &key);
142165c5d31SHartmut Brandt 		if (r == 0) {
143165c5d31SHartmut Brandt #ifdef DEBUG_ROUTE
144165c5d31SHartmut Brandt 			syslog(LOG_WARNING, "%s: DELETE: %u.%u.%u.%u "
145165c5d31SHartmut Brandt 			    "%u.%u.%u.%u %u %u.%u.%u.%u not found", __func__,
146165c5d31SHartmut Brandt 			    key.index[0], key.index[1], key.index[2],
147165c5d31SHartmut Brandt 			    key.index[3], key.index[4], key.index[5],
148165c5d31SHartmut Brandt 			    key.index[6], key.index[7], key.index[8],
149165c5d31SHartmut Brandt 			    key.index[9], key.index[10], key.index[11],
150165c5d31SHartmut Brandt 			    key.index[12]);
151165c5d31SHartmut Brandt #endif
152165c5d31SHartmut Brandt 			return;
153165c5d31SHartmut Brandt 		}
154165c5d31SHartmut Brandt 		RB_REMOVE(sroutes, &sroutes, r);
155165c5d31SHartmut Brandt 		free(r);
156165c5d31SHartmut Brandt 		route_total--;
157165c5d31SHartmut Brandt #ifdef DEBUG_ROUTE
158165c5d31SHartmut Brandt 		printf("%s: DELETE: %u.%u.%u.%u "
159165c5d31SHartmut Brandt 		    "%u.%u.%u.%u %u %u.%u.%u.%u\n", __func__,
160165c5d31SHartmut Brandt 		    key.index[0], key.index[1], key.index[2],
161165c5d31SHartmut Brandt 		    key.index[3], key.index[4], key.index[5],
162165c5d31SHartmut Brandt 		    key.index[6], key.index[7], key.index[8],
163165c5d31SHartmut Brandt 		    key.index[9], key.index[10], key.index[11],
164165c5d31SHartmut Brandt 		    key.index[12]);
165165c5d31SHartmut Brandt #endif
166165c5d31SHartmut Brandt 		return;
167165c5d31SHartmut Brandt 	}
168165c5d31SHartmut Brandt 
169165c5d31SHartmut Brandt 	/* GET or ADD */
170165c5d31SHartmut Brandt 	ifp = NULL;
171165c5d31SHartmut Brandt 	if ((ifp = mib_find_if_sys(rtm->rtm_index)) == NULL) {
172165c5d31SHartmut Brandt 		if (rtm->rtm_type == RTM_ADD) {
173165c5d31SHartmut Brandt 			/* make it a get so the kernel fills the index */
174165c5d31SHartmut Brandt 			mib_send_rtmsg(rtm, gw, dst, mask);
175165c5d31SHartmut Brandt 			return;
176165c5d31SHartmut Brandt 		}
177165c5d31SHartmut Brandt 		mib_iflist_bad = 1;
178165c5d31SHartmut Brandt 	}
179165c5d31SHartmut Brandt 
180165c5d31SHartmut Brandt 	if ((r = malloc(sizeof(*r))) == NULL) {
181165c5d31SHartmut Brandt 		syslog(LOG_ERR, "%m");
182165c5d31SHartmut Brandt 		return;
183165c5d31SHartmut Brandt 	}
184165c5d31SHartmut Brandt 
185165c5d31SHartmut Brandt 	memcpy(r->index, key.index, sizeof(r->index));
186165c5d31SHartmut Brandt 	r->ifindex = (ifp == NULL) ? 0 : ifp->index;
187165c5d31SHartmut Brandt 
1886e6b3f7cSQing Li 	r->type = (rtm->rtm_flags & RTF_REJECT) ? 2 : 4;
189165c5d31SHartmut Brandt 
190165c5d31SHartmut Brandt 	/* cannot really know, what protocol it runs */
191165c5d31SHartmut Brandt 	r->proto = (rtm->rtm_flags & RTF_LOCAL) ? 2 :
192165c5d31SHartmut Brandt 	    (rtm->rtm_flags & RTF_STATIC) ? 3 :
193165c5d31SHartmut Brandt 	    (rtm->rtm_flags & RTF_DYNAMIC) ? 4 : 10;
194165c5d31SHartmut Brandt 
195165c5d31SHartmut Brandt 	r1 = RB_INSERT(sroutes, &sroutes, r);
196165c5d31SHartmut Brandt 	if (r1 != NULL) {
197165c5d31SHartmut Brandt #ifdef DEBUG_ROUTE
198165c5d31SHartmut Brandt 		syslog(LOG_WARNING, "%s: %u.%u.%u.%u "
199165c5d31SHartmut Brandt 		    "%u.%u.%u.%u %u %u.%u.%u.%u duplicate route", __func__,
200165c5d31SHartmut Brandt 		    key.index[0], key.index[1], key.index[2],
201165c5d31SHartmut Brandt 		    key.index[3], key.index[4], key.index[5],
202165c5d31SHartmut Brandt 		    key.index[6], key.index[7], key.index[8],
203165c5d31SHartmut Brandt 		    key.index[9], key.index[10], key.index[11],
204165c5d31SHartmut Brandt 		    key.index[12]);
205165c5d31SHartmut Brandt #endif
206165c5d31SHartmut Brandt 		r1->ifindex = r->ifindex;
207165c5d31SHartmut Brandt 		r1->type = r->type;
208165c5d31SHartmut Brandt 		r1->proto = r->proto;
209165c5d31SHartmut Brandt 		free(r);
210165c5d31SHartmut Brandt 		return;
211165c5d31SHartmut Brandt 	}
212165c5d31SHartmut Brandt 
213165c5d31SHartmut Brandt 	route_total++;
214165c5d31SHartmut Brandt #ifdef DEBUG_ROUTE
215165c5d31SHartmut Brandt 	printf("%s: ADD/GET: %u.%u.%u.%u "
216165c5d31SHartmut Brandt 	    "%u.%u.%u.%u %u %u.%u.%u.%u\n", __func__,
217165c5d31SHartmut Brandt 	    key.index[0], key.index[1], key.index[2],
218165c5d31SHartmut Brandt 	    key.index[3], key.index[4], key.index[5],
219165c5d31SHartmut Brandt 	    key.index[6], key.index[7], key.index[8],
220165c5d31SHartmut Brandt 	    key.index[9], key.index[10], key.index[11],
221165c5d31SHartmut Brandt 	    key.index[12]);
222165c5d31SHartmut Brandt #endif
223165c5d31SHartmut Brandt }
224165c5d31SHartmut Brandt 
225165c5d31SHartmut Brandt int
mib_fetch_route(void)226165c5d31SHartmut Brandt mib_fetch_route(void)
227f06ca4afSHartmut Brandt {
228f06ca4afSHartmut Brandt 	u_char *rtab, *next;
229f06ca4afSHartmut Brandt 	size_t len;
230165c5d31SHartmut Brandt 	struct sroute *r, *r1;
231f06ca4afSHartmut Brandt 	struct rt_msghdr *rtm;
232f06ca4afSHartmut Brandt 	struct sockaddr *addrs[RTAX_MAX];
233f06ca4afSHartmut Brandt 
234165c5d31SHartmut Brandt 	if (route_tick != 0 && route_tick + ROUTE_UPDATE_INTERVAL > this_tick)
235165c5d31SHartmut Brandt 		return (0);
236165c5d31SHartmut Brandt 
237165c5d31SHartmut Brandt 	/*
238165c5d31SHartmut Brandt 	 * Remove all routes
239165c5d31SHartmut Brandt 	 */
240165c5d31SHartmut Brandt 	r = RB_MIN(sroutes, &sroutes);
241165c5d31SHartmut Brandt 	while (r != NULL) {
242165c5d31SHartmut Brandt 		r1 = RB_NEXT(sroutes, &sroutes, r);
243165c5d31SHartmut Brandt 		RB_REMOVE(sroutes, &sroutes, r);
244f06ca4afSHartmut Brandt 		free(r);
245165c5d31SHartmut Brandt 		r = r1;
246f06ca4afSHartmut Brandt 	}
247f06ca4afSHartmut Brandt 	route_total = 0;
248f06ca4afSHartmut Brandt 
249f06ca4afSHartmut Brandt 	if ((rtab = mib_fetch_rtab(AF_INET, NET_RT_DUMP, 0, &len)) == NULL)
250f06ca4afSHartmut Brandt 		return (-1);
251f06ca4afSHartmut Brandt 
252f06ca4afSHartmut Brandt 	next = rtab;
253f06ca4afSHartmut Brandt 	for (next = rtab; next < rtab + len; next += rtm->rtm_msglen) {
254f06ca4afSHartmut Brandt 		rtm = (struct rt_msghdr *)(void *)next;
255f06ca4afSHartmut Brandt 		if (rtm->rtm_type != RTM_GET ||
256f06ca4afSHartmut Brandt 		    !(rtm->rtm_flags & RTF_UP))
257f06ca4afSHartmut Brandt 			continue;
258f06ca4afSHartmut Brandt 		mib_extract_addrs(rtm->rtm_addrs, (u_char *)(rtm + 1), addrs);
259f06ca4afSHartmut Brandt 
260f06ca4afSHartmut Brandt 
261165c5d31SHartmut Brandt 		mib_sroute_process(rtm, addrs[RTAX_GATEWAY], addrs[RTAX_DST],
262165c5d31SHartmut Brandt 		    addrs[RTAX_NETMASK]);
263f06ca4afSHartmut Brandt 	}
264f06ca4afSHartmut Brandt 
265165c5d31SHartmut Brandt #if 0
266165c5d31SHartmut Brandt 	u_int n = 0;
267165c5d31SHartmut Brandt 	r = RB_MIN(sroutes, &sroutes);
268165c5d31SHartmut Brandt 	while (r != NULL) {
269165c5d31SHartmut Brandt 		printf("%u: ", n++);
270165c5d31SHartmut Brandt 		sroute_print(r);
271165c5d31SHartmut Brandt 		printf("\n");
272165c5d31SHartmut Brandt 		r = RB_NEXT(sroutes, &sroutes, r);
273f06ca4afSHartmut Brandt 	}
274165c5d31SHartmut Brandt #endif
275f06ca4afSHartmut Brandt 	free(rtab);
276f06ca4afSHartmut Brandt 	route_tick = get_ticks();
277f06ca4afSHartmut Brandt 
278f06ca4afSHartmut Brandt 	return (0);
279f06ca4afSHartmut Brandt }
280f06ca4afSHartmut Brandt 
281165c5d31SHartmut Brandt /**
282165c5d31SHartmut Brandt  * Find a route in the table.
283165c5d31SHartmut Brandt  */
284165c5d31SHartmut Brandt static struct sroute *
sroute_get(const struct asn_oid * oid,u_int sub)285165c5d31SHartmut Brandt sroute_get(const struct asn_oid *oid, u_int sub)
286165c5d31SHartmut Brandt {
287165c5d31SHartmut Brandt 	struct sroute key;
288165c5d31SHartmut Brandt 	int i;
289165c5d31SHartmut Brandt 
290165c5d31SHartmut Brandt 	if (oid->len - sub != 13)
291165c5d31SHartmut Brandt 		return (NULL);
292165c5d31SHartmut Brandt 	for (i = 0; i < 13; i++)
293165c5d31SHartmut Brandt 		key.index[i] = oid->subs[sub + i];
294165c5d31SHartmut Brandt 	return (RB_FIND(sroutes, &sroutes, &key));
295165c5d31SHartmut Brandt }
296165c5d31SHartmut Brandt 
297165c5d31SHartmut Brandt /**
298165c5d31SHartmut Brandt  * Find next route in the table. There is no such RB_ macro, so must
299165c5d31SHartmut Brandt  * dig into the innards of the RB stuff.
300165c5d31SHartmut Brandt  */
301165c5d31SHartmut Brandt static struct sroute *
sroute_getnext(struct asn_oid * oid,u_int sub)302165c5d31SHartmut Brandt sroute_getnext(struct asn_oid *oid, u_int sub)
303165c5d31SHartmut Brandt {
304165c5d31SHartmut Brandt 	u_int i;
305165c5d31SHartmut Brandt 	int comp;
306165c5d31SHartmut Brandt 	struct sroute key;
307165c5d31SHartmut Brandt 	struct sroute *best;
308165c5d31SHartmut Brandt 	struct sroute *s;
309165c5d31SHartmut Brandt 
310165c5d31SHartmut Brandt 	/*
311165c5d31SHartmut Brandt 	 * We now, that the OID is at least the tableEntry OID. If it is,
312165c5d31SHartmut Brandt 	 * the user wants the first route.
313165c5d31SHartmut Brandt 	 */
314165c5d31SHartmut Brandt 	if (oid->len == sub)
315165c5d31SHartmut Brandt 		return (RB_MIN(sroutes, &sroutes));
316165c5d31SHartmut Brandt 
317165c5d31SHartmut Brandt 	/*
318165c5d31SHartmut Brandt 	 * This is also true for any index that consists of zeros and is
319165c5d31SHartmut Brandt 	 * shorter than the full index.
320165c5d31SHartmut Brandt 	 */
321165c5d31SHartmut Brandt 	if (oid->len < sub + 13) {
322165c5d31SHartmut Brandt 		for (i = sub; i < oid->len; i++)
323165c5d31SHartmut Brandt 			if (oid->subs[i] != 0)
324165c5d31SHartmut Brandt 				break;
325165c5d31SHartmut Brandt 		if (i == oid->len)
326165c5d31SHartmut Brandt 			return (RB_MIN(sroutes, &sroutes));
327165c5d31SHartmut Brandt 
328165c5d31SHartmut Brandt 		/*
329165c5d31SHartmut Brandt 		 * Now if the index is too short, we fill it with zeros and then
330165c5d31SHartmut Brandt 		 * subtract one from the index. We can do this, because we now,
331165c5d31SHartmut Brandt 		 * that there is at least one index element that is not zero.
332165c5d31SHartmut Brandt 		 */
333165c5d31SHartmut Brandt 		for (i = oid->len; i < sub + 13; i++)
334165c5d31SHartmut Brandt 			oid->subs[i] = 0;
335165c5d31SHartmut Brandt 
336165c5d31SHartmut Brandt 		for (i = sub + 13 - 1; i >= sub; i--) {
337165c5d31SHartmut Brandt 			if (oid->subs[i] != 0) {
338165c5d31SHartmut Brandt 				oid->subs[i]--;
339165c5d31SHartmut Brandt 				break;
340165c5d31SHartmut Brandt 			}
341165c5d31SHartmut Brandt 			oid->subs[i] = ASN_MAXID;
342165c5d31SHartmut Brandt 		}
343165c5d31SHartmut Brandt 		oid->len = sub + 13;
344165c5d31SHartmut Brandt 	}
345165c5d31SHartmut Brandt 
346165c5d31SHartmut Brandt 	/* build the index */
347165c5d31SHartmut Brandt 	for (i = sub; i < sub + 13; i++)
348165c5d31SHartmut Brandt 		key.index[i - sub] = oid->subs[i];
349165c5d31SHartmut Brandt 
350165c5d31SHartmut Brandt 	/* now find the element */
351165c5d31SHartmut Brandt 	best = NULL;
352165c5d31SHartmut Brandt 	s = RB_ROOT(&sroutes);
353165c5d31SHartmut Brandt 
354165c5d31SHartmut Brandt 	while (s != NULL) {
355165c5d31SHartmut Brandt 		comp = sroute_compare(&key, s);
356165c5d31SHartmut Brandt 		if (comp >= 0) {
357165c5d31SHartmut Brandt 			/* The current element is smaller than what we search.
358165c5d31SHartmut Brandt 			 * Forget about it and move to the right subtree. */
359165c5d31SHartmut Brandt 			s = RB_RIGHT(s, link);
360165c5d31SHartmut Brandt 			continue;
361165c5d31SHartmut Brandt 		}
362165c5d31SHartmut Brandt 		/* the current element is larger than what we search.
363165c5d31SHartmut Brandt 		 * forget about the right subtree (its even larger), but
364165c5d31SHartmut Brandt 		 * the current element may be what we need. */
365165c5d31SHartmut Brandt 		if (best == NULL || sroute_compare(s, best) < 0)
366165c5d31SHartmut Brandt 			/* this one's better */
367165c5d31SHartmut Brandt 			best = s;
368165c5d31SHartmut Brandt 
369165c5d31SHartmut Brandt 		s = RB_LEFT(s, link);
370165c5d31SHartmut Brandt 	}
371165c5d31SHartmut Brandt 	return (best);
372165c5d31SHartmut Brandt }
373165c5d31SHartmut Brandt 
374f06ca4afSHartmut Brandt /*
375f06ca4afSHartmut Brandt  * Table
376f06ca4afSHartmut Brandt  */
377f06ca4afSHartmut Brandt int
op_route_table(struct snmp_context * ctx __unused,struct snmp_value * value,u_int sub,u_int iidx __unused,enum snmp_op op)378f06ca4afSHartmut Brandt op_route_table(struct snmp_context *ctx __unused, struct snmp_value *value,
379f06ca4afSHartmut Brandt     u_int sub, u_int iidx __unused, enum snmp_op op)
380f06ca4afSHartmut Brandt {
381165c5d31SHartmut Brandt 	struct sroute *r;
382f06ca4afSHartmut Brandt 
383165c5d31SHartmut Brandt 	if (mib_fetch_route() == -1)
384f06ca4afSHartmut Brandt 		return (SNMP_ERR_GENERR);
385f06ca4afSHartmut Brandt 
386f06ca4afSHartmut Brandt 	switch (op) {
387f06ca4afSHartmut Brandt 
388f06ca4afSHartmut Brandt 	  case SNMP_OP_GETNEXT:
389165c5d31SHartmut Brandt 		if ((r = sroute_getnext(&value->var, sub)) == NULL)
390f06ca4afSHartmut Brandt 			return (SNMP_ERR_NOSUCHNAME);
391165c5d31SHartmut Brandt 		sroute_index_append(&value->var, sub, r);
392f06ca4afSHartmut Brandt 		break;
393f06ca4afSHartmut Brandt 
394f06ca4afSHartmut Brandt 	  case SNMP_OP_GET:
395165c5d31SHartmut Brandt 		if ((r = sroute_get(&value->var, sub)) == NULL)
396f06ca4afSHartmut Brandt 			return (SNMP_ERR_NOSUCHNAME);
397f06ca4afSHartmut Brandt 		break;
398f06ca4afSHartmut Brandt 
399f06ca4afSHartmut Brandt 	  case SNMP_OP_SET:
400165c5d31SHartmut Brandt 		if ((r = sroute_get(&value->var, sub)) == NULL)
401165c5d31SHartmut Brandt 			return (SNMP_ERR_NOSUCHNAME);
402f06ca4afSHartmut Brandt 		return (SNMP_ERR_NOT_WRITEABLE);
403f06ca4afSHartmut Brandt 
404f06ca4afSHartmut Brandt 	  case SNMP_OP_ROLLBACK:
405f06ca4afSHartmut Brandt 	  case SNMP_OP_COMMIT:
406f06ca4afSHartmut Brandt 		abort();
407f06ca4afSHartmut Brandt 
408f06ca4afSHartmut Brandt 	  default:
409f06ca4afSHartmut Brandt 		abort();
410f06ca4afSHartmut Brandt 	}
411f06ca4afSHartmut Brandt 
412f06ca4afSHartmut Brandt 	switch (value->var.subs[sub - 1]) {
413f06ca4afSHartmut Brandt 
414f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteDest:
415165c5d31SHartmut Brandt 		value->v.ipaddress[0] = r->index[0];
416165c5d31SHartmut Brandt 		value->v.ipaddress[1] = r->index[1];
417165c5d31SHartmut Brandt 		value->v.ipaddress[2] = r->index[2];
418165c5d31SHartmut Brandt 		value->v.ipaddress[3] = r->index[3];
419f06ca4afSHartmut Brandt 		break;
420f06ca4afSHartmut Brandt 
421f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteMask:
422165c5d31SHartmut Brandt 		value->v.ipaddress[0] = r->index[4];
423165c5d31SHartmut Brandt 		value->v.ipaddress[1] = r->index[5];
424165c5d31SHartmut Brandt 		value->v.ipaddress[2] = r->index[6];
425165c5d31SHartmut Brandt 		value->v.ipaddress[3] = r->index[7];
426f06ca4afSHartmut Brandt 		break;
427f06ca4afSHartmut Brandt 
428f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteTos:
429165c5d31SHartmut Brandt 		value->v.integer = r->index[8];
430f06ca4afSHartmut Brandt 		break;
431f06ca4afSHartmut Brandt 
432f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteNextHop:
433165c5d31SHartmut Brandt 		value->v.ipaddress[0] = r->index[9];
434165c5d31SHartmut Brandt 		value->v.ipaddress[1] = r->index[10];
435165c5d31SHartmut Brandt 		value->v.ipaddress[2] = r->index[11];
436165c5d31SHartmut Brandt 		value->v.ipaddress[3] = r->index[12];
437f06ca4afSHartmut Brandt 		break;
438f06ca4afSHartmut Brandt 
439f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteIfIndex:
440f06ca4afSHartmut Brandt 		value->v.integer = r->ifindex;
441f06ca4afSHartmut Brandt 		break;
442f06ca4afSHartmut Brandt 
443f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteType:
444f06ca4afSHartmut Brandt 		value->v.integer = r->type;
445f06ca4afSHartmut Brandt 		break;
446f06ca4afSHartmut Brandt 
447f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteProto:
448f06ca4afSHartmut Brandt 		value->v.integer = r->proto;
449f06ca4afSHartmut Brandt 		break;
450f06ca4afSHartmut Brandt 
451f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteAge:
452f06ca4afSHartmut Brandt 		value->v.integer = 0;
453f06ca4afSHartmut Brandt 		break;
454f06ca4afSHartmut Brandt 
455f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteInfo:
456f06ca4afSHartmut Brandt 		value->v.oid = oid_zeroDotZero;
457f06ca4afSHartmut Brandt 		break;
458f06ca4afSHartmut Brandt 
459f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteNextHopAS:
460f06ca4afSHartmut Brandt 		value->v.integer = 0;
461f06ca4afSHartmut Brandt 		break;
462f06ca4afSHartmut Brandt 
463f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteMetric1:
464f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteMetric2:
465f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteMetric3:
466f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteMetric4:
467f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteMetric5:
468f06ca4afSHartmut Brandt 		value->v.integer = -1;
469f06ca4afSHartmut Brandt 		break;
470f06ca4afSHartmut Brandt 
471f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteStatus:
472f06ca4afSHartmut Brandt 		value->v.integer = 1;
473f06ca4afSHartmut Brandt 		break;
474f06ca4afSHartmut Brandt 	}
475f06ca4afSHartmut Brandt 	return (SNMP_ERR_NOERROR);
476f06ca4afSHartmut Brandt }
477f06ca4afSHartmut Brandt 
478f06ca4afSHartmut Brandt /*
479f06ca4afSHartmut Brandt  * scalars
480f06ca4afSHartmut Brandt  */
481f06ca4afSHartmut Brandt int
op_route(struct snmp_context * ctx __unused,struct snmp_value * value,u_int sub,u_int iidx __unused,enum snmp_op op)482f06ca4afSHartmut Brandt op_route(struct snmp_context *ctx __unused, struct snmp_value *value,
483f06ca4afSHartmut Brandt     u_int sub, u_int iidx __unused, enum snmp_op op)
484f06ca4afSHartmut Brandt {
485f06ca4afSHartmut Brandt 	switch (op) {
486f06ca4afSHartmut Brandt 
487f06ca4afSHartmut Brandt 	  case SNMP_OP_GETNEXT:
488f06ca4afSHartmut Brandt 		abort();
489f06ca4afSHartmut Brandt 
490f06ca4afSHartmut Brandt 	  case SNMP_OP_GET:
491f06ca4afSHartmut Brandt 		break;
492f06ca4afSHartmut Brandt 
493f06ca4afSHartmut Brandt 	  case SNMP_OP_SET:
494f06ca4afSHartmut Brandt 		return (SNMP_ERR_NOT_WRITEABLE);
495f06ca4afSHartmut Brandt 
496f06ca4afSHartmut Brandt 	  case SNMP_OP_ROLLBACK:
497f06ca4afSHartmut Brandt 	  case SNMP_OP_COMMIT:
498f06ca4afSHartmut Brandt 		abort();
499f06ca4afSHartmut Brandt 	}
500f06ca4afSHartmut Brandt 
501165c5d31SHartmut Brandt 	if (mib_fetch_route() == -1)
502f06ca4afSHartmut Brandt 		return (SNMP_ERR_GENERR);
503f06ca4afSHartmut Brandt 
504f06ca4afSHartmut Brandt 	switch (value->var.subs[sub - 1]) {
505f06ca4afSHartmut Brandt 
506f06ca4afSHartmut Brandt 	  case LEAF_ipCidrRouteNumber:
507f06ca4afSHartmut Brandt 		value->v.uint32 = route_total;
508f06ca4afSHartmut Brandt 		break;
509f06ca4afSHartmut Brandt 
510f06ca4afSHartmut Brandt 	}
511f06ca4afSHartmut Brandt 	return (SNMP_ERR_NOERROR);
512f06ca4afSHartmut Brandt }
513165c5d31SHartmut Brandt 
514165c5d31SHartmut Brandt RB_GENERATE(sroutes, sroute, link, sroute_compare);
515