/* * Copyright (C) 2015 Martin Landsmann * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup sys_shell_commands * @{ * * @file * @brief Provides shell commands to manage and show FIB Entries * * @author 2015 Martin Landsmann * * @} */ #include #include #include #include #include #include "net/af.h" #include "net/fib.h" #include "net/gnrc/ipv6.h" #include "net/gnrc/netif.h" #include "shell.h" #include "thread.h" #define INFO1_TXT "fibroute add via [dev ]" #define INFO2_TXT " [lifetime ]" #define INFO3_TXT " - the destination address with optional prefix size, e.g. /116\n" \ " - the address of the next-hop towards the \n" \ " - the device id of the Interface to use." \ " Optional if only one interface is available.\n" #define INFO4_TXT " - optional lifetime in ms when the entry automatically invalidates\n" #define INFO5_TXT "fibroute del \n" \ " - the destination address of the entry to be deleted\n" static unsigned char tmp_ipv4_dst[INADDRSZ]; /**< buffer for ipv4 address conversion */ static unsigned char tmp_ipv4_nxt[INADDRSZ]; /**< buffer for ipv4 address conversion */ static unsigned char tmp_ipv6_dst[IN6ADDRSZ]; /**< buffer for ipv6 address conversion */ static unsigned char tmp_ipv6_nxt[IN6ADDRSZ]; /**< buffer for ipv6 address conversion */ static void _fib_usage(int info) { switch (info) { case 0: { puts("\nsee for more information\n" " removes all entries [associated with interface]\n"); break; } case 1: { puts("\nbrief: adds a new entry to the FIB.\nusage: " INFO1_TXT "\n" INFO3_TXT); break; } case 2: { puts("\nbrief: adds a new entry to the FIB.\nusage: " INFO1_TXT INFO2_TXT "\n" INFO3_TXT INFO4_TXT); break; } case 3: { puts("\nbrief: deletes an entry from the FIB.\nusage: " INFO5_TXT); break; } default: break; }; } static void _fib_add(const char *dest, const char *next, kernel_pid_t pid, uint32_t lifetime) { uint32_t prefix = 0; /* Get the prefix length */ size_t i = 0; for (i = strlen(dest); i > 0; --i) { if (dest[i] == '/') { prefix = atoi(&dest[i+1]); break; } if (dest[i] == ':' || dest[i] == '.') { i = strlen(dest); break; } } size_t dst_size = (i+1); unsigned char dst_arr[dst_size]; memset(dst_arr, 0, dst_size); memcpy(dst_arr, dest, i); unsigned char *dst = &dst_arr[0]; uint32_t dst_flags = 0; unsigned char *nxt = (unsigned char *)next; size_t nxt_size = (strlen(next)); uint32_t nxt_flags = 0; /* determine destination address */ if (inet_pton(AF_INET6, (char*)dst, tmp_ipv6_dst)) { dst = tmp_ipv6_dst; dst_size = IN6ADDRSZ; } else if (inet_pton(AF_INET, dest, tmp_ipv4_dst)) { dst = tmp_ipv4_dst; dst_size = INADDRSZ; } /* determine next-hop address */ if (inet_pton(AF_INET6, next, tmp_ipv6_nxt)) { nxt = tmp_ipv6_nxt; nxt_size = IN6ADDRSZ; } else if (inet_pton(AF_INET, next, tmp_ipv4_nxt)) { nxt = tmp_ipv4_nxt; nxt_size = INADDRSZ; } dst_flags |= (prefix << FIB_FLAG_NET_PREFIX_SHIFT); fib_add_entry(&gnrc_ipv6_fib_table, pid, dst, dst_size, dst_flags, nxt, nxt_size, nxt_flags, lifetime); } static int _fib_route_handler(int argc, char **argv) { /* e.g. fibroute right now don't care about the address/protocol family */ if (argc == 1) { fib_print_routes(&gnrc_ipv6_fib_table); return 0; } /* e.g. firoute [add|del] */ if (argc == 2) { if ((strcmp("add", argv[1]) == 0)) { _fib_usage(2); } else if ((strcmp("del", argv[1]) == 0)) { _fib_usage(3); } else if ((strcmp("flush", argv[1]) == 0)) { fib_flush(&gnrc_ipv6_fib_table, KERNEL_PID_UNDEF); puts("successfully flushed all entries"); } else { _fib_usage(0); } return 1; } if (argc > 2 && !((strcmp("add", argv[1]) == 0) || (strcmp("del", argv[1]) == 0) || (strcmp("flush", argv[1]) == 0))) { puts("\nunrecognized parameter2.\nPlease enter fibroute [add|del] for more information."); return 1; } /* e.g. fibroute del */ if (argc == 3) { if ((strcmp("flush", argv[1]) == 0)) { kernel_pid_t iface = atoi(argv[2]); if (gnrc_netif_get_by_pid(iface) != NULL) { fib_flush(&gnrc_ipv6_fib_table, iface); printf("successfully flushed all entries for interface %" PRIu16"\n", iface); } else { printf("interface %" PRIu16" does not exist\n", iface); } } else if (inet_pton(AF_INET6, argv[2], tmp_ipv6_dst)) { fib_remove_entry(&gnrc_ipv6_fib_table, tmp_ipv6_dst, IN6ADDRSZ); } else if (inet_pton(AF_INET, argv[2], tmp_ipv4_dst)) { fib_remove_entry(&gnrc_ipv6_fib_table, tmp_ipv4_dst, INADDRSZ); } else { fib_remove_entry(&gnrc_ipv6_fib_table, (uint8_t *)argv[2], (strlen(argv[2]))); } return 0; } /* e.g. fibroute add via */ if ((argc == 5) && (strcmp("add", argv[1]) == 0) && (strcmp("via", argv[3]) == 0)) { size_t ifnum = gnrc_netif_numof(); if (ifnum == 1) { gnrc_netif_t *netif = gnrc_netif_iter(NULL); _fib_add(argv[2], argv[4], netif->pid, (uint32_t)FIB_LIFETIME_NO_EXPIRE); } else { _fib_usage(1); return 1; } return 0; } /* e.g. fibroute add via lifetime */ if ((argc == 7) && (strcmp("add", argv[1]) == 0) && (strcmp("via", argv[3]) == 0) && (strcmp("lifetime", argv[5]) == 0)) { size_t ifnum = gnrc_netif_numof(); if (ifnum == 1) { gnrc_netif_t *netif = gnrc_netif_iter(NULL); _fib_add(argv[2], argv[4], netif->pid, (uint32_t)atoi(argv[6])); } else { _fib_usage(1); return 1; } return 0; } /* e.g. fibroute add via dev */ if (argc == 7) { if ((strcmp("add", argv[1]) == 0) && (strcmp("via", argv[3]) == 0) && (strcmp("dev", argv[5]) == 0)) { _fib_add(argv[2], argv[4], (kernel_pid_t)atoi(argv[6]), (uint32_t)FIB_LIFETIME_NO_EXPIRE); } else { _fib_usage(1); return 1; } return 0; } /* e.g. fibroute add via dev lifetime */ if (argc == 9) { if ((strcmp("add", argv[1]) == 0) && (strcmp("via", argv[3]) == 0) && (strcmp("dev", argv[5]) == 0) && (strcmp("lifetime", argv[7]) == 0)) { _fib_add(argv[2], argv[4], (kernel_pid_t )atoi(argv[6]), (uint32_t)atoi(argv[8])); } else { _fib_usage(2); return 1; } return 0; } puts("\nunrecognized parameters.\nPlease enter fibroute [add|del] for more information."); return 1; } SHELL_COMMAND(fibroute, "Manipulate the FIB (info: 'fibroute [add|del]')", _fib_route_handler);