]> git.ipfire.org Git - thirdparty/bird.git/blame - sysdep/unix/krt.c
Added missing semicolons.
[thirdparty/bird.git] / sysdep / unix / krt.c
CommitLineData
2d140452
MM
1/*
2 * BIRD -- UNIX Kernel Synchronization
3 *
50fe90ed 4 * (c) 1998--2000 Martin Mares <mj@ucw.cz>
2d140452
MM
5 *
6 * Can be freely distributed and used under the terms of the GNU GPL.
7 */
8
73275d85
MM
9/**
10 * DOC: Kernel synchronization
11 *
12 * This system dependent module implements the Kernel and Device protocol,
13 * that is synchronization of interface lists and routing tables with the
14 * OS kernel.
15 *
16 * The whole kernel synchronization is a bit messy and touches some internals
17 * of the routing table engine, because routing table maintenance is a typical
18 * example of the proverbial compatibility between different Unices and we want
2e9b2421 19 * to keep the overhead of our KRT business as low as possible and avoid maintaining
73275d85
MM
20 * a local routing table copy.
21 *
22 * The kernel syncer can work in three different modes (according to system config header):
725270cb 23 * Either with a single routing table and single KRT protocol [traditional UNIX]
2e9b2421 24 * or with many routing tables and separate KRT protocols for all of them
73275d85 25 * or with many routing tables, but every scan including all tables, so we start
2e9b2421 26 * separate KRT protocols which cooperate with each other [Linux 2.2].
73275d85
MM
27 * In this case, we keep only a single scan timer.
28 *
58f7d004
MM
29 * We use FIB node flags in the routing table to keep track of route
30 * synchronization status. We also attach temporary &rte's to the routing table,
31 * but it cannot do any harm to the rest of BIRD since table synchronization is
32 * an atomic process.
73275d85
MM
33 *
34 * When starting up, we cheat by looking if there is another
35 * KRT instance to be initialized later and performing table scan
725270cb
MM
36 * only once for all the instances.
37 */
73275d85
MM
38
39/*
40 * If you are brave enough, continue now. You cannot say you haven't been warned.
41 */
42
832fa033 43#undef LOCAL_DEBUG
2d140452
MM
44
45#include "nest/bird.h"
46#include "nest/iface.h"
47#include "nest/route.h"
48#include "nest/protocol.h"
49#include "lib/timer.h"
7de45ba4 50#include "conf/conf.h"
2d140452
MM
51
52#include "unix.h"
53#include "krt.h"
54
c10421d3
MM
55static int krt_uptodate(rte *k, rte *e);
56
7e5f5ffd
MM
57/*
58 * Global resources
59 */
60
7de45ba4
MM
61pool *krt_pool;
62
7e5f5ffd
MM
63void
64krt_io_init(void)
65{
7de45ba4 66 krt_pool = rp_new(&root_pool, "Kernel Syncer");
7e5f5ffd
MM
67 krt_if_io_init();
68}
69
70/*
71 * Interfaces
72 */
73
74struct proto_config *cf_kif;
75
76static struct kif_proto *kif_proto;
77static timer *kif_scan_timer;
78static bird_clock_t kif_last_shot;
79
50fe90ed
MM
80static void
81kif_preconfig(struct protocol *P, struct config *c)
82{
83 cf_kif = NULL;
84}
85
7e5f5ffd
MM
86static void
87kif_scan(timer *t)
88{
89 struct kif_proto *p = t->data;
90
832fa033 91 KRT_TRACE(p, D_EVENTS, "Scanning interfaces");
7e5f5ffd
MM
92 kif_last_shot = now;
93 krt_if_scan(p);
94}
95
96static void
97kif_force_scan(void)
98{
99 if (kif_proto && kif_last_shot + 2 < now)
100 {
101 kif_scan(kif_scan_timer);
102 tm_start(kif_scan_timer, ((struct kif_config *) kif_proto->p.cf)->scan_time);
103 }
104}
105
106static struct proto *
107kif_init(struct proto_config *c)
108{
109 struct kif_proto *p = proto_new(c, sizeof(struct kif_proto));
110 return &p->p;
111}
112
113static int
114kif_start(struct proto *P)
115{
116 struct kif_proto *p = (struct kif_proto *) P;
117
118 kif_proto = p;
119 krt_if_start(p);
120
121 /* Start periodic interface scanning */
122 kif_scan_timer = tm_new(P->pool);
123 kif_scan_timer->hook = kif_scan;
124 kif_scan_timer->data = p;
125 kif_scan_timer->recurrent = KIF_CF->scan_time;
126 kif_scan(kif_scan_timer);
127 tm_start(kif_scan_timer, KIF_CF->scan_time);
128
129 return PS_UP;
130}
131
132static int
133kif_shutdown(struct proto *P)
134{
135 struct kif_proto *p = (struct kif_proto *) P;
136
137 tm_stop(kif_scan_timer);
138 krt_if_shutdown(p);
139 kif_proto = NULL;
140
141 if_start_update(); /* Remove all interfaces */
142 if_end_update();
7de45ba4
MM
143 /*
144 * FIXME: Is it really a good idea? It causes routes to be flushed,
145 * but at the same time it avoids sending of these deletions to the kernel,
146 * because krt thinks the kernel itself has already removed the route
147 * when downing the interface. Sad.
148 */
7e5f5ffd
MM
149
150 return PS_DOWN;
151}
152
f7fcb752
MM
153static int
154kif_reconfigure(struct proto *p, struct proto_config *new)
155{
156 struct kif_config *o = (struct kif_config *) p->cf;
157 struct kif_config *n = (struct kif_config *) new;
158
159 if (!kif_params_same(&o->iface, &n->iface))
160 return 0;
161 if (o->scan_time != n->scan_time)
162 {
163 tm_stop(kif_scan_timer);
164 kif_scan_timer->recurrent = n->scan_time;
165 kif_scan(kif_scan_timer);
166 tm_start(kif_scan_timer, n->scan_time);
167 }
168 return 1;
169}
170
7e5f5ffd
MM
171struct protocol proto_unix_iface = {
172 name: "Device",
d272fe22 173 template: "device%d",
50fe90ed 174 preconfig: kif_preconfig,
7e5f5ffd
MM
175 init: kif_init,
176 start: kif_start,
177 shutdown: kif_shutdown,
f7fcb752 178 reconfigure: kif_reconfigure,
7e5f5ffd 179};
2d140452 180
832fa033
MM
181/*
182 * Tracing of routes
183 */
184
185static void
186krt_trace_in_print(struct krt_proto *p, rte *e, char *msg)
187{
188 DBG("KRT: %I/%d: %s\n", e->net->n.prefix, e->net->n.pxlen, msg);
189 log(L_TRACE "%s: %I/%d: %s", p->p.name, e->net->n.prefix, e->net->n.pxlen, msg);
190}
191
192static inline void
193krt_trace_in(struct krt_proto *p, rte *e, char *msg)
194{
195 if (p->p.debug & D_PACKETS)
196 krt_trace_in_print(p, e, msg);
197}
198
c10421d3
MM
199/*
200 * Inherited Routes
201 */
202
203#ifdef KRT_ALLOW_LEARN
204
205static inline int
206krt_same_key(rte *a, rte *b)
207{
208 return a->u.krt.proto == b->u.krt.proto &&
209 a->u.krt.metric == b->u.krt.metric &&
210 a->u.krt.type == b->u.krt.type;
211}
212
213static void
214krt_learn_announce_update(struct krt_proto *p, rte *e)
215{
216 net *n = e->net;
217 rta *aa = rta_clone(e->attrs);
218 rte *ee = rte_get_temp(aa);
08e2d625 219 net *nn = net_get(p->p.table, n->n.prefix, n->n.pxlen);
c10421d3
MM
220 ee->net = nn;
221 ee->pflags = 0;
1151401e 222 ee->pref = p->p.preference;
c10421d3 223 ee->u.krt = e->u.krt;
4f1a6d27 224 rte_update(p->p.table, nn, &p->p, ee);
c10421d3
MM
225}
226
227static void
228krt_learn_announce_delete(struct krt_proto *p, net *n)
229{
08e2d625 230 n = net_find(p->p.table, n->n.prefix, n->n.pxlen);
c10421d3 231 if (n)
4f1a6d27 232 rte_update(p->p.table, n, &p->p, NULL);
c10421d3
MM
233}
234
235static void
236krt_learn_scan(struct krt_proto *p, rte *e)
237{
238 net *n0 = e->net;
08e2d625 239 net *n = net_get(&p->krt_table, n0->n.prefix, n0->n.pxlen);
c10421d3
MM
240 rte *m, **mm;
241
242 e->attrs->source = RTS_INHERIT;
243
244 for(mm=&n->routes; m = *mm; mm=&m->next)
245 if (krt_same_key(m, e))
246 break;
247 if (m)
248 {
249 if (krt_uptodate(m, e))
250 {
832fa033 251 krt_trace_in(p, e, "[alien] seen");
c10421d3
MM
252 rte_free(e);
253 m->u.krt.seen = 1;
254 }
255 else
256 {
832fa033 257 krt_trace_in(p, e, "[alien] updated");
c10421d3
MM
258 *mm = m->next;
259 rte_free(m);
260 m = NULL;
261 }
262 }
263 else
832fa033 264 krt_trace_in(p, e, "[alien] created");
c10421d3
MM
265 if (!m)
266 {
267 e->attrs = rta_lookup(e->attrs);
268 e->next = n->routes;
269 n->routes = e;
270 e->u.krt.seen = 1;
271 }
272}
273
c10421d3
MM
274static void
275krt_learn_prune(struct krt_proto *p)
276{
277 struct fib *fib = &p->krt_table.fib;
278 struct fib_iterator fit;
279
832fa033 280 KRT_TRACE(p, D_EVENTS, "Pruning inherited routes");
c10421d3
MM
281
282 FIB_ITERATE_INIT(&fit, fib);
283again:
284 FIB_ITERATE_START(fib, &fit, f)
285 {
286 net *n = (net *) f;
287 rte *e, **ee, *best, **pbest, *old_best;
288
289 old_best = n->routes;
290 best = NULL;
291 pbest = NULL;
292 ee = &n->routes;
293 while (e = *ee)
294 {
295 if (!e->u.krt.seen)
296 {
297 *ee = e->next;
298 rte_free(e);
299 continue;
300 }
301 if (!best || best->u.krt.metric > e->u.krt.metric)
302 {
303 best = e;
304 pbest = ee;
305 }
306 e->u.krt.seen = 0;
307 ee = &e->next;
308 }
309 if (!n->routes)
310 {
311 DBG("%I/%d: deleting\n", n->n.prefix, n->n.pxlen);
312 if (old_best)
313 {
314 krt_learn_announce_delete(p, n);
315 n->n.flags &= ~KRF_INSTALLED;
316 }
317 FIB_ITERATE_PUT(&fit, f);
318 fib_delete(fib, f);
319 goto again;
320 }
321 *pbest = best->next;
322 best->next = n->routes;
323 n->routes = best;
324 if (best != old_best || !(n->n.flags & KRF_INSTALLED))
325 {
326 DBG("%I/%d: announcing (metric=%d)\n", n->n.prefix, n->n.pxlen, best->u.krt.metric);
327 krt_learn_announce_update(p, best);
328 n->n.flags |= KRF_INSTALLED;
329 }
330 else
331 DBG("%I/%d: uptodate (metric=%d)\n", n->n.prefix, n->n.pxlen, best->u.krt.metric);
332 }
333 FIB_ITERATE_END(f);
334}
335
336static void
337krt_learn_async(struct krt_proto *p, rte *e, int new)
338{
339 net *n0 = e->net;
08e2d625 340 net *n = net_get(&p->krt_table, n0->n.prefix, n0->n.pxlen);
c10421d3
MM
341 rte *g, **gg, *best, **bestp, *old_best;
342
343 e->attrs->source = RTS_INHERIT;
344
345 old_best = n->routes;
346 for(gg=&n->routes; g = *gg; gg = &g->next)
347 if (krt_same_key(g, e))
348 break;
349 if (new)
350 {
351 if (g)
352 {
353 if (krt_uptodate(g, e))
354 {
832fa033 355 krt_trace_in(p, e, "[alien async] same");
c10421d3
MM
356 rte_free(e);
357 return;
358 }
832fa033 359 krt_trace_in(p, e, "[alien async] updated");
c10421d3
MM
360 *gg = g->next;
361 rte_free(g);
362 }
363 else
832fa033 364 krt_trace_in(p, e, "[alien async] created");
c10421d3
MM
365 e->attrs = rta_lookup(e->attrs);
366 e->next = n->routes;
367 n->routes = e;
368 }
369 else if (!g)
370 {
832fa033 371 krt_trace_in(p, e, "[alien async] delete failed");
c10421d3
MM
372 rte_free(e);
373 return;
374 }
375 else
376 {
832fa033 377 krt_trace_in(p, e, "[alien async] removed");
c10421d3
MM
378 *gg = g->next;
379 rte_free(e);
380 rte_free(g);
381 }
382 best = n->routes;
383 bestp = &n->routes;
384 for(gg=&n->routes; g=*gg; gg=&g->next)
385 if (best->u.krt.metric > g->u.krt.metric)
386 {
387 best = g;
388 bestp = gg;
389 }
390 if (best)
391 {
392 *bestp = best->next;
393 best->next = n->routes;
394 n->routes = best;
395 }
396 if (best != old_best)
397 {
398 DBG("krt_learn_async: distributing change\n");
399 if (best)
400 {
401 krt_learn_announce_update(p, best);
402 n->n.flags |= KRF_INSTALLED;
403 }
404 else
405 {
406 n->routes = NULL;
407 krt_learn_announce_delete(p, n);
408 n->n.flags &= ~KRF_INSTALLED;
409 }
410 }
411}
412
413static void
414krt_learn_init(struct krt_proto *p)
415{
416 if (KRT_CF->learn)
b9626ec6 417 rt_setup(p->p.pool, &p->krt_table, "Inherited", NULL);
c10421d3
MM
418}
419
420static void
421krt_dump(struct proto *P)
422{
423 struct krt_proto *p = (struct krt_proto *) P;
424
425 if (!KRT_CF->learn)
426 return;
427 debug("KRT: Table of inheritable routes\n");
428 rt_dump(&p->krt_table);
429}
430
431static void
432krt_dump_attrs(rte *e)
433{
434 debug(" [m=%d,p=%d,t=%d]", e->u.krt.metric, e->u.krt.proto, e->u.krt.type);
435}
436
437#endif
438
2d140452
MM
439/*
440 * Routes
441 */
442
7de45ba4
MM
443#ifdef CONFIG_ALL_TABLES_AT_ONCE
444static timer *krt_scan_timer;
445static int krt_instance_count;
446static list krt_instance_list;
447#endif
448
2d140452
MM
449static void
450krt_flush_routes(struct krt_proto *p)
451{
4f1a6d27 452 struct rtable *t = p->p.table;
2d140452 453
832fa033 454 KRT_TRACE(p, D_EVENTS, "Flushing kernel routes");
2d140452
MM
455 FIB_WALK(&t->fib, f)
456 {
457 net *n = (net *) f;
458 rte *e = n->routes;
459 if (e)
460 {
461 rta *a = e->attrs;
462 if (a->source != RTS_DEVICE && a->source != RTS_INHERIT)
c10421d3 463 krt_set_notify(p, e->net, NULL, e);
2d140452
MM
464 }
465 }
466 FIB_WALK_END;
467}
468
2d140452
MM
469static int
470krt_uptodate(rte *k, rte *e)
471{
472 rta *ka = k->attrs, *ea = e->attrs;
473
474 if (ka->dest != ea->dest)
475 return 0;
476 switch (ka->dest)
477 {
478 case RTD_ROUTER:
479 return ipa_equal(ka->gw, ea->gw);
480 case RTD_DEVICE:
481 return !strcmp(ka->iface->name, ea->iface->name);
482 default:
483 return 1;
484 }
485}
486
487/*
488 * This gets called back when the low-level scanning code discovers a route.
489 * We expect that the route is a temporary rte and its attributes are uncached.
490 */
491
492void
493krt_got_route(struct krt_proto *p, rte *e)
494{
495 rte *old;
496 net *net = e->net;
c10421d3 497 int src = e->u.krt.src;
2d140452
MM
498 int verdict;
499
c10421d3
MM
500#ifdef CONFIG_AUTO_ROUTES
501 if (e->attrs->dest == RTD_DEVICE)
502 {
503 /* It's a device route. Probably a kernel-generated one. */
504 verdict = KRF_IGNORE;
505 goto sentenced;
506 }
507#endif
508
509#ifdef KRT_ALLOW_LEARN
510 if (src == KRT_SRC_ALIEN)
511 {
512 if (KRT_CF->learn)
513 krt_learn_scan(p, e);
514 else
832fa033 515 krt_trace_in(p, e, "alien route, ignored");
c10421d3
MM
516 return;
517 }
518#endif
519
520 if (net->n.flags & KRF_VERDICT_MASK)
2d140452
MM
521 {
522 /* Route to this destination was already seen. Strange, but it happens... */
832fa033 523 krt_trace_in(p, e, "already seen");
2d140452
MM
524 return;
525 }
526
c10421d3 527 if (net->n.flags & KRF_INSTALLED)
2d140452 528 {
c10421d3
MM
529 old = net->routes;
530 ASSERT(old);
531 if (krt_uptodate(e, old))
2d140452
MM
532 verdict = KRF_SEEN;
533 else
534 verdict = KRF_UPDATE;
535 }
2d140452
MM
536 else
537 verdict = KRF_DELETE;
538
c10421d3 539sentenced:
832fa033 540 krt_trace_in(p, e, ((char *[]) { "?", "seen", "will be updated", "will be removed", "ignored" }) [verdict]);
c10421d3
MM
541 net->n.flags = (net->n.flags & ~KRF_VERDICT_MASK) | verdict;
542 if (verdict == KRF_UPDATE || verdict == KRF_DELETE)
2d140452
MM
543 {
544 /* Get a cached copy of attributes and link the route */
545 rta *a = e->attrs;
546 a->source = RTS_DUMMY;
547 e->attrs = rta_lookup(a);
548 e->next = net->routes;
549 net->routes = e;
550 }
551 else
552 rte_free(e);
553}
554
555static void
556krt_prune(struct krt_proto *p)
557{
4f1a6d27 558 struct rtable *t = p->p.table;
2d140452 559
832fa033 560 KRT_TRACE(p, D_EVENTS, "Pruning table %s", t->name);
2d140452
MM
561 FIB_WALK(&t->fib, f)
562 {
563 net *n = (net *) f;
c10421d3 564 int verdict = f->flags & KRF_VERDICT_MASK;
2d140452
MM
565 rte *new, *old;
566
c10421d3 567 if (verdict != KRF_CREATE && verdict != KRF_SEEN && verdict != KRF_IGNORE)
2d140452
MM
568 {
569 old = n->routes;
570 n->routes = old->next;
571 }
572 else
573 old = NULL;
574 new = n->routes;
575
576 switch (verdict)
577 {
578 case KRF_CREATE:
c10421d3 579 if (new && (f->flags & KRF_INSTALLED))
2d140452 580 {
832fa033 581 krt_trace_in(p, new, "reinstalling");
c10421d3 582 krt_set_notify(p, n, new, NULL);
2d140452
MM
583 }
584 break;
585 case KRF_SEEN:
c10421d3 586 case KRF_IGNORE:
2d140452
MM
587 /* Nothing happens */
588 break;
589 case KRF_UPDATE:
832fa033 590 krt_trace_in(p, new, "updating");
c10421d3 591 krt_set_notify(p, n, new, old);
2d140452
MM
592 break;
593 case KRF_DELETE:
1151401e 594 krt_trace_in(p, old, "deleting");
c10421d3 595 krt_set_notify(p, n, NULL, old);
2d140452
MM
596 break;
597 default:
598 bug("krt_prune: invalid route status");
599 }
2d140452
MM
600 if (old)
601 rte_free(old);
c10421d3 602 f->flags &= ~KRF_VERDICT_MASK;
2d140452
MM
603 }
604 FIB_WALK_END;
c10421d3
MM
605
606#ifdef KRT_ALLOW_LEARN
607 if (KRT_CF->learn)
608 krt_learn_prune(p);
609#endif
aa8761de 610 p->initialized = 1;
2d140452
MM
611}
612
e16155ae
MM
613void
614krt_got_route_async(struct krt_proto *p, rte *e, int new)
615{
616 net *net = e->net;
c10421d3 617 int src = e->u.krt.src;
e16155ae
MM
618
619 switch (src)
620 {
621 case KRT_SRC_BIRD:
832fa033 622 ASSERT(0); /* Should be filtered by the back end */
e16155ae
MM
623 case KRT_SRC_REDIRECT:
624 DBG("It's a redirect, kill him! Kill! Kill!\n");
c10421d3 625 krt_set_notify(p, net, NULL, e);
e16155ae 626 break;
c10421d3
MM
627 case KRT_SRC_ALIEN:
628#ifdef KRT_ALLOW_LEARN
629 if (KRT_CF->learn)
e16155ae 630 {
c10421d3
MM
631 krt_learn_async(p, e, new);
632 return;
e16155ae 633 }
c10421d3 634#endif
e16155ae 635 }
c10421d3 636 rte_free(e);
e16155ae
MM
637}
638
2d140452
MM
639/*
640 * Periodic scanning
641 */
642
2d140452
MM
643static void
644krt_scan(timer *t)
645{
7de45ba4 646 struct krt_proto *p;
2d140452 647
7e5f5ffd 648 kif_force_scan();
7de45ba4
MM
649#ifdef CONFIG_ALL_TABLES_AT_ONCE
650 {
651 void *q;
832fa033
MM
652 /* We need some node to decide whether to print the debug messages or not */
653 p = SKIP_BACK(struct krt_proto, instance_node, HEAD(krt_instance_list));
654 if (p->instance_node.next)
655 KRT_TRACE(p, D_EVENTS, "Scanning routing table");
7de45ba4
MM
656 krt_scan_fire(NULL);
657 WALK_LIST(q, krt_instance_list)
658 {
659 p = SKIP_BACK(struct krt_proto, instance_node, q);
660 krt_prune(p);
661 }
662 }
663#else
664 p = t->data;
832fa033 665 KRT_TRACE(p, D_EVENTS, "Scanning routing table");
7e5f5ffd
MM
666 krt_scan_fire(p);
667 krt_prune(p);
7de45ba4 668#endif
2d140452
MM
669}
670
c10421d3
MM
671/*
672 * Updates
673 */
674
675static void
02bd064a 676krt_notify(struct proto *P, net *net, rte *new, rte *old, struct ea_list *attrs)
c10421d3
MM
677{
678 struct krt_proto *p = (struct krt_proto *) P;
679
f990fc61
MM
680 if (shutting_down && KRT_CF->persist)
681 return;
c10421d3
MM
682 if (new && (!krt_capable(new) || new->attrs->source == RTS_INHERIT))
683 new = NULL;
684 if (!(net->n.flags & KRF_INSTALLED))
685 old = NULL;
686 if (new)
687 net->n.flags |= KRF_INSTALLED;
688 else
689 net->n.flags &= ~KRF_INSTALLED;
aa8761de
MM
690 if (p->initialized) /* Before first scan we don't touch the routes */
691 krt_set_notify(p, net, new, old);
c10421d3
MM
692}
693
2d140452
MM
694/*
695 * Protocol glue
696 */
697
7e5f5ffd
MM
698struct proto_config *cf_krt;
699
7de45ba4
MM
700static void
701krt_preconfig(struct protocol *P, struct config *c)
702{
50fe90ed 703 cf_krt = NULL;
7de45ba4
MM
704 krt_scan_preconfig(c);
705}
706
707static void
708krt_postconfig(struct proto_config *C)
709{
710 struct krt_config *c = (struct krt_config *) C;
711
712#ifdef CONFIG_ALL_TABLES_AT_ONCE
713 struct krt_config *first = (struct krt_config *) cf_krt;
714 if (first->scan_time != c->scan_time)
715 cf_error("All kernel syncers must use the same table scan interval");
716#endif
717
718 if (C->table->krt_attached)
719 cf_error("Kernel syncer (%s) already attached to table %s", C->table->krt_attached->name, C->table->name);
720 C->table->krt_attached = C;
721 krt_scan_postconfig(c);
722}
723
724static timer *
725krt_start_timer(struct krt_proto *p)
726{
727 timer *t;
728
729 t = tm_new(p->krt_pool);
730 t->hook = krt_scan;
731 t->data = p;
732 t->recurrent = KRT_CF->scan_time;
aa8761de 733 tm_start(t, 0);
7de45ba4
MM
734 return t;
735}
736
2d140452
MM
737static int
738krt_start(struct proto *P)
739{
740 struct krt_proto *p = (struct krt_proto *) P;
7de45ba4
MM
741 int first = 1;
742
743#ifdef CONFIG_ALL_TABLES_AT_ONCE
744 if (!krt_instance_count++)
745 init_list(&krt_instance_list);
746 else
747 first = 0;
748 p->krt_pool = krt_pool;
749 add_tail(&krt_instance_list, &p->instance_node);
750#else
751 p->krt_pool = P->pool;
752#endif
2d140452 753
c10421d3
MM
754#ifdef KRT_ALLOW_LEARN
755 krt_learn_init(p);
756#endif
757
7de45ba4
MM
758 krt_scan_start(p, first);
759 krt_set_start(p, first);
2d140452 760
7e5f5ffd 761 /* Start periodic routing table scanning */
7de45ba4
MM
762#ifdef CONFIG_ALL_TABLES_AT_ONCE
763 if (first)
764 krt_scan_timer = krt_start_timer(p);
aa8761de
MM
765 else
766 tm_start(p->scan_timer, 0);
7de45ba4 767 p->scan_timer = krt_scan_timer;
7de45ba4
MM
768#else
769 p->scan_timer = krt_start_timer(p);
7de45ba4 770#endif
2d140452
MM
771
772 return PS_UP;
773}
774
7de45ba4 775static int
2d140452
MM
776krt_shutdown(struct proto *P)
777{
778 struct krt_proto *p = (struct krt_proto *) P;
7de45ba4 779 int last = 1;
2d140452 780
7de45ba4
MM
781#ifdef CONFIG_ALL_TABLES_AT_ONCE
782 rem_node(&p->instance_node);
783 if (--krt_instance_count)
784 last = 0;
785 else
786#endif
787 tm_stop(p->scan_timer);
7e5f5ffd 788
2d140452
MM
789 if (!KRT_CF->persist)
790 krt_flush_routes(p);
791
7de45ba4
MM
792 krt_set_shutdown(p, last);
793 krt_scan_shutdown(p, last);
794
795#ifdef CONFIG_ALL_TABLES_AT_ONCE
796 if (last)
797 rfree(krt_scan_timer);
798#endif
2d140452 799
2d140452
MM
800 return PS_DOWN;
801}
802
2d140452
MM
803static struct proto *
804krt_init(struct proto_config *c)
805{
806 struct krt_proto *p = proto_new(c, sizeof(struct krt_proto));
807
c10421d3 808 p->p.rt_notify = krt_notify;
0da472d7 809 p->p.min_scope = SCOPE_HOST;
2d140452
MM
810 return &p->p;
811}
812
aa8761de
MM
813static int
814krt_reconfigure(struct proto *p, struct proto_config *new)
815{
816 struct krt_config *o = (struct krt_config *) p->cf;
817 struct krt_config *n = (struct krt_config *) new;
818
819 return o->scan_time == n->scan_time
820 && o->learn == n->learn /* persist needn't be the same */
821 && krt_set_params_same(&o->set, &n->set)
822 && krt_scan_params_same(&o->scan, &n->scan)
823 ;
824}
825
2d140452
MM
826struct protocol proto_unix_kernel = {
827 name: "Kernel",
d272fe22 828 template: "kernel%d",
7de45ba4
MM
829 preconfig: krt_preconfig,
830 postconfig: krt_postconfig,
2d140452
MM
831 init: krt_init,
832 start: krt_start,
833 shutdown: krt_shutdown,
aa8761de 834 reconfigure: krt_reconfigure,
c10421d3
MM
835#ifdef KRT_ALLOW_LEARN
836 dump: krt_dump,
837 dump_attrs: krt_dump_attrs,
838#endif
2d140452 839};