]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/bsnmp/snmp_mibII/mibII_route.c
This commit was generated by cvs2svn to compensate for changes in r155518,
[FreeBSD/FreeBSD.git] / contrib / bsnmp / snmp_mibII / mibII_route.c
1 /*
2  * Copyright (c) 2001-2003
3  *      Fraunhofer Institute for Open Communication Systems (FhG Fokus).
4  *      All rights reserved.
5  *
6  * Author: Harti Brandt <harti@freebsd.org>
7  * 
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  * 
17  * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  *
29  * $Begemot: bsnmp/snmp_mibII/mibII_route.c,v 1.7 2005/06/09 12:36:53 brandt_h Exp $
30  *
31  * Routing table
32  */
33 #include <sys/tree.h>
34 #include "mibII.h"
35 #include "mibII_oid.h"
36
37 struct sroute {
38         RB_ENTRY(sroute) link;
39         uint32_t        ifindex;
40         uint8_t         index[13];
41         uint8_t         type;
42         uint8_t         proto;
43 };
44 RB_HEAD(sroutes, sroute) sroutes = RB_INITIALIZER(&sroutes);
45
46 RB_PROTOTYPE(sroutes, sroute, link, sroute_compare);
47
48 #define ROUTE_UPDATE_INTERVAL   (100 * 60 * 10) /* 10 min */
49 static uint64_t route_tick;
50 static u_int route_total;
51
52 /*
53  * Compare two routes
54  */
55 static int
56 sroute_compare(struct sroute *s1, struct sroute *s2)
57 {
58
59         return (memcmp(s1->index, s2->index, 13));
60 }
61
62 static void
63 sroute_index_append(struct asn_oid *oid, u_int sub, const struct sroute *s)
64 {
65         int i;
66
67         oid->len = sub + 13;
68         for (i = 0; i < 13; i++)
69                 oid->subs[sub + i] = s->index[i];
70 }
71
72 #if 0
73 static void
74 sroute_print(const struct sroute *r)
75 {
76         u_int i;
77
78         for (i = 0; i < 13 - 1; i++)
79                 printf("%u.", r->index[i]);
80         printf("%u proto=%u type=%u", r->index[i], r->proto, r->type);
81 }
82 #endif
83
84 /*
85  * process routing message
86  */
87 void
88 mib_sroute_process(struct rt_msghdr *rtm, struct sockaddr *gw,
89     struct sockaddr *dst, struct sockaddr *mask)
90 {
91         struct sockaddr_in *in_dst, *in_gw;
92         struct in_addr in_mask;
93         struct mibif *ifp;
94         struct sroute key;
95         struct sroute *r, *r1;
96         in_addr_t ha;
97
98         if (dst == NULL || gw == NULL || dst->sa_family != AF_INET ||
99             gw->sa_family != AF_INET)
100                 return;
101
102         in_dst = (struct sockaddr_in *)(void *)dst;
103         in_gw = (struct sockaddr_in *)(void *)gw;
104
105         if (rtm->rtm_flags & RTF_HOST)
106                 in_mask.s_addr = 0xffffffff;
107         else if (mask == NULL || mask->sa_len == 0)
108                 in_mask.s_addr = 0;
109         else
110                 in_mask = ((struct sockaddr_in *)(void *)mask)->sin_addr;
111
112         /* build the index */
113         ha = ntohl(in_dst->sin_addr.s_addr);
114         key.index[0] = (ha >> 24) & 0xff;
115         key.index[1] = (ha >> 16) & 0xff;
116         key.index[2] = (ha >>  8) & 0xff;
117         key.index[3] = (ha >>  0) & 0xff;
118
119         ha = ntohl(in_mask.s_addr);
120         key.index[4] = (ha >> 24) & 0xff;
121         key.index[5] = (ha >> 16) & 0xff;
122         key.index[6] = (ha >>  8) & 0xff;
123         key.index[7] = (ha >>  0) & 0xff;
124
125         /* ToS */
126         key.index[8] = 0;
127
128         ha = ntohl(in_gw->sin_addr.s_addr);
129         key.index[9] = (ha >> 24) & 0xff;
130         key.index[10] = (ha >> 16) & 0xff;
131         key.index[11] = (ha >>  8) & 0xff;
132         key.index[12] = (ha >>  0) & 0xff;
133
134         if (rtm->rtm_type == RTM_DELETE) {
135                 r = RB_FIND(sroutes, &sroutes, &key);
136                 if (r == 0) {
137 #ifdef DEBUG_ROUTE
138                         syslog(LOG_WARNING, "%s: DELETE: %u.%u.%u.%u "
139                             "%u.%u.%u.%u %u %u.%u.%u.%u not found", __func__,
140                             key.index[0], key.index[1], key.index[2],
141                             key.index[3], key.index[4], key.index[5],
142                             key.index[6], key.index[7], key.index[8],
143                             key.index[9], key.index[10], key.index[11],
144                             key.index[12]);
145 #endif
146                         return;
147                 }
148                 RB_REMOVE(sroutes, &sroutes, r);
149                 free(r);
150                 route_total--;
151 #ifdef DEBUG_ROUTE
152                 printf("%s: DELETE: %u.%u.%u.%u "
153                     "%u.%u.%u.%u %u %u.%u.%u.%u\n", __func__,
154                     key.index[0], key.index[1], key.index[2],
155                     key.index[3], key.index[4], key.index[5],
156                     key.index[6], key.index[7], key.index[8],
157                     key.index[9], key.index[10], key.index[11],
158                     key.index[12]);
159 #endif
160                 return;
161         }
162
163         /* GET or ADD */
164         ifp = NULL;
165         if ((ifp = mib_find_if_sys(rtm->rtm_index)) == NULL) {
166                 if (rtm->rtm_type == RTM_ADD) {
167                         /* make it a get so the kernel fills the index */
168                         mib_send_rtmsg(rtm, gw, dst, mask);
169                         return;
170                 }
171                 mib_iflist_bad = 1;
172         }
173
174         if ((r = malloc(sizeof(*r))) == NULL) {
175                 syslog(LOG_ERR, "%m");
176                 return;
177         }
178
179         memcpy(r->index, key.index, sizeof(r->index));
180         r->ifindex = (ifp == NULL) ? 0 : ifp->index;
181
182         r->type = (rtm->rtm_flags & RTF_LLINFO) ? 3 :
183             (rtm->rtm_flags & RTF_REJECT) ? 2 : 4;
184
185         /* cannot really know, what protocol it runs */
186         r->proto = (rtm->rtm_flags & RTF_LOCAL) ? 2 :
187             (rtm->rtm_flags & RTF_STATIC) ? 3 :
188             (rtm->rtm_flags & RTF_DYNAMIC) ? 4 : 10;
189
190         r1 = RB_INSERT(sroutes, &sroutes, r);
191         if (r1 != NULL) {
192 #ifdef DEBUG_ROUTE
193                 syslog(LOG_WARNING, "%s: %u.%u.%u.%u "
194                     "%u.%u.%u.%u %u %u.%u.%u.%u duplicate route", __func__,
195                     key.index[0], key.index[1], key.index[2],
196                     key.index[3], key.index[4], key.index[5],
197                     key.index[6], key.index[7], key.index[8],
198                     key.index[9], key.index[10], key.index[11],
199                     key.index[12]);
200 #endif
201                 r1->ifindex = r->ifindex;
202                 r1->type = r->type;
203                 r1->proto = r->proto;
204                 free(r);
205                 return;
206         }
207
208         route_total++;
209 #ifdef DEBUG_ROUTE
210         printf("%s: ADD/GET: %u.%u.%u.%u "
211             "%u.%u.%u.%u %u %u.%u.%u.%u\n", __func__,
212             key.index[0], key.index[1], key.index[2],
213             key.index[3], key.index[4], key.index[5],
214             key.index[6], key.index[7], key.index[8],
215             key.index[9], key.index[10], key.index[11],
216             key.index[12]);
217 #endif
218 }
219
220 int
221 mib_fetch_route(void)
222 {
223         u_char *rtab, *next;
224         size_t len;
225         struct sroute *r, *r1;
226         struct rt_msghdr *rtm;
227         struct sockaddr *addrs[RTAX_MAX];
228
229         if (route_tick != 0 && route_tick + ROUTE_UPDATE_INTERVAL > this_tick)
230                 return (0);
231
232         /*
233          * Remove all routes
234          */
235         r = RB_MIN(sroutes, &sroutes);
236         while (r != NULL) {
237                 r1 = RB_NEXT(sroutes, &sroutes, r);
238                 RB_REMOVE(sroutes, &sroutes, r);
239                 free(r);
240                 r = r1;
241         }
242         route_total = 0;
243
244         if ((rtab = mib_fetch_rtab(AF_INET, NET_RT_DUMP, 0, &len)) == NULL)
245                 return (-1);
246
247         next = rtab;
248         for (next = rtab; next < rtab + len; next += rtm->rtm_msglen) {
249                 rtm = (struct rt_msghdr *)(void *)next;
250                 if (rtm->rtm_type != RTM_GET ||
251                     !(rtm->rtm_flags & RTF_UP))
252                         continue;
253                 mib_extract_addrs(rtm->rtm_addrs, (u_char *)(rtm + 1), addrs);
254
255                 
256                 mib_sroute_process(rtm, addrs[RTAX_GATEWAY], addrs[RTAX_DST],
257                     addrs[RTAX_NETMASK]);
258         }
259
260 #if 0
261         u_int n = 0;
262         r = RB_MIN(sroutes, &sroutes);
263         while (r != NULL) {
264                 printf("%u: ", n++);
265                 sroute_print(r);
266                 printf("\n");
267                 r = RB_NEXT(sroutes, &sroutes, r);
268         }
269 #endif
270         free(rtab);
271         route_tick = get_ticks();
272
273         return (0);
274 }
275
276 /**
277  * Find a route in the table.
278  */
279 static struct sroute *
280 sroute_get(const struct asn_oid *oid, u_int sub)
281 {
282         struct sroute key;
283         int i;
284
285         if (oid->len - sub != 13)
286                 return (NULL);
287         for (i = 0; i < 13; i++)
288                 key.index[i] = oid->subs[sub + i];
289         return (RB_FIND(sroutes, &sroutes, &key));
290 }
291
292 /**
293  * Find next route in the table. There is no such RB_ macro, so must
294  * dig into the innards of the RB stuff.
295  */
296 static struct sroute *
297 sroute_getnext(struct asn_oid *oid, u_int sub)
298 {
299         u_int i;
300         int comp;
301         struct sroute key;
302         struct sroute *best;
303         struct sroute *s;
304
305         /*
306          * We now, that the OID is at least the tableEntry OID. If it is,
307          * the user wants the first route.
308          */
309         if (oid->len == sub)
310                 return (RB_MIN(sroutes, &sroutes));
311
312         /*
313          * This is also true for any index that consists of zeros and is
314          * shorter than the full index.
315          */
316         if (oid->len < sub + 13) {
317                 for (i = sub; i < oid->len; i++)
318                         if (oid->subs[i] != 0)
319                                 break;
320                 if (i == oid->len)
321                         return (RB_MIN(sroutes, &sroutes));
322
323                 /*
324                  * Now if the index is too short, we fill it with zeros and then
325                  * subtract one from the index. We can do this, because we now,
326                  * that there is at least one index element that is not zero.
327                  */
328                 for (i = oid->len; i < sub + 13; i++)
329                         oid->subs[i] = 0;
330
331                 for (i = sub + 13 - 1; i >= sub; i--) {
332                         if (oid->subs[i] != 0) {
333                                 oid->subs[i]--;
334                                 break;
335                         }
336                         oid->subs[i] = ASN_MAXID;
337                 }
338                 oid->len = sub + 13;
339         }
340
341         /* build the index */
342         for (i = sub; i < sub + 13; i++)
343                 key.index[i - sub] = oid->subs[i];
344
345         /* now find the element */
346         best = NULL;
347         s = RB_ROOT(&sroutes);
348
349         while (s != NULL) {
350                 comp = sroute_compare(&key, s);
351                 if (comp >= 0) {
352                         /* The current element is smaller than what we search.
353                          * Forget about it and move to the right subtree. */
354                         s = RB_RIGHT(s, link);
355                         continue;
356                 }
357                 /* the current element is larger than what we search.
358                  * forget about the right subtree (its even larger), but
359                  * the current element may be what we need. */
360                 if (best == NULL || sroute_compare(s, best) < 0)
361                         /* this one's better */
362                         best = s;
363
364                 s = RB_LEFT(s, link);
365         }
366         return (best);
367 }
368
369 /*
370  * Table
371  */
372 int
373 op_route_table(struct snmp_context *ctx __unused, struct snmp_value *value,
374     u_int sub, u_int iidx __unused, enum snmp_op op)
375 {
376         struct sroute *r;
377
378         if (mib_fetch_route() == -1)
379                 return (SNMP_ERR_GENERR);
380
381         switch (op) {
382
383           case SNMP_OP_GETNEXT:
384                 if ((r = sroute_getnext(&value->var, sub)) == NULL)
385                         return (SNMP_ERR_NOSUCHNAME);
386                 sroute_index_append(&value->var, sub, r);
387                 break;
388
389           case SNMP_OP_GET:
390                 if ((r = sroute_get(&value->var, sub)) == NULL)
391                         return (SNMP_ERR_NOSUCHNAME);
392                 break;
393
394           case SNMP_OP_SET:
395                 if ((r = sroute_get(&value->var, sub)) == NULL)
396                         return (SNMP_ERR_NOSUCHNAME);
397                 return (SNMP_ERR_NOT_WRITEABLE);
398
399           case SNMP_OP_ROLLBACK:
400           case SNMP_OP_COMMIT:
401                 abort();
402
403           default:
404                 abort();
405         }
406
407         switch (value->var.subs[sub - 1]) {
408
409           case LEAF_ipCidrRouteDest:
410                 value->v.ipaddress[0] = r->index[0];
411                 value->v.ipaddress[1] = r->index[1];
412                 value->v.ipaddress[2] = r->index[2];
413                 value->v.ipaddress[3] = r->index[3];
414                 break;
415
416           case LEAF_ipCidrRouteMask:
417                 value->v.ipaddress[0] = r->index[4];
418                 value->v.ipaddress[1] = r->index[5];
419                 value->v.ipaddress[2] = r->index[6];
420                 value->v.ipaddress[3] = r->index[7];
421                 break;
422
423           case LEAF_ipCidrRouteTos:
424                 value->v.integer = r->index[8];
425                 break;
426
427           case LEAF_ipCidrRouteNextHop:
428                 value->v.ipaddress[0] = r->index[9];
429                 value->v.ipaddress[1] = r->index[10];
430                 value->v.ipaddress[2] = r->index[11];
431                 value->v.ipaddress[3] = r->index[12];
432                 break;
433
434           case LEAF_ipCidrRouteIfIndex:
435                 value->v.integer = r->ifindex;
436                 break;
437
438           case LEAF_ipCidrRouteType:
439                 value->v.integer = r->type;
440                 break;
441
442           case LEAF_ipCidrRouteProto:
443                 value->v.integer = r->proto;
444                 break;
445
446           case LEAF_ipCidrRouteAge:
447                 value->v.integer = 0;
448                 break;
449
450           case LEAF_ipCidrRouteInfo:
451                 value->v.oid = oid_zeroDotZero;
452                 break;
453
454           case LEAF_ipCidrRouteNextHopAS:
455                 value->v.integer = 0;
456                 break;
457
458           case LEAF_ipCidrRouteMetric1:
459           case LEAF_ipCidrRouteMetric2:
460           case LEAF_ipCidrRouteMetric3:
461           case LEAF_ipCidrRouteMetric4:
462           case LEAF_ipCidrRouteMetric5:
463                 value->v.integer = -1;
464                 break;
465
466           case LEAF_ipCidrRouteStatus:
467                 value->v.integer = 1;
468                 break;
469         }
470         return (SNMP_ERR_NOERROR);
471 }
472
473 /*
474  * scalars
475  */
476 int
477 op_route(struct snmp_context *ctx __unused, struct snmp_value *value,
478     u_int sub, u_int iidx __unused, enum snmp_op op)
479 {
480         switch (op) {
481
482           case SNMP_OP_GETNEXT:
483                 abort();
484
485           case SNMP_OP_GET:
486                 break;
487
488           case SNMP_OP_SET:
489                 return (SNMP_ERR_NOT_WRITEABLE);
490
491           case SNMP_OP_ROLLBACK:
492           case SNMP_OP_COMMIT:
493                 abort();
494         }
495
496         if (mib_fetch_route() == -1)
497                 return (SNMP_ERR_GENERR);
498
499         switch (value->var.subs[sub - 1]) {
500
501           case LEAF_ipCidrRouteNumber:
502                 value->v.uint32 = route_total;
503                 break;
504
505         }
506         return (SNMP_ERR_NOERROR);
507 }
508
509 RB_GENERATE(sroutes, sroute, link, sroute_compare);