Annotation of embedaddon/quagga/babeld/route.c, revision 1.1
1.1 ! misho 1: /*
! 2: * This file is free software: you may copy, redistribute and/or modify it
! 3: * under the terms of the GNU General Public License as published by the
! 4: * Free Software Foundation, either version 2 of the License, or (at your
! 5: * option) any later version.
! 6: *
! 7: * This file is distributed in the hope that it will be useful, but
! 8: * WITHOUT ANY WARRANTY; without even the implied warranty of
! 9: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
! 10: * General Public License for more details.
! 11: *
! 12: * You should have received a copy of the GNU General Public License
! 13: * along with this program. If not, see <http://www.gnu.org/licenses/>.
! 14: *
! 15: * This file incorporates work covered by the following copyright and
! 16: * permission notice:
! 17: *
! 18: Copyright (c) 2007, 2008 by Juliusz Chroboczek
! 19: Copyright 2011 by Matthieu Boutier and Juliusz Chroboczek
! 20:
! 21: Permission is hereby granted, free of charge, to any person obtaining a copy
! 22: of this software and associated documentation files (the "Software"), to deal
! 23: in the Software without restriction, including without limitation the rights
! 24: to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
! 25: copies of the Software, and to permit persons to whom the Software is
! 26: furnished to do so, subject to the following conditions:
! 27:
! 28: The above copyright notice and this permission notice shall be included in
! 29: all copies or substantial portions of the Software.
! 30:
! 31: THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
! 32: IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
! 33: FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
! 34: AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
! 35: LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
! 36: OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
! 37: THE SOFTWARE.
! 38: */
! 39:
! 40: #include <zebra.h>
! 41: #include "if.h"
! 42:
! 43: #include "babeld.h"
! 44: #include "util.h"
! 45: #include "kernel.h"
! 46: #include "babel_interface.h"
! 47: #include "source.h"
! 48: #include "neighbour.h"
! 49: #include "route.h"
! 50: #include "xroute.h"
! 51: #include "message.h"
! 52: #include "resend.h"
! 53:
! 54: static void consider_route(struct babel_route *route);
! 55:
! 56: struct babel_route **routes = NULL;
! 57: static int route_slots = 0, max_route_slots = 0;
! 58: int kernel_metric = 0;
! 59: int allow_duplicates = -1;
! 60: int diversity_kind = DIVERSITY_NONE;
! 61: int diversity_factor = 256; /* in units of 1/256 */
! 62: int keep_unfeasible = 0;
! 63:
! 64: /* We maintain a list of "slots", ordered by prefix. Every slot
! 65: contains a linked list of the routes to this prefix, with the
! 66: installed route, if any, at the head of the list. */
! 67:
! 68: static int
! 69: route_compare(const unsigned char *prefix, unsigned char plen,
! 70: struct babel_route *route)
! 71: {
! 72: int i = memcmp(prefix, route->src->prefix, 16);
! 73: if(i != 0)
! 74: return i;
! 75:
! 76: if(plen < route->src->plen)
! 77: return -1;
! 78: else if(plen > route->src->plen)
! 79: return 1;
! 80: else
! 81: return 0;
! 82: }
! 83:
! 84: /* Performs binary search, returns -1 in case of failure. In the latter
! 85: case, new_return is the place where to insert the new element. */
! 86:
! 87: static int
! 88: find_route_slot(const unsigned char *prefix, unsigned char plen,
! 89: int *new_return)
! 90: {
! 91: int p, m, g, c;
! 92:
! 93: if(route_slots < 1) {
! 94: if(new_return)
! 95: *new_return = 0;
! 96: return -1;
! 97: }
! 98:
! 99: p = 0; g = route_slots - 1;
! 100:
! 101: do {
! 102: m = (p + g) / 2;
! 103: c = route_compare(prefix, plen, routes[m]);
! 104: if(c == 0)
! 105: return m;
! 106: else if(c < 0)
! 107: g = m - 1;
! 108: else
! 109: p = m + 1;
! 110: } while(p <= g);
! 111:
! 112: if(new_return)
! 113: *new_return = p;
! 114:
! 115: return -1;
! 116: }
! 117:
! 118: struct babel_route *
! 119: find_route(const unsigned char *prefix, unsigned char plen,
! 120: struct neighbour *neigh, const unsigned char *nexthop)
! 121: {
! 122: struct babel_route *route;
! 123: int i = find_route_slot(prefix, plen, NULL);
! 124:
! 125: if(i < 0)
! 126: return NULL;
! 127:
! 128: route = routes[i];
! 129:
! 130: while(route) {
! 131: if(route->neigh == neigh && memcmp(route->nexthop, nexthop, 16) == 0)
! 132: return route;
! 133: route = route->next;
! 134: }
! 135:
! 136: return NULL;
! 137: }
! 138:
! 139: struct babel_route *
! 140: find_installed_route(const unsigned char *prefix, unsigned char plen)
! 141: {
! 142: int i = find_route_slot(prefix, plen, NULL);
! 143:
! 144: if(i >= 0 && routes[i]->installed)
! 145: return routes[i];
! 146:
! 147: return NULL;
! 148: }
! 149:
! 150: /* Returns an overestimate of the number of installed routes. */
! 151: int
! 152: installed_routes_estimate(void)
! 153: {
! 154: return route_slots;
! 155: }
! 156:
! 157: static int
! 158: resize_route_table(int new_slots)
! 159: {
! 160: struct babel_route **new_routes;
! 161: assert(new_slots >= route_slots);
! 162:
! 163: if(new_slots == 0) {
! 164: new_routes = NULL;
! 165: free(routes);
! 166: } else {
! 167: new_routes = realloc(routes, new_slots * sizeof(struct babel_route*));
! 168: if(new_routes == NULL)
! 169: return -1;
! 170: }
! 171:
! 172: max_route_slots = new_slots;
! 173: routes = new_routes;
! 174: return 1;
! 175: }
! 176:
! 177: /* Insert a route into the table. If successful, retains the route.
! 178: On failure, caller must free the route. */
! 179: static struct babel_route *
! 180: insert_route(struct babel_route *route)
! 181: {
! 182: int i, n;
! 183:
! 184: assert(!route->installed);
! 185:
! 186: i = find_route_slot(route->src->prefix, route->src->plen, &n);
! 187:
! 188: if(i < 0) {
! 189: if(route_slots >= max_route_slots)
! 190: resize_route_table(max_route_slots < 1 ? 8 : 2 * max_route_slots);
! 191: if(route_slots >= max_route_slots)
! 192: return NULL;
! 193: route->next = NULL;
! 194: if(n < route_slots)
! 195: memmove(routes + n + 1, routes + n,
! 196: (route_slots - n) * sizeof(struct babel_route*));
! 197: route_slots++;
! 198: routes[n] = route;
! 199: } else {
! 200: struct babel_route *r;
! 201: r = routes[i];
! 202: while(r->next)
! 203: r = r->next;
! 204: r->next = route;
! 205: route->next = NULL;
! 206: }
! 207:
! 208: return route;
! 209: }
! 210:
! 211: void
! 212: flush_route(struct babel_route *route)
! 213: {
! 214: int i;
! 215: struct source *src;
! 216: unsigned oldmetric;
! 217: int lost = 0;
! 218:
! 219: oldmetric = route_metric(route);
! 220: src = route->src;
! 221:
! 222: if(route->installed) {
! 223: uninstall_route(route);
! 224: lost = 1;
! 225: }
! 226:
! 227: i = find_route_slot(route->src->prefix, route->src->plen, NULL);
! 228: assert(i >= 0 && i < route_slots);
! 229:
! 230: if(route == routes[i]) {
! 231: routes[i] = route->next;
! 232: route->next = NULL;
! 233: free(route);
! 234:
! 235: if(routes[i] == NULL) {
! 236: if(i < route_slots - 1)
! 237: memmove(routes + i, routes + i + 1,
! 238: (route_slots - i - 1) * sizeof(struct babel_route*));
! 239: routes[route_slots - 1] = NULL;
! 240: route_slots--;
! 241: }
! 242:
! 243: if(route_slots == 0)
! 244: resize_route_table(0);
! 245: else if(max_route_slots > 8 && route_slots < max_route_slots / 4)
! 246: resize_route_table(max_route_slots / 2);
! 247: } else {
! 248: struct babel_route *r = routes[i];
! 249: while(r->next != route)
! 250: r = r->next;
! 251: r->next = route->next;
! 252: route->next = NULL;
! 253: free(route);
! 254: }
! 255:
! 256: if(lost)
! 257: route_lost(src, oldmetric);
! 258:
! 259: release_source(src);
! 260: }
! 261:
! 262: void
! 263: flush_all_routes()
! 264: {
! 265: int i;
! 266:
! 267: /* Start from the end, to avoid shifting the table. */
! 268: i = route_slots - 1;
! 269: while(i >= 0) {
! 270: while(i < route_slots) {
! 271: /* Uninstall first, to avoid calling route_lost. */
! 272: if(routes[i]->installed)
! 273: uninstall_route(routes[0]);
! 274: flush_route(routes[i]);
! 275: }
! 276: i--;
! 277: }
! 278:
! 279: check_sources_released();
! 280: }
! 281:
! 282: void
! 283: flush_neighbour_routes(struct neighbour *neigh)
! 284: {
! 285: int i;
! 286:
! 287: i = 0;
! 288: while(i < route_slots) {
! 289: struct babel_route *r;
! 290: r = routes[i];
! 291: while(r) {
! 292: if(r->neigh == neigh) {
! 293: flush_route(r);
! 294: goto again;
! 295: }
! 296: r = r->next;
! 297: }
! 298: i++;
! 299: again:
! 300: ;
! 301: }
! 302: }
! 303:
! 304: void
! 305: flush_interface_routes(struct interface *ifp, int v4only)
! 306: {
! 307: int i;
! 308:
! 309: i = 0;
! 310: while(i < route_slots) {
! 311: struct babel_route *r;
! 312: r = routes[i];
! 313: while(r) {
! 314: if(r->neigh->ifp == ifp &&
! 315: (!v4only || v4mapped(r->nexthop))) {
! 316: flush_route(r);
! 317: goto again;
! 318: }
! 319: r = r->next;
! 320: }
! 321: i++;
! 322: again:
! 323: ;
! 324: }
! 325: }
! 326:
! 327: /* Iterate a function over all routes. */
! 328: void
! 329: for_all_routes(void (*f)(struct babel_route*, void*), void *closure)
! 330: {
! 331: int i;
! 332:
! 333: for(i = 0; i < route_slots; i++) {
! 334: struct babel_route *r = routes[i];
! 335: while(r) {
! 336: (*f)(r, closure);
! 337: r = r->next;
! 338: }
! 339: }
! 340: }
! 341:
! 342: void
! 343: for_all_installed_routes(void (*f)(struct babel_route*, void*), void *closure)
! 344: {
! 345: int i;
! 346:
! 347: for(i = 0; i < route_slots; i++) {
! 348: if(routes[i]->installed)
! 349: (*f)(routes[i], closure);
! 350: }
! 351: }
! 352:
! 353: static int
! 354: metric_to_kernel(int metric)
! 355: {
! 356: return metric < INFINITY ? kernel_metric : KERNEL_INFINITY;
! 357: }
! 358:
! 359: /* This is used to maintain the invariant that the installed route is at
! 360: the head of the list. */
! 361: static void
! 362: move_installed_route(struct babel_route *route, int i)
! 363: {
! 364: assert(i >= 0 && i < route_slots);
! 365: assert(route->installed);
! 366:
! 367: if(route != routes[i]) {
! 368: struct babel_route *r = routes[i];
! 369: while(r->next != route)
! 370: r = r->next;
! 371: r->next = route->next;
! 372: route->next = routes[i];
! 373: routes[i] = route;
! 374: }
! 375: }
! 376:
! 377: void
! 378: install_route(struct babel_route *route)
! 379: {
! 380: int i, rc;
! 381:
! 382: if(route->installed)
! 383: return;
! 384:
! 385: if(!route_feasible(route))
! 386: zlog_err("WARNING: installing unfeasible route "
! 387: "(this shouldn't happen).");
! 388:
! 389: i = find_route_slot(route->src->prefix, route->src->plen, NULL);
! 390: assert(i >= 0 && i < route_slots);
! 391:
! 392: if(routes[i] != route && routes[i]->installed) {
! 393: fprintf(stderr, "WARNING: attempting to install duplicate route "
! 394: "(this shouldn't happen).");
! 395: return;
! 396: }
! 397:
! 398: rc = kernel_route(ROUTE_ADD, route->src->prefix, route->src->plen,
! 399: route->nexthop,
! 400: route->neigh->ifp->ifindex,
! 401: metric_to_kernel(route_metric(route)), NULL, 0, 0);
! 402: if(rc < 0) {
! 403: int save = errno;
! 404: zlog_err("kernel_route(ADD): %s", safe_strerror(errno));
! 405: if(save != EEXIST)
! 406: return;
! 407: }
! 408: route->installed = 1;
! 409: move_installed_route(route, i);
! 410:
! 411: }
! 412:
! 413: void
! 414: uninstall_route(struct babel_route *route)
! 415: {
! 416: int rc;
! 417:
! 418: if(!route->installed)
! 419: return;
! 420:
! 421: rc = kernel_route(ROUTE_FLUSH, route->src->prefix, route->src->plen,
! 422: route->nexthop,
! 423: route->neigh->ifp->ifindex,
! 424: metric_to_kernel(route_metric(route)), NULL, 0, 0);
! 425: if(rc < 0)
! 426: zlog_err("kernel_route(FLUSH): %s", safe_strerror(errno));
! 427:
! 428: route->installed = 0;
! 429: }
! 430:
! 431: /* This is equivalent to uninstall_route followed with install_route,
! 432: but without the race condition. The destination of both routes
! 433: must be the same. */
! 434:
! 435: static void
! 436: switch_routes(struct babel_route *old, struct babel_route *new)
! 437: {
! 438: int rc;
! 439:
! 440: if(!old) {
! 441: install_route(new);
! 442: return;
! 443: }
! 444:
! 445: if(!old->installed)
! 446: return;
! 447:
! 448: if(!route_feasible(new))
! 449: zlog_err("WARNING: switching to unfeasible route "
! 450: "(this shouldn't happen).");
! 451:
! 452: rc = kernel_route(ROUTE_MODIFY, old->src->prefix, old->src->plen,
! 453: old->nexthop, old->neigh->ifp->ifindex,
! 454: metric_to_kernel(route_metric(old)),
! 455: new->nexthop, new->neigh->ifp->ifindex,
! 456: metric_to_kernel(route_metric(new)));
! 457: if(rc < 0) {
! 458: zlog_err("kernel_route(MODIFY): %s", safe_strerror(errno));
! 459: return;
! 460: }
! 461:
! 462: old->installed = 0;
! 463: new->installed = 1;
! 464: move_installed_route(new, find_route_slot(new->src->prefix, new->src->plen,
! 465: NULL));
! 466: }
! 467:
! 468: static void
! 469: change_route_metric(struct babel_route *route,
! 470: unsigned refmetric, unsigned cost, unsigned add)
! 471: {
! 472: int old, new;
! 473: int newmetric = MIN(refmetric + cost + add, INFINITY);
! 474:
! 475: old = metric_to_kernel(route_metric(route));
! 476: new = metric_to_kernel(newmetric);
! 477:
! 478: if(route->installed && old != new) {
! 479: int rc;
! 480: rc = kernel_route(ROUTE_MODIFY, route->src->prefix, route->src->plen,
! 481: route->nexthop, route->neigh->ifp->ifindex,
! 482: old,
! 483: route->nexthop, route->neigh->ifp->ifindex,
! 484: new);
! 485: if(rc < 0) {
! 486: zlog_err("kernel_route(MODIFY metric): %s", safe_strerror(errno));
! 487: return;
! 488: }
! 489: }
! 490:
! 491: route->refmetric = refmetric;
! 492: route->cost = cost;
! 493: route->add_metric = add;
! 494: }
! 495:
! 496: static void
! 497: retract_route(struct babel_route *route)
! 498: {
! 499: change_route_metric(route, INFINITY, INFINITY, 0);
! 500: }
! 501:
! 502: int
! 503: route_feasible(struct babel_route *route)
! 504: {
! 505: return update_feasible(route->src, route->seqno, route->refmetric);
! 506: }
! 507:
! 508: int
! 509: route_old(struct babel_route *route)
! 510: {
! 511: return route->time < babel_now.tv_sec - route->hold_time * 7 / 8;
! 512: }
! 513:
! 514: int
! 515: route_expired(struct babel_route *route)
! 516: {
! 517: return route->time < babel_now.tv_sec - route->hold_time;
! 518: }
! 519:
! 520: static int
! 521: channels_interfere(int ch1, int ch2)
! 522: {
! 523: if(ch1 == BABEL_IF_CHANNEL_NONINTERFERING
! 524: || ch2 == BABEL_IF_CHANNEL_NONINTERFERING)
! 525: return 0;
! 526: if(ch1 == BABEL_IF_CHANNEL_INTERFERING
! 527: || ch2 == BABEL_IF_CHANNEL_INTERFERING)
! 528: return 1;
! 529: return ch1 == ch2;
! 530: }
! 531:
! 532: int
! 533: route_interferes(struct babel_route *route, struct interface *ifp)
! 534: {
! 535: struct babel_interface *babel_ifp = NULL;
! 536: switch(diversity_kind) {
! 537: case DIVERSITY_NONE:
! 538: return 1;
! 539: case DIVERSITY_INTERFACE_1:
! 540: return route->neigh->ifp == ifp;
! 541: case DIVERSITY_CHANNEL_1:
! 542: case DIVERSITY_CHANNEL:
! 543: if(route->neigh->ifp == ifp)
! 544: return 1;
! 545: babel_ifp = babel_get_if_nfo(ifp);
! 546: if(channels_interfere(babel_ifp->channel,
! 547: babel_get_if_nfo(route->neigh->ifp)->channel))
! 548: return 1;
! 549: if(diversity_kind == DIVERSITY_CHANNEL) {
! 550: int i;
! 551: for(i = 0; i < DIVERSITY_HOPS; i++) {
! 552: if(route->channels[i] == 0)
! 553: break;
! 554: if(channels_interfere(babel_ifp->channel, route->channels[i]))
! 555: return 1;
! 556: }
! 557: }
! 558: return 0;
! 559: default:
! 560: fprintf(stderr, "Unknown kind of diversity.\n");
! 561: return 1;
! 562: }
! 563: }
! 564:
! 565: int
! 566: update_feasible(struct source *src,
! 567: unsigned short seqno, unsigned short refmetric)
! 568: {
! 569: if(src == NULL)
! 570: return 1;
! 571:
! 572: if(src->time < babel_now.tv_sec - SOURCE_GC_TIME)
! 573: /* Never mind what is probably stale data */
! 574: return 1;
! 575:
! 576: if(refmetric >= INFINITY)
! 577: /* Retractions are always feasible */
! 578: return 1;
! 579:
! 580: return (seqno_compare(seqno, src->seqno) > 0 ||
! 581: (src->seqno == seqno && refmetric < src->metric));
! 582: }
! 583:
! 584: /* This returns the feasible route with the smallest metric. */
! 585: struct babel_route *
! 586: find_best_route(const unsigned char *prefix, unsigned char plen, int feasible,
! 587: struct neighbour *exclude)
! 588: {
! 589: struct babel_route *route = NULL, *r = NULL;
! 590: int i = find_route_slot(prefix, plen, NULL);
! 591:
! 592: if(i < 0)
! 593: return NULL;
! 594:
! 595: route = routes[i];
! 596:
! 597: r = route->next;
! 598: while(r) {
! 599: if(!route_expired(r) &&
! 600: (!feasible || route_feasible(r)) &&
! 601: (!exclude || r->neigh != exclude) &&
! 602: (route_metric(r) < route_metric(route)))
! 603: route = r;
! 604: r = r->next;
! 605: }
! 606: return route;
! 607: }
! 608:
! 609: void
! 610: update_route_metric(struct babel_route *route)
! 611: {
! 612: int oldmetric = route_metric(route);
! 613:
! 614: if(route_expired(route)) {
! 615: if(route->refmetric < INFINITY) {
! 616: route->seqno = seqno_plus(route->src->seqno, 1);
! 617: retract_route(route);
! 618: if(oldmetric < INFINITY)
! 619: route_changed(route, route->src, oldmetric);
! 620: }
! 621: } else {
! 622: struct neighbour *neigh = route->neigh;
! 623: int add_metric = input_filter(route->src->id,
! 624: route->src->prefix, route->src->plen,
! 625: neigh->address,
! 626: neigh->ifp->ifindex);
! 627: change_route_metric(route, route->refmetric,
! 628: neighbour_cost(route->neigh), add_metric);
! 629: if(route_metric(route) != oldmetric)
! 630: route_changed(route, route->src, oldmetric);
! 631: }
! 632: }
! 633:
! 634: /* Called whenever a neighbour's cost changes, to update the metric of
! 635: all routes through that neighbour. Calls local_notify_neighbour. */
! 636: void
! 637: update_neighbour_metric(struct neighbour *neigh, int changed)
! 638: {
! 639:
! 640: if(changed) {
! 641: int i;
! 642:
! 643: for(i = 0; i < route_slots; i++) {
! 644: struct babel_route *r = routes[i];
! 645: while(r) {
! 646: if(r->neigh == neigh)
! 647: update_route_metric(r);
! 648: r = r->next;
! 649: }
! 650: }
! 651: }
! 652: }
! 653:
! 654: void
! 655: update_interface_metric(struct interface *ifp)
! 656: {
! 657: int i;
! 658:
! 659: for(i = 0; i < route_slots; i++) {
! 660: struct babel_route *r = routes[i];
! 661: while(r) {
! 662: if(r->neigh->ifp == ifp)
! 663: update_route_metric(r);
! 664: r = r->next;
! 665: }
! 666: }
! 667: }
! 668:
! 669: /* This is called whenever we receive an update. */
! 670: struct babel_route *
! 671: update_route(const unsigned char *router_id,
! 672: const unsigned char *prefix, unsigned char plen,
! 673: unsigned short seqno, unsigned short refmetric,
! 674: unsigned short interval,
! 675: struct neighbour *neigh, const unsigned char *nexthop,
! 676: const unsigned char *channels, int channels_len)
! 677: {
! 678: struct babel_route *route;
! 679: struct source *src;
! 680: int metric, feasible;
! 681: int add_metric;
! 682: int hold_time = MAX((4 * interval) / 100 + interval / 50, 15);
! 683:
! 684: if(memcmp(router_id, myid, 8) == 0)
! 685: return NULL;
! 686:
! 687: if(martian_prefix(prefix, plen)) {
! 688: zlog_err("Rejecting martian route to %s through %s.",
! 689: format_prefix(prefix, plen), format_address(router_id));
! 690: return NULL;
! 691: }
! 692:
! 693: add_metric = input_filter(router_id, prefix, plen,
! 694: neigh->address, neigh->ifp->ifindex);
! 695: if(add_metric >= INFINITY)
! 696: return NULL;
! 697:
! 698: route = find_route(prefix, plen, neigh, nexthop);
! 699:
! 700: if(route && memcmp(route->src->id, router_id, 8) == 0)
! 701: /* Avoid scanning the source table. */
! 702: src = route->src;
! 703: else
! 704: src = find_source(router_id, prefix, plen, 1, seqno);
! 705:
! 706: if(src == NULL)
! 707: return NULL;
! 708:
! 709: feasible = update_feasible(src, seqno, refmetric);
! 710: metric = MIN((int)refmetric + neighbour_cost(neigh) + add_metric, INFINITY);
! 711:
! 712: if(route) {
! 713: struct source *oldsrc;
! 714: unsigned short oldmetric;
! 715: int lost = 0;
! 716:
! 717: oldsrc = route->src;
! 718: oldmetric = route_metric(route);
! 719:
! 720: /* If a successor switches sources, we must accept his update even
! 721: if it makes a route unfeasible in order to break any routing loops
! 722: in a timely manner. If the source remains the same, we ignore
! 723: the update. */
! 724: if(!feasible && route->installed) {
! 725: debugf(BABEL_DEBUG_COMMON,"Unfeasible update for installed route to %s "
! 726: "(%s %d %d -> %s %d %d).",
! 727: format_prefix(src->prefix, src->plen),
! 728: format_address(route->src->id),
! 729: route->seqno, route->refmetric,
! 730: format_address(src->id), seqno, refmetric);
! 731: if(src != route->src) {
! 732: uninstall_route(route);
! 733: lost = 1;
! 734: }
! 735: }
! 736:
! 737: route->src = retain_source(src);
! 738: if((feasible || keep_unfeasible) && refmetric < INFINITY)
! 739: route->time = babel_now.tv_sec;
! 740: route->seqno = seqno;
! 741: change_route_metric(route,
! 742: refmetric, neighbour_cost(neigh), add_metric);
! 743: route->hold_time = hold_time;
! 744:
! 745: route_changed(route, oldsrc, oldmetric);
! 746: if(lost)
! 747: route_lost(oldsrc, oldmetric);
! 748:
! 749: if(!feasible)
! 750: send_unfeasible_request(neigh, route->installed && route_old(route),
! 751: seqno, metric, src);
! 752: release_source(oldsrc);
! 753: } else {
! 754: struct babel_route *new_route;
! 755:
! 756: if(refmetric >= INFINITY)
! 757: /* Somebody's retracting a route we never saw. */
! 758: return NULL;
! 759: if(!feasible) {
! 760: send_unfeasible_request(neigh, 0, seqno, metric, src);
! 761: if(!keep_unfeasible)
! 762: return NULL;
! 763: }
! 764:
! 765: route = malloc(sizeof(struct babel_route));
! 766: if(route == NULL) {
! 767: perror("malloc(route)");
! 768: return NULL;
! 769: }
! 770:
! 771: route->src = retain_source(src);
! 772: route->refmetric = refmetric;
! 773: route->cost = neighbour_cost(neigh);
! 774: route->add_metric = add_metric;
! 775: route->seqno = seqno;
! 776: route->neigh = neigh;
! 777: memcpy(route->nexthop, nexthop, 16);
! 778: route->time = babel_now.tv_sec;
! 779: route->hold_time = hold_time;
! 780: route->installed = 0;
! 781: memset(&route->channels, 0, sizeof(route->channels));
! 782: if(channels_len > 0)
! 783: memcpy(&route->channels, channels,
! 784: MIN(channels_len, DIVERSITY_HOPS));
! 785: route->next = NULL;
! 786: new_route = insert_route(route);
! 787: if(new_route == NULL) {
! 788: fprintf(stderr, "Couldn't insert route.\n");
! 789: free(route);
! 790: return NULL;
! 791: }
! 792: consider_route(route);
! 793: }
! 794: return route;
! 795: }
! 796:
! 797: /* We just received an unfeasible update. If it's any good, send
! 798: a request for a new seqno. */
! 799: void
! 800: send_unfeasible_request(struct neighbour *neigh, int force,
! 801: unsigned short seqno, unsigned short metric,
! 802: struct source *src)
! 803: {
! 804: struct babel_route *route = find_installed_route(src->prefix, src->plen);
! 805:
! 806: if(seqno_minus(src->seqno, seqno) > 100) {
! 807: /* Probably a source that lost its seqno. Let it time-out. */
! 808: return;
! 809: }
! 810:
! 811: if(force || !route || route_metric(route) >= metric + 512) {
! 812: send_unicast_multihop_request(neigh, src->prefix, src->plen,
! 813: src->metric >= INFINITY ?
! 814: src->seqno :
! 815: seqno_plus(src->seqno, 1),
! 816: src->id, 127);
! 817: }
! 818: }
! 819:
! 820: /* This takes a feasible route and decides whether to install it. */
! 821: static void
! 822: consider_route(struct babel_route *route)
! 823: {
! 824: struct babel_route *installed;
! 825: struct xroute *xroute;
! 826:
! 827: if(route->installed)
! 828: return;
! 829:
! 830: if(!route_feasible(route))
! 831: return;
! 832:
! 833: xroute = find_xroute(route->src->prefix, route->src->plen);
! 834: if(xroute && (allow_duplicates < 0 || xroute->metric >= allow_duplicates))
! 835: return;
! 836:
! 837: installed = find_installed_route(route->src->prefix, route->src->plen);
! 838:
! 839: if(installed == NULL)
! 840: goto install;
! 841:
! 842: if(route_metric(route) >= INFINITY)
! 843: return;
! 844:
! 845: if(route_metric(installed) >= INFINITY)
! 846: goto install;
! 847:
! 848: if(route_metric(installed) >= route_metric(route) + 64)
! 849: goto install;
! 850:
! 851: return;
! 852:
! 853: install:
! 854: switch_routes(installed, route);
! 855: if(installed && route->installed)
! 856: send_triggered_update(route, installed->src, route_metric(installed));
! 857: else
! 858: send_update(NULL, 1, route->src->prefix, route->src->plen);
! 859: return;
! 860: }
! 861:
! 862: void
! 863: retract_neighbour_routes(struct neighbour *neigh)
! 864: {
! 865: int i;
! 866:
! 867: for(i = 0; i < route_slots; i++) {
! 868: struct babel_route *r = routes[i];
! 869: while(r) {
! 870: if(r->neigh == neigh) {
! 871: if(r->refmetric != INFINITY) {
! 872: unsigned short oldmetric = route_metric(r);
! 873: retract_route(r);
! 874: if(oldmetric != INFINITY)
! 875: route_changed(r, r->src, oldmetric);
! 876: }
! 877: }
! 878: r = r->next;
! 879: }
! 880: i++;
! 881: }
! 882: }
! 883:
! 884: void
! 885: send_triggered_update(struct babel_route *route, struct source *oldsrc,
! 886: unsigned oldmetric)
! 887: {
! 888: unsigned newmetric, diff;
! 889: /* 1 means send speedily, 2 means resend */
! 890: int urgent;
! 891:
! 892: if(!route->installed)
! 893: return;
! 894:
! 895: newmetric = route_metric(route);
! 896: diff =
! 897: newmetric >= oldmetric ? newmetric - oldmetric : oldmetric - newmetric;
! 898:
! 899: if(route->src != oldsrc || (oldmetric < INFINITY && newmetric >= INFINITY))
! 900: /* Switching sources can cause transient routing loops.
! 901: Retractions can cause blackholes. */
! 902: urgent = 2;
! 903: else if(newmetric > oldmetric && oldmetric < 6 * 256 && diff >= 512)
! 904: /* Route getting significantly worse */
! 905: urgent = 1;
! 906: else if(unsatisfied_request(route->src->prefix, route->src->plen,
! 907: route->seqno, route->src->id))
! 908: /* Make sure that requests are satisfied speedily */
! 909: urgent = 1;
! 910: else if(oldmetric >= INFINITY && newmetric < INFINITY)
! 911: /* New route */
! 912: urgent = 0;
! 913: else if(newmetric < oldmetric && diff < 1024)
! 914: /* Route getting better. This may be a transient fluctuation, so
! 915: don't advertise it to avoid making routes unfeasible later on. */
! 916: return;
! 917: else if(diff < 384)
! 918: /* Don't fret about trivialities */
! 919: return;
! 920: else
! 921: urgent = 0;
! 922:
! 923: if(urgent >= 2)
! 924: send_update_resend(NULL, route->src->prefix, route->src->plen);
! 925: else
! 926: send_update(NULL, urgent, route->src->prefix, route->src->plen);
! 927:
! 928: if(oldmetric < INFINITY) {
! 929: if(newmetric >= oldmetric + 512) {
! 930: send_request_resend(NULL, route->src->prefix, route->src->plen,
! 931: route->src->metric >= INFINITY ?
! 932: route->src->seqno :
! 933: seqno_plus(route->src->seqno, 1),
! 934: route->src->id);
! 935: } else if(newmetric >= oldmetric + 288) {
! 936: send_request(NULL, route->src->prefix, route->src->plen);
! 937: }
! 938: }
! 939: }
! 940:
! 941: /* A route has just changed. Decide whether to switch to a different route or
! 942: send an update. */
! 943: void
! 944: route_changed(struct babel_route *route,
! 945: struct source *oldsrc, unsigned short oldmetric)
! 946: {
! 947: if(route->installed) {
! 948: if(route_metric(route) > oldmetric) {
! 949: struct babel_route *better_route;
! 950: better_route =
! 951: find_best_route(route->src->prefix, route->src->plen, 1, NULL);
! 952: if(better_route &&
! 953: route_metric(better_route) <= route_metric(route) - 96)
! 954: consider_route(better_route);
! 955: }
! 956:
! 957: if(route->installed)
! 958: /* We didn't change routes after all. */
! 959: send_triggered_update(route, oldsrc, oldmetric);
! 960: } else {
! 961: /* Reconsider routes even when their metric didn't decrease,
! 962: they may not have been feasible before. */
! 963: consider_route(route);
! 964: }
! 965: }
! 966:
! 967: /* We just lost the installed route to a given destination. */
! 968: void
! 969: route_lost(struct source *src, unsigned oldmetric)
! 970: {
! 971: struct babel_route *new_route;
! 972: new_route = find_best_route(src->prefix, src->plen, 1, NULL);
! 973: if(new_route) {
! 974: consider_route(new_route);
! 975: } else if(oldmetric < INFINITY) {
! 976: /* Complain loudly. */
! 977: send_update_resend(NULL, src->prefix, src->plen);
! 978: send_request_resend(NULL, src->prefix, src->plen,
! 979: src->metric >= INFINITY ?
! 980: src->seqno : seqno_plus(src->seqno, 1),
! 981: src->id);
! 982: }
! 983: }
! 984:
! 985: /* This is called periodically to flush old routes. It will also send
! 986: requests for routes that are about to expire. */
! 987: void
! 988: expire_routes(void)
! 989: {
! 990: struct babel_route *r;
! 991: int i;
! 992:
! 993: debugf(BABEL_DEBUG_COMMON,"Expiring old routes.");
! 994:
! 995: i = 0;
! 996: while(i < route_slots) {
! 997: r = routes[i];
! 998: while(r) {
! 999: /* Protect against clock being stepped. */
! 1000: if(r->time > babel_now.tv_sec || route_old(r)) {
! 1001: flush_route(r);
! 1002: goto again;
! 1003: }
! 1004:
! 1005: update_route_metric(r);
! 1006:
! 1007: if(r->installed && r->refmetric < INFINITY) {
! 1008: if(route_old(r))
! 1009: /* Route about to expire, send a request. */
! 1010: send_unicast_request(r->neigh,
! 1011: r->src->prefix, r->src->plen);
! 1012: }
! 1013: r = r->next;
! 1014: }
! 1015: i++;
! 1016: again:
! 1017: ;
! 1018: }
! 1019: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>