Annotation of embedaddon/quagga/babeld/babel_zebra.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 2011 by Matthieu Boutier and Juliusz Chroboczek
! 19:
! 20: Permission is hereby granted, free of charge, to any person obtaining a copy
! 21: of this software and associated documentation files (the "Software"), to deal
! 22: in the Software without restriction, including without limitation the rights
! 23: to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
! 24: copies of the Software, and to permit persons to whom the Software is
! 25: furnished to do so, subject to the following conditions:
! 26:
! 27: The above copyright notice and this permission notice shall be included in
! 28: all copies or substantial portions of the Software.
! 29:
! 30: THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
! 31: IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
! 32: FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
! 33: AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
! 34: LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
! 35: OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
! 36: THE SOFTWARE.
! 37: */
! 38:
! 39: /* quagga's includes */
! 40: #include <zebra.h>
! 41: #include "command.h"
! 42: #include "zclient.h"
! 43: #include "stream.h"
! 44:
! 45: /* babel's includes*/
! 46: #include "babel_zebra.h"
! 47: #include "babel_interface.h"
! 48: #include "xroute.h"
! 49: #include "util.h"
! 50:
! 51: void babelz_zebra_init(void);
! 52:
! 53:
! 54: /* we must use a pointer because of zclient.c's functions (new, free). */
! 55: struct zclient *zclient;
! 56: static int zebra_config_write (struct vty *vty);
! 57:
! 58: /* Debug types */
! 59: static struct {
! 60: int type;
! 61: int str_min_len;
! 62: const char *str;
! 63: } debug_type[] = {
! 64: {BABEL_DEBUG_COMMON, 1, "common"},
! 65: {BABEL_DEBUG_KERNEL, 1, "kernel"},
! 66: {BABEL_DEBUG_FILTER, 1, "filter"},
! 67: {BABEL_DEBUG_TIMEOUT, 1, "timeout"},
! 68: {BABEL_DEBUG_IF, 1, "interface"},
! 69: {BABEL_DEBUG_ROUTE, 1, "route"},
! 70: {BABEL_DEBUG_ALL, 1, "all"},
! 71: {0, 0, NULL}
! 72: };
! 73:
! 74: /* Zebra node structure. */
! 75: struct cmd_node zebra_node =
! 76: {
! 77: ZEBRA_NODE,
! 78: "%s(config-router)# ",
! 79: 1 /* vtysh? yes */
! 80: };
! 81:
! 82:
! 83: /* Zebra route add and delete treatment (ipv6). */
! 84: static int
! 85: babel_zebra_read_ipv6 (int command, struct zclient *zclient,
! 86: zebra_size_t length)
! 87: {
! 88: struct stream *s;
! 89: struct zapi_ipv6 api;
! 90: unsigned long ifindex = -1;
! 91: struct in6_addr nexthop;
! 92: struct prefix_ipv6 prefix;
! 93:
! 94: s = zclient->ibuf;
! 95: ifindex = 0;
! 96: memset (&nexthop, 0, sizeof (struct in6_addr));
! 97: memset (&api, 0, sizeof(struct zapi_ipv6));
! 98: memset (&prefix, 0, sizeof (struct prefix_ipv6));
! 99:
! 100: /* Type, flags, message. */
! 101: api.type = stream_getc (s);
! 102: api.flags = stream_getc (s);
! 103: api.message = stream_getc (s);
! 104:
! 105: /* IPv6 prefix. */
! 106: prefix.family = AF_INET6;
! 107: prefix.prefixlen = stream_getc (s);
! 108: stream_get (&prefix.prefix, s, PSIZE (prefix.prefixlen));
! 109:
! 110: /* Nexthop, ifindex, distance, metric. */
! 111: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP)) {
! 112: api.nexthop_num = stream_getc (s);
! 113: stream_get (&nexthop, s, sizeof(nexthop));
! 114: }
! 115: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX)) {
! 116: api.ifindex_num = stream_getc (s);
! 117: ifindex = stream_getl (s);
! 118: }
! 119: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
! 120: api.distance = stream_getc (s);
! 121: else
! 122: api.distance = 0;
! 123: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
! 124: api.metric = stream_getl (s);
! 125: else
! 126: api.metric = 0;
! 127:
! 128: if (command == ZEBRA_IPV6_ROUTE_ADD)
! 129: babel_ipv6_route_add(&api, &prefix, ifindex, &nexthop);
! 130: else
! 131: babel_ipv6_route_delete(&api, &prefix, ifindex);
! 132:
! 133: return 0;
! 134: }
! 135:
! 136: static int
! 137: babel_zebra_read_ipv4 (int command, struct zclient *zclient,
! 138: zebra_size_t length)
! 139: {
! 140: struct stream *s;
! 141: struct zapi_ipv4 api;
! 142: unsigned long ifindex = -1;
! 143: struct in_addr nexthop;
! 144: struct prefix_ipv4 prefix;
! 145:
! 146: s = zclient->ibuf;
! 147: ifindex = 0;
! 148: memset (&nexthop, 0, sizeof (struct in_addr));
! 149: memset (&api, 0, sizeof(struct zapi_ipv4));
! 150: memset (&prefix, 0, sizeof (struct prefix_ipv4));
! 151:
! 152: /* Type, flags, message. */
! 153: api.type = stream_getc (s);
! 154: api.flags = stream_getc (s);
! 155: api.message = stream_getc (s);
! 156:
! 157: /* IPv6 prefix. */
! 158: prefix.family = AF_INET;
! 159: prefix.prefixlen = stream_getc (s);
! 160: stream_get (&prefix.prefix, s, PSIZE (prefix.prefixlen));
! 161:
! 162: /* Nexthop, ifindex, distance, metric. */
! 163: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP)) {
! 164: api.nexthop_num = stream_getc (s);
! 165: stream_get (&nexthop, s, sizeof(nexthop));
! 166: }
! 167: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX)) {
! 168: api.ifindex_num = stream_getc (s);
! 169: ifindex = stream_getl (s);
! 170: }
! 171: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
! 172: api.distance = stream_getc (s);
! 173: else
! 174: api.distance = 0;
! 175: if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
! 176: api.metric = stream_getl (s);
! 177: else
! 178: api.metric = 0;
! 179:
! 180: if (command == ZEBRA_IPV6_ROUTE_ADD) {
! 181: babel_ipv4_route_add(&api, &prefix, ifindex, &nexthop);
! 182: } else {
! 183: babel_ipv4_route_delete(&api, &prefix, ifindex);
! 184: }
! 185:
! 186: return 0;
! 187: }
! 188:
! 189: /* [Babel Command] */
! 190: DEFUN (babel_redistribute_type,
! 191: babel_redistribute_type_cmd,
! 192: "redistribute " QUAGGA_REDIST_STR_BABELD,
! 193: "Redistribute\n"
! 194: QUAGGA_REDIST_HELP_STR_BABELD)
! 195: {
! 196: int type;
! 197:
! 198: type = proto_redistnum(AFI_IP6, argv[0]);
! 199:
! 200: if (type < 0)
! 201: type = proto_redistnum(AFI_IP, argv[0]);
! 202:
! 203: if (type < 0) {
! 204: vty_out(vty, "Invalid type %s%s", argv[0], VTY_NEWLINE);
! 205: return CMD_WARNING;
! 206: }
! 207:
! 208: zclient_redistribute (ZEBRA_REDISTRIBUTE_ADD, zclient, type);
! 209: return CMD_SUCCESS;
! 210: }
! 211:
! 212: /* [Babel Command] */
! 213: DEFUN (no_babel_redistribute_type,
! 214: no_babel_redistribute_type_cmd,
! 215: "no redistribute " QUAGGA_REDIST_STR_BABELD,
! 216: NO_STR
! 217: "Redistribute\n"
! 218: QUAGGA_REDIST_HELP_STR_BABELD)
! 219: {
! 220: int type;
! 221:
! 222: type = proto_redistnum(AFI_IP6, argv[0]);
! 223:
! 224: if (type < 0)
! 225: type = proto_redistnum(AFI_IP, argv[0]);
! 226:
! 227: if (type < 0) {
! 228: vty_out(vty, "Invalid type %s%s", argv[0], VTY_NEWLINE);
! 229: return CMD_WARNING;
! 230: }
! 231:
! 232: zclient_redistribute (ZEBRA_REDISTRIBUTE_DELETE, zclient, type);
! 233: /* perhaps should we remove xroutes having the same type... */
! 234: return CMD_SUCCESS;
! 235: }
! 236:
! 237: #ifndef NO_DEBUG
! 238: /* [Babel Command] */
! 239: DEFUN (debug_babel,
! 240: debug_babel_cmd,
! 241: "debug babel (common|kernel|filter|timeout|interface|route|all)",
! 242: "Enable debug messages for specific or all part.\n"
! 243: "Babel information\n"
! 244: "Common messages (default)\n"
! 245: "Kernel messages\n"
! 246: "Filter messages\n"
! 247: "Timeout messages\n"
! 248: "Interface messages\n"
! 249: "Route messages\n"
! 250: "All messages\n")
! 251: {
! 252: int i;
! 253:
! 254: for(i = 0; debug_type[i].str != NULL; i++) {
! 255: if (strncmp (debug_type[i].str, argv[0],
! 256: debug_type[i].str_min_len) == 0) {
! 257: debug |= debug_type[i].type;
! 258: return CMD_SUCCESS;
! 259: }
! 260: }
! 261:
! 262: vty_out(vty, "Invalid type %s%s", argv[0], VTY_NEWLINE);
! 263:
! 264: return CMD_WARNING;
! 265: }
! 266:
! 267: /* [Babel Command] */
! 268: DEFUN (no_debug_babel,
! 269: no_debug_babel_cmd,
! 270: "no debug babel (common|kernel|filter|timeout|interface|route|all)",
! 271: NO_STR
! 272: "Disable debug messages for specific or all part.\n"
! 273: "Babel information\n"
! 274: "Common messages (default)\n"
! 275: "Kernel messages\n"
! 276: "Filter messages\n"
! 277: "Timeout messages\n"
! 278: "Interface messages\n"
! 279: "Route messages\n"
! 280: "All messages\n")
! 281: {
! 282: int i;
! 283:
! 284: for (i = 0; debug_type[i].str; i++) {
! 285: if (strncmp(debug_type[i].str, argv[0],
! 286: debug_type[i].str_min_len) == 0) {
! 287: debug &= ~debug_type[i].type;
! 288: return CMD_SUCCESS;
! 289: }
! 290: }
! 291:
! 292: vty_out(vty, "Invalid type %s%s", argv[0], VTY_NEWLINE);
! 293:
! 294: return CMD_WARNING;
! 295: }
! 296: #endif /* NO_DEBUG */
! 297:
! 298: /* Output "debug" statement lines, if necessary. */
! 299: int
! 300: debug_babel_config_write (struct vty * vty)
! 301: {
! 302: #ifdef NO_DEBUG
! 303: return 0;
! 304: #else
! 305: int i, lines = 0;
! 306:
! 307: if (debug == BABEL_DEBUG_ALL)
! 308: {
! 309: vty_out (vty, "debug babel all%s", VTY_NEWLINE);
! 310: lines++;
! 311: }
! 312: else
! 313: for (i = 0; debug_type[i].str != NULL; i++)
! 314: if
! 315: (
! 316: debug_type[i].type != BABEL_DEBUG_ALL
! 317: && CHECK_FLAG (debug, debug_type[i].type)
! 318: )
! 319: {
! 320: vty_out (vty, "debug babel %s%s", debug_type[i].str, VTY_NEWLINE);
! 321: lines++;
! 322: }
! 323: if (lines)
! 324: {
! 325: vty_out (vty, "!%s", VTY_NEWLINE);
! 326: lines++;
! 327: }
! 328: return lines;
! 329: #endif /* NO_DEBUG */
! 330: }
! 331:
! 332: void babelz_zebra_init(void)
! 333: {
! 334: zclient = zclient_new();
! 335: zclient_init(zclient, ZEBRA_ROUTE_BABEL);
! 336:
! 337: zclient->interface_add = babel_interface_add;
! 338: zclient->interface_delete = babel_interface_delete;
! 339: zclient->interface_up = babel_interface_up;
! 340: zclient->interface_down = babel_interface_down;
! 341: zclient->interface_address_add = babel_interface_address_add;
! 342: zclient->interface_address_delete = babel_interface_address_delete;
! 343: zclient->ipv4_route_add = babel_zebra_read_ipv4;
! 344: zclient->ipv4_route_delete = babel_zebra_read_ipv4;
! 345: zclient->ipv6_route_add = babel_zebra_read_ipv6;
! 346: zclient->ipv6_route_delete = babel_zebra_read_ipv6;
! 347:
! 348: install_node (&zebra_node, zebra_config_write);
! 349: install_element(BABEL_NODE, &babel_redistribute_type_cmd);
! 350: install_element(BABEL_NODE, &no_babel_redistribute_type_cmd);
! 351: install_element(ENABLE_NODE, &debug_babel_cmd);
! 352: install_element(ENABLE_NODE, &no_debug_babel_cmd);
! 353: install_element(CONFIG_NODE, &debug_babel_cmd);
! 354: install_element(CONFIG_NODE, &no_debug_babel_cmd);
! 355: }
! 356:
! 357: static int
! 358: zebra_config_write (struct vty *vty)
! 359: {
! 360: if (! zclient->enable)
! 361: {
! 362: vty_out (vty, "no router zebra%s", VTY_NEWLINE);
! 363: return 1;
! 364: }
! 365: else if (! zclient->redist[ZEBRA_ROUTE_BABEL])
! 366: {
! 367: vty_out (vty, "router zebra%s", VTY_NEWLINE);
! 368: vty_out (vty, " no redistribute babel%s", VTY_NEWLINE);
! 369: return 1;
! 370: }
! 371: return 0;
! 372: }
! 373:
! 374: void
! 375: babel_zebra_close_connexion(void)
! 376: {
! 377: zclient_stop(zclient);
! 378: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>