Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 1 | /* |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 2 | * Copyright (c) 2016 Cisco and/or its affiliates. |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 3 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | * you may not use this file except in compliance with the License. |
| 5 | * You may obtain a copy of the License at: |
| 6 | * |
| 7 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | * |
| 9 | * Unless required by applicable law or agreed to in writing, software |
| 10 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | * See the License for the specific language governing permissions and |
| 13 | * limitations under the License. |
| 14 | */ |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 15 | /** |
| 16 | * @file |
| 17 | * @brief Common utility functions for IPv4, IPv6 and L2 LISP-GPE tunnels. |
| 18 | * |
| 19 | */ |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 20 | |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 21 | #include <vnet/lisp-gpe/lisp_gpe.h> |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 22 | #include <vnet/lisp-gpe/lisp_gpe_fwd_entry.h> |
Neale Ranns | 0bfe5d8 | 2016-08-25 15:29:12 +0100 | [diff] [blame] | 23 | #include <vnet/lisp-gpe/lisp_gpe_adjacency.h> |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 24 | #include <vnet/lisp-gpe/lisp_gpe_tenant.h> |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 25 | #include <vnet/fib/fib_path_list.h> |
| 26 | #include <vnet/fib/fib_table.h> |
| 27 | #include <vnet/fib/fib_internal.h> |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 28 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 29 | /** LISP-GPE global state */ |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 30 | lisp_gpe_main_t lisp_gpe_main; |
| 31 | |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 32 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 33 | /** CLI command to add/del forwarding entry. */ |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 34 | static clib_error_t * |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 35 | lisp_gpe_add_del_fwd_entry_command_fn (vlib_main_t * vm, |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 36 | unformat_input_t * input, |
| 37 | vlib_cli_command_t * cmd) |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 38 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 39 | unformat_input_t _line_input, *line_input = &_line_input; |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 40 | u8 is_add = 1; |
Florin Coras | bb5c22f | 2016-08-02 02:31:03 +0200 | [diff] [blame] | 41 | ip_address_t lloc, rloc; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 42 | clib_error_t *error = 0; |
| 43 | gid_address_t _reid, *reid = &_reid, _leid, *leid = &_leid; |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 44 | u8 reid_set = 0, leid_set = 0, is_negative = 0, dp_table_set = 0, |
| 45 | vni_set = 0; |
| 46 | u32 vni = 0, dp_table = 0, action = ~0, w; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 47 | locator_pair_t pair, *pairs = 0; |
Andrej Kozemcak | 8ebb2a1 | 2016-06-07 12:25:20 +0200 | [diff] [blame] | 48 | int rv; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 49 | |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 50 | memset (leid, 0, sizeof (*leid)); |
| 51 | memset (reid, 0, sizeof (*reid)); |
| 52 | |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 53 | /* Get a line of input. */ |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 54 | if (!unformat_user (input, unformat_line_input, line_input)) |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 55 | return 0; |
| 56 | |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 57 | while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT) |
| 58 | { |
| 59 | if (unformat (line_input, "del")) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 60 | is_add = 0; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 61 | else if (unformat (line_input, "add")) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 62 | is_add = 1; |
| 63 | else if (unformat (line_input, "leid %U", unformat_gid_address, leid)) |
| 64 | { |
| 65 | leid_set = 1; |
| 66 | } |
| 67 | else if (unformat (line_input, "reid %U", unformat_gid_address, reid)) |
| 68 | { |
| 69 | reid_set = 1; |
| 70 | } |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 71 | else if (unformat (line_input, "vni %u", &vni)) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 72 | { |
| 73 | gid_address_vni (leid) = vni; |
| 74 | gid_address_vni (reid) = vni; |
| 75 | vni_set = 1; |
| 76 | } |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 77 | else if (unformat (line_input, "vrf %u", &dp_table)) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 78 | { |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 79 | dp_table_set = 1; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 80 | } |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 81 | else if (unformat (line_input, "bd %u", &dp_table)) |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 82 | { |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 83 | dp_table_set = 1; |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 84 | } |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 85 | else if (unformat (line_input, "negative action %U", |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 86 | unformat_negative_mapping_action, &action)) |
| 87 | { |
| 88 | is_negative = 1; |
| 89 | } |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 90 | else if (unformat (line_input, "loc-pair %U %U w %d", |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 91 | unformat_ip_address, &lloc, |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 92 | unformat_ip_address, &rloc, &w)) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 93 | { |
Florin Coras | af8c8e5 | 2017-08-17 15:57:20 -0700 | [diff] [blame] | 94 | ip_address_copy (&pair.lcl_loc, &lloc); |
| 95 | ip_address_copy (&pair.rmt_loc, &rloc); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 96 | pair.weight = w; |
Florin Coras | 42e480d | 2017-01-16 00:57:02 -0800 | [diff] [blame] | 97 | pair.priority = 0; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 98 | vec_add1 (pairs, pair); |
| 99 | } |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 100 | else |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 101 | { |
| 102 | error = unformat_parse_error (line_input); |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 103 | vlib_cli_output (vm, "parse error: '%U'", |
| 104 | format_unformat_error, line_input); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 105 | goto done; |
| 106 | } |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 107 | } |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 108 | |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 109 | if (!reid_set) |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 110 | { |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 111 | vlib_cli_output (vm, "remote eid must be set!"); |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 112 | goto done; |
| 113 | } |
| 114 | |
Florin Coras | b69111e | 2017-02-13 23:55:27 -0800 | [diff] [blame] | 115 | if (gid_address_type (reid) != GID_ADDR_NSH && (!vni_set || !dp_table_set)) |
| 116 | { |
| 117 | vlib_cli_output (vm, "vni and vrf/bd must be set!"); |
| 118 | goto done; |
| 119 | } |
| 120 | |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 121 | if (is_negative) |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 122 | { |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 123 | if (~0 == action) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 124 | { |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 125 | vlib_cli_output (vm, "no action set for negative tunnel!"); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 126 | goto done; |
| 127 | } |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 128 | } |
| 129 | else |
| 130 | { |
Florin Coras | bb5c22f | 2016-08-02 02:31:03 +0200 | [diff] [blame] | 131 | if (vec_len (pairs) == 0) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 132 | { |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 133 | vlib_cli_output (vm, "expected ip4/ip6 locators"); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 134 | goto done; |
| 135 | } |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 136 | } |
| 137 | |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 138 | if (!leid_set) |
| 139 | { |
| 140 | /* if leid not set, make sure it's the same AFI like reid */ |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 141 | gid_address_type (leid) = gid_address_type (reid); |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 142 | if (GID_ADDR_IP_PREFIX == gid_address_type (reid)) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 143 | gid_address_ip_version (leid) = gid_address_ip_version (reid); |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 144 | } |
| 145 | |
| 146 | /* add fwd entry */ |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 147 | vnet_lisp_gpe_add_del_fwd_entry_args_t _a, *a = &_a; |
| 148 | memset (a, 0, sizeof (a[0])); |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 149 | |
| 150 | a->is_add = is_add; |
Florin Coras | 82bf7cd | 2016-09-26 18:59:44 +0300 | [diff] [blame] | 151 | a->is_negative = is_negative; |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 152 | a->vni = vni; |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 153 | a->table_id = dp_table; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 154 | gid_address_copy (&a->lcl_eid, leid); |
| 155 | gid_address_copy (&a->rmt_eid, reid); |
Florin Coras | bb5c22f | 2016-08-02 02:31:03 +0200 | [diff] [blame] | 156 | a->locator_pairs = pairs; |
Florin Coras | ce1b4c7 | 2017-01-26 14:25:34 -0800 | [diff] [blame] | 157 | a->action = action; |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 158 | |
| 159 | rv = vnet_lisp_gpe_add_del_fwd_entry (a, 0); |
| 160 | if (0 != rv) |
| 161 | { |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 162 | vlib_cli_output (vm, "failed to %s gpe tunnel!", |
| 163 | is_add ? "add" : "delete"); |
Florin Coras | 03c4f99 | 2016-07-19 15:27:58 +0200 | [diff] [blame] | 164 | } |
| 165 | |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 166 | done: |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 167 | unformat_free (line_input); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 168 | vec_free (pairs); |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 169 | return error; |
| 170 | } |
| 171 | |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 172 | /* *INDENT-OFF* */ |
Florin Coras | ed09a05 | 2016-05-06 14:22:40 +0200 | [diff] [blame] | 173 | VLIB_CLI_COMMAND (lisp_gpe_add_del_fwd_entry_command, static) = { |
Filip Tehlar | 82786c4 | 2017-02-20 15:20:37 +0100 | [diff] [blame] | 174 | .path = "gpe entry", |
| 175 | .short_help = "gpe entry add/del vni <vni> vrf/bd <id> [leid <leid>]" |
Filip Tehlar | c3af7bf | 2017-01-13 14:13:09 +0100 | [diff] [blame] | 176 | "reid <reid> [loc-pair <lloc> <rloc> w <weight>] " |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 177 | "[negative action <action>]", |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 178 | .function = lisp_gpe_add_del_fwd_entry_command_fn, |
| 179 | }; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 180 | /* *INDENT-ON* */ |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 181 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 182 | /** Check if LISP-GPE is enabled. */ |
Andrej Kozemcak | a9edd85 | 2016-05-02 12:14:33 +0200 | [diff] [blame] | 183 | u8 |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 184 | vnet_lisp_gpe_enable_disable_status (void) |
Andrej Kozemcak | a9edd85 | 2016-05-02 12:14:33 +0200 | [diff] [blame] | 185 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 186 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
Andrej Kozemcak | a9edd85 | 2016-05-02 12:14:33 +0200 | [diff] [blame] | 187 | |
| 188 | return lgm->is_en; |
| 189 | } |
| 190 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 191 | /** Enable/disable LISP-GPE. */ |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 192 | clib_error_t * |
| 193 | vnet_lisp_gpe_enable_disable (vnet_lisp_gpe_enable_disable_args_t * a) |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 194 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 195 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
Florin Coras | fab2393 | 2018-10-21 10:50:48 -0700 | [diff] [blame] | 196 | vlib_main_t *vm = vlib_get_main (); |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 197 | |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 198 | if (a->is_en) |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 199 | { |
Andrej Kozemcak | a9edd85 | 2016-05-02 12:14:33 +0200 | [diff] [blame] | 200 | lgm->is_en = 1; |
Florin Coras | fab2393 | 2018-10-21 10:50:48 -0700 | [diff] [blame] | 201 | udp_register_dst_port (vm, UDP_DST_PORT_lisp_gpe, |
| 202 | lisp_gpe_ip4_input_node.index, 1 /* is_ip4 */ ); |
| 203 | udp_register_dst_port (vm, UDP_DST_PORT_lisp_gpe6, |
| 204 | lisp_gpe_ip6_input_node.index, 0 /* is_ip4 */ ); |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 205 | } |
| 206 | else |
| 207 | { |
Neale Ranns | 0bfe5d8 | 2016-08-25 15:29:12 +0100 | [diff] [blame] | 208 | /* remove all entries */ |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 209 | vnet_lisp_gpe_fwd_entry_flush (); |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 210 | |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 211 | /* disable all l3 ifaces */ |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 212 | lisp_gpe_tenant_flush (); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 213 | |
Florin Coras | fab2393 | 2018-10-21 10:50:48 -0700 | [diff] [blame] | 214 | udp_unregister_dst_port (vm, UDP_DST_PORT_lisp_gpe, 0 /* is_ip4 */ ); |
| 215 | udp_unregister_dst_port (vm, UDP_DST_PORT_lisp_gpe6, 1 /* is_ip4 */ ); |
Andrej Kozemcak | a9edd85 | 2016-05-02 12:14:33 +0200 | [diff] [blame] | 216 | lgm->is_en = 0; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 217 | } |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 218 | |
| 219 | return 0; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 220 | } |
| 221 | |
Filip Tehlar | 3e7b5693 | 2017-02-21 18:28:34 +0100 | [diff] [blame] | 222 | /** Set GPE encapsulation mode. */ |
| 223 | int |
| 224 | vnet_gpe_set_encap_mode (gpe_encap_mode_t mode) |
| 225 | { |
| 226 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
| 227 | |
| 228 | if (mode >= GPE_ENCAP_COUNT) |
| 229 | return VNET_API_ERROR_INVALID_GPE_MODE; |
| 230 | |
| 231 | if (pool_elts (lgm->lisp_fwd_entry_pool) != 0) |
| 232 | return VNET_API_ERROR_LISP_GPE_ENTRIES_PRESENT; |
| 233 | |
| 234 | lgm->encap_mode = mode; |
| 235 | return 0; |
| 236 | } |
| 237 | |
| 238 | /** CLI command to set GPE encap */ |
| 239 | static clib_error_t * |
| 240 | gpe_set_encap_mode_command_fn (vlib_main_t * vm, |
| 241 | unformat_input_t * input, |
| 242 | vlib_cli_command_t * cmd) |
| 243 | { |
| 244 | unformat_input_t _line_input, *line_input = &_line_input; |
| 245 | gpe_encap_mode_t mode = GPE_ENCAP_COUNT; |
| 246 | vnet_api_error_t rv; |
| 247 | |
| 248 | /* Get a line of input. */ |
| 249 | if (!unformat_user (input, unformat_line_input, line_input)) |
| 250 | return 0; |
| 251 | |
| 252 | while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT) |
| 253 | { |
| 254 | if (unformat (line_input, "lisp")) |
| 255 | mode = GPE_ENCAP_LISP; |
| 256 | else if (unformat (line_input, "vxlan")) |
| 257 | mode = GPE_ENCAP_VXLAN; |
| 258 | else |
| 259 | { |
| 260 | return clib_error_return (0, "parse error: '%U'", |
| 261 | format_unformat_error, line_input); |
| 262 | } |
| 263 | } |
| 264 | rv = vnet_gpe_set_encap_mode (mode); |
| 265 | if (rv) |
| 266 | { |
| 267 | return clib_error_return (0, |
| 268 | "Error: invalid mode or GPE entries are present!"); |
| 269 | } |
| 270 | |
| 271 | return 0; |
| 272 | } |
| 273 | |
| 274 | /* *INDENT-OFF* */ |
| 275 | VLIB_CLI_COMMAND (gpe_set_encap_mode_command, static) = { |
| 276 | .path = "gpe encap", |
| 277 | .short_help = "gpe encap [lisp|vxlan]", |
| 278 | .function = gpe_set_encap_mode_command_fn, |
| 279 | }; |
| 280 | /* *INDENT-ON* */ |
| 281 | |
| 282 | /** Format GPE encap mode. */ |
| 283 | u8 * |
| 284 | format_vnet_gpe_encap_mode (u8 * s, va_list * args) |
| 285 | { |
| 286 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
| 287 | |
| 288 | switch (lgm->encap_mode) |
| 289 | { |
| 290 | case GPE_ENCAP_LISP: |
| 291 | return format (s, "lisp"); |
| 292 | case GPE_ENCAP_VXLAN: |
| 293 | return format (s, "vxlan"); |
| 294 | default: |
| 295 | return 0; |
| 296 | } |
| 297 | return 0; |
| 298 | } |
| 299 | |
| 300 | /** CLI command to show GPE encap */ |
| 301 | static clib_error_t * |
| 302 | gpe_show_encap_mode_command_fn (vlib_main_t * vm, |
| 303 | unformat_input_t * input, |
| 304 | vlib_cli_command_t * cmd) |
| 305 | { |
| 306 | vlib_cli_output (vm, "encap mode: %U", format_vnet_gpe_encap_mode); |
| 307 | return 0; |
| 308 | } |
| 309 | |
| 310 | /* *INDENT-OFF* */ |
| 311 | VLIB_CLI_COMMAND (gpe_show_encap_mode_command, static) = { |
| 312 | .path = "show gpe encap", |
| 313 | .short_help = "show GPE encapulation mode", |
| 314 | .function = gpe_show_encap_mode_command_fn, |
| 315 | }; |
| 316 | /* *INDENT-ON* */ |
| 317 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 318 | /** CLI command to enable/disable LISP-GPE. */ |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 319 | static clib_error_t * |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 320 | lisp_gpe_enable_disable_command_fn (vlib_main_t * vm, |
| 321 | unformat_input_t * input, |
| 322 | vlib_cli_command_t * cmd) |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 323 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 324 | unformat_input_t _line_input, *line_input = &_line_input; |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 325 | u8 is_en = 1; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 326 | vnet_lisp_gpe_enable_disable_args_t _a, *a = &_a; |
Billy McFall | a9a20e7 | 2017-02-15 11:39:12 -0500 | [diff] [blame] | 327 | clib_error_t *error = NULL; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 328 | |
| 329 | /* Get a line of input. */ |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 330 | if (!unformat_user (input, unformat_line_input, line_input)) |
Florin Coras | 9fa1581 | 2017-12-01 04:54:06 -0800 | [diff] [blame] | 331 | return clib_error_return (0, "expected enable | disable"); |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 332 | |
| 333 | while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT) |
| 334 | { |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 335 | if (unformat (line_input, "enable")) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 336 | is_en = 1; |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 337 | else if (unformat (line_input, "disable")) |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 338 | is_en = 0; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 339 | else |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 340 | { |
Billy McFall | a9a20e7 | 2017-02-15 11:39:12 -0500 | [diff] [blame] | 341 | error = clib_error_return (0, "parse error: '%U'", |
| 342 | format_unformat_error, line_input); |
| 343 | goto done; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 344 | } |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 345 | } |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 346 | a->is_en = is_en; |
Billy McFall | a9a20e7 | 2017-02-15 11:39:12 -0500 | [diff] [blame] | 347 | error = vnet_lisp_gpe_enable_disable (a); |
| 348 | |
| 349 | done: |
| 350 | unformat_free (line_input); |
| 351 | |
| 352 | return error; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 353 | } |
| 354 | |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 355 | /* *INDENT-OFF* */ |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 356 | VLIB_CLI_COMMAND (enable_disable_lisp_gpe_command, static) = { |
Filip Tehlar | 82786c4 | 2017-02-20 15:20:37 +0100 | [diff] [blame] | 357 | .path = "gpe", |
| 358 | .short_help = "gpe [enable|disable]", |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 359 | .function = lisp_gpe_enable_disable_command_fn, |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 360 | }; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 361 | /* *INDENT-ON* */ |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 362 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 363 | /** CLI command to show LISP-GPE interfaces. */ |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 364 | static clib_error_t * |
| 365 | lisp_show_iface_command_fn (vlib_main_t * vm, |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 366 | unformat_input_t * input, |
| 367 | vlib_cli_command_t * cmd) |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 368 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 369 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
| 370 | hash_pair_t *p; |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 371 | |
| 372 | vlib_cli_output (vm, "%=10s%=12s", "vrf", "hw_if_index"); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 373 | |
| 374 | /* *INDENT-OFF* */ |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 375 | hash_foreach_pair (p, lgm->l3_ifaces.hw_if_index_by_dp_table, ({ |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 376 | vlib_cli_output (vm, "%=10d%=10d", p->key, p->value[0]); |
| 377 | })); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 378 | /* *INDENT-ON* */ |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 379 | |
| 380 | if (0 != lgm->l2_ifaces.hw_if_index_by_dp_table) |
| 381 | { |
| 382 | vlib_cli_output (vm, "%=10s%=12s", "bd_id", "hw_if_index"); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 383 | /* *INDENT-OFF* */ |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 384 | hash_foreach_pair (p, lgm->l2_ifaces.hw_if_index_by_dp_table, ({ |
| 385 | vlib_cli_output (vm, "%=10d%=10d", p->key, p->value[0]); |
| 386 | })); |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 387 | /* *INDENT-ON* */ |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 388 | } |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 389 | return 0; |
| 390 | } |
| 391 | |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 392 | /* *INDENT-OFF* */ |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 393 | VLIB_CLI_COMMAND (lisp_show_iface_command) = { |
Filip Tehlar | 82786c4 | 2017-02-20 15:20:37 +0100 | [diff] [blame] | 394 | .path = "show gpe interface", |
| 395 | .short_help = "show gpe interface", |
Filip Tehlar | 53f09e3 | 2016-05-19 14:25:44 +0200 | [diff] [blame] | 396 | .function = lisp_show_iface_command_fn, |
| 397 | }; |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 398 | /* *INDENT-ON* */ |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 399 | |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 400 | /** CLI command to show GPE fwd native route path. */ |
| 401 | static clib_error_t * |
| 402 | gpe_show_native_fwd_rpath_command_fn (vlib_main_t * vm, |
| 403 | unformat_input_t * input, |
| 404 | vlib_cli_command_t * cmd) |
| 405 | { |
| 406 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
| 407 | fib_route_path_t *rpath; |
| 408 | |
| 409 | if (vec_len (lgm->native_fwd_rpath[IP4])) |
| 410 | { |
| 411 | vec_foreach (rpath, lgm->native_fwd_rpath[IP4]) |
| 412 | { |
| 413 | vlib_cli_output (vm, "nh: %U fib_index %u sw_if_index %u", |
Filip Tehlar | b4243aa | 2017-06-14 14:39:42 +0200 | [diff] [blame] | 414 | format_ip46_address, &rpath->frp_addr, |
| 415 | IP46_TYPE_IP4, rpath->frp_fib_index, |
| 416 | rpath->frp_sw_if_index); |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 417 | } |
| 418 | } |
| 419 | if (vec_len (lgm->native_fwd_rpath[IP6])) |
| 420 | { |
| 421 | vec_foreach (rpath, lgm->native_fwd_rpath[IP6]) |
| 422 | { |
| 423 | vlib_cli_output (vm, "nh: %U fib_index %u sw_if_index %u", |
Filip Tehlar | b4243aa | 2017-06-14 14:39:42 +0200 | [diff] [blame] | 424 | format_ip46_address, &rpath->frp_addr, IP46_TYPE_IP6, |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 425 | rpath->frp_fib_index, rpath->frp_sw_if_index); |
| 426 | } |
| 427 | } |
| 428 | return 0; |
| 429 | } |
| 430 | |
| 431 | /* *INDENT-OFF* */ |
| 432 | VLIB_CLI_COMMAND (gpe_show_native_fwd_rpath_command) = { |
| 433 | .path = "show gpe native-forward", |
| 434 | .short_help = "show gpe native-forward", |
| 435 | .function = gpe_show_native_fwd_rpath_command_fn, |
| 436 | }; |
| 437 | /* *INDENT-ON* */ |
| 438 | |
| 439 | void |
| 440 | gpe_update_native_fwd_path (u8 ip_version) |
| 441 | { |
| 442 | lisp_gpe_main_t *lgm = vnet_lisp_gpe_get_main (); |
| 443 | lisp_gpe_fwd_entry_t *lfe; |
| 444 | fib_prefix_t fib_prefix; |
| 445 | u32 *lfei; |
| 446 | |
| 447 | vec_foreach (lfei, lgm->native_fwd_lfes[ip_version]) |
| 448 | { |
| 449 | lfe = pool_elt_at_index (lgm->lisp_fwd_entry_pool, lfei[0]); |
| 450 | ip_prefix_to_fib_prefix (&lfe->key->rmt.ippref, &fib_prefix); |
| 451 | fib_table_entry_update (lfe->eid_fib_index, &fib_prefix, FIB_SOURCE_LISP, |
| 452 | FIB_ENTRY_FLAG_NONE, |
| 453 | lgm->native_fwd_rpath[ip_version]); |
| 454 | } |
| 455 | } |
| 456 | |
| 457 | int |
| 458 | vnet_gpe_add_del_native_fwd_rpath (vnet_gpe_native_fwd_rpath_args_t * a) |
| 459 | { |
| 460 | lisp_gpe_main_t *lgm = vnet_lisp_gpe_get_main (); |
| 461 | fib_route_path_t *rpath; |
| 462 | u8 ip_version; |
| 463 | |
Neale Ranns | da78f95 | 2017-05-24 09:15:43 -0700 | [diff] [blame] | 464 | ip_version = a->rpath.frp_proto == DPO_PROTO_IP4 ? IP4 : IP6; |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 465 | |
| 466 | if (a->is_add) |
| 467 | { |
| 468 | vec_add1 (lgm->native_fwd_rpath[ip_version], a->rpath); |
| 469 | } |
| 470 | else |
| 471 | { |
| 472 | vec_foreach (rpath, lgm->native_fwd_rpath[ip_version]) |
| 473 | { |
| 474 | if (!fib_route_path_cmp (rpath, &a->rpath)) |
| 475 | { |
| 476 | vec_del1 (lgm->native_fwd_rpath[ip_version], |
| 477 | rpath - lgm->native_fwd_rpath[ip_version]); |
| 478 | break; |
| 479 | } |
| 480 | } |
| 481 | } |
| 482 | gpe_update_native_fwd_path (ip_version); |
| 483 | return 0; |
| 484 | } |
| 485 | |
| 486 | /** |
| 487 | * CLI command to add action for native forward. |
| 488 | */ |
| 489 | static clib_error_t * |
| 490 | gpe_native_forward_command_fn (vlib_main_t * vm, unformat_input_t * input, |
| 491 | vlib_cli_command_t * cmd) |
| 492 | { |
| 493 | vnet_main_t *vnm = vnet_get_main (); |
| 494 | unformat_input_t _line_input, *line_input = &_line_input; |
| 495 | vnet_api_error_t rv; |
| 496 | fib_route_path_t rpath; |
| 497 | u32 table_id = ~0; |
| 498 | vnet_gpe_native_fwd_rpath_args_t _a, *a = &_a; |
| 499 | u8 is_add = 1; |
| 500 | clib_error_t *error = 0; |
| 501 | |
| 502 | /* Get a line of input. */ |
| 503 | if (!unformat_user (input, unformat_line_input, line_input)) |
| 504 | return 0; |
| 505 | |
| 506 | memset (&rpath, 0, sizeof (rpath)); |
| 507 | |
| 508 | while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT) |
| 509 | { |
| 510 | if (unformat (line_input, "table %d", &table_id)) |
| 511 | ; |
| 512 | else if (unformat (line_input, "del")) |
| 513 | is_add = 0; |
| 514 | else if (unformat (line_input, "via %U %U", |
| 515 | unformat_ip4_address, |
| 516 | &rpath.frp_addr.ip4, |
| 517 | unformat_vnet_sw_interface, vnm, |
| 518 | &rpath.frp_sw_if_index)) |
| 519 | { |
| 520 | rpath.frp_weight = 1; |
Neale Ranns | da78f95 | 2017-05-24 09:15:43 -0700 | [diff] [blame] | 521 | rpath.frp_proto = DPO_PROTO_IP4; |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 522 | } |
| 523 | else if (unformat (line_input, "via %U %U", |
| 524 | unformat_ip6_address, |
| 525 | &rpath.frp_addr.ip6, |
| 526 | unformat_vnet_sw_interface, vnm, |
| 527 | &rpath.frp_sw_if_index)) |
| 528 | { |
| 529 | rpath.frp_weight = 1; |
Neale Ranns | da78f95 | 2017-05-24 09:15:43 -0700 | [diff] [blame] | 530 | rpath.frp_proto = DPO_PROTO_IP6; |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 531 | } |
| 532 | else if (unformat (line_input, "via %U", |
| 533 | unformat_ip4_address, &rpath.frp_addr.ip4)) |
| 534 | { |
| 535 | rpath.frp_weight = 1; |
| 536 | rpath.frp_sw_if_index = ~0; |
Neale Ranns | da78f95 | 2017-05-24 09:15:43 -0700 | [diff] [blame] | 537 | rpath.frp_proto = DPO_PROTO_IP4; |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 538 | } |
| 539 | else if (unformat (line_input, "via %U", |
| 540 | unformat_ip6_address, &rpath.frp_addr.ip6)) |
| 541 | { |
| 542 | rpath.frp_weight = 1; |
| 543 | rpath.frp_sw_if_index = ~0; |
Neale Ranns | da78f95 | 2017-05-24 09:15:43 -0700 | [diff] [blame] | 544 | rpath.frp_proto = DPO_PROTO_IP6; |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 545 | } |
| 546 | else |
| 547 | { |
| 548 | return clib_error_return (0, "parse error: '%U'", |
| 549 | format_unformat_error, line_input); |
| 550 | } |
| 551 | } |
| 552 | |
| 553 | if ((u32) ~ 0 == table_id) |
| 554 | { |
| 555 | rpath.frp_fib_index = 0; |
| 556 | } |
| 557 | else |
| 558 | { |
Neale Ranns | da78f95 | 2017-05-24 09:15:43 -0700 | [diff] [blame] | 559 | rpath.frp_fib_index = |
| 560 | fib_table_find (dpo_proto_to_fib (rpath.frp_proto), table_id); |
Florin Coras | a4e63e5 | 2017-06-07 21:50:57 -0700 | [diff] [blame] | 561 | if ((u32) ~ 0 == rpath.frp_fib_index) |
| 562 | { |
| 563 | error = clib_error_return (0, "Nonexistent table id %d", table_id); |
| 564 | goto done; |
| 565 | } |
| 566 | } |
| 567 | |
| 568 | a->rpath = rpath; |
| 569 | a->is_add = is_add; |
| 570 | |
| 571 | rv = vnet_gpe_add_del_native_fwd_rpath (a); |
| 572 | if (rv) |
| 573 | { |
| 574 | return clib_error_return (0, "Error: couldn't add path!"); |
| 575 | } |
| 576 | |
| 577 | done: |
| 578 | return error; |
| 579 | } |
| 580 | |
| 581 | /* *INDENT-OFF* */ |
| 582 | VLIB_CLI_COMMAND (gpe_native_forward_command) = { |
| 583 | .path = "gpe native-forward", |
| 584 | .short_help = "gpe native-forward [del] via <nh-ip-addr> [iface] " |
| 585 | "[table <table>]", |
| 586 | .function = gpe_native_forward_command_fn, |
| 587 | }; |
| 588 | /* *INDENT-ON* */ |
| 589 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 590 | /** Format LISP-GPE status. */ |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 591 | u8 * |
| 592 | format_vnet_lisp_gpe_status (u8 * s, va_list * args) |
| 593 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 594 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 595 | return format (s, "%s", lgm->is_en ? "enabled" : "disabled"); |
| 596 | } |
| 597 | |
Florin Coras | ff0bf13 | 2016-09-05 19:30:35 +0200 | [diff] [blame] | 598 | /** LISP-GPE init function. */ |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 599 | clib_error_t * |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 600 | lisp_gpe_init (vlib_main_t * vm) |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 601 | { |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 602 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
| 603 | clib_error_t *error = 0; |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 604 | |
| 605 | if ((error = vlib_call_init_function (vm, ip_main_init))) |
| 606 | return error; |
| 607 | |
| 608 | if ((error = vlib_call_init_function (vm, ip4_lookup_init))) |
| 609 | return error; |
| 610 | |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 611 | lgm->vnet_main = vnet_get_main (); |
Florin Coras | e127a7e | 2016-02-18 22:20:01 +0100 | [diff] [blame] | 612 | lgm->vlib_main = vm; |
| 613 | lgm->im4 = &ip4_main; |
Florin Coras | 02655bd | 2016-04-26 00:17:24 +0200 | [diff] [blame] | 614 | lgm->im6 = &ip6_main; |
| 615 | lgm->lm4 = &ip4_main.lookup_main; |
| 616 | lgm->lm6 = &ip6_main.lookup_main; |
Filip Tehlar | 3e7b5693 | 2017-02-21 18:28:34 +0100 | [diff] [blame] | 617 | lgm->encap_mode = GPE_ENCAP_LISP; |
Florin Coras | 577c355 | 2016-04-21 00:45:40 +0200 | [diff] [blame] | 618 | |
Neale Ranns | 5e575b1 | 2016-10-03 09:40:25 +0100 | [diff] [blame] | 619 | lgm->lisp_gpe_fwd_entries = |
| 620 | hash_create_mem (0, sizeof (lisp_gpe_fwd_entry_key_t), sizeof (uword)); |
Florin Coras | 1a1adc7 | 2016-07-22 01:45:30 +0200 | [diff] [blame] | 621 | |
Filip Tehlar | 4868ff6 | 2017-03-09 16:48:39 +0100 | [diff] [blame] | 622 | lgm->lisp_stats_index_by_key = |
| 623 | hash_create_mem (0, sizeof (lisp_stats_key_t), sizeof (uword)); |
Filip Tehlar | 2151191 | 2017-04-07 10:41:42 +0200 | [diff] [blame] | 624 | memset (&lgm->counters, 0, sizeof (lgm->counters)); |
| 625 | lgm->counters.name = "LISP counters"; |
| 626 | |
Ed Warnicke | cb9cada | 2015-12-08 15:45:58 -0700 | [diff] [blame] | 627 | return 0; |
| 628 | } |
| 629 | |
Filip Tehlar | 3e7b5693 | 2017-02-21 18:28:34 +0100 | [diff] [blame] | 630 | gpe_encap_mode_t |
| 631 | vnet_gpe_get_encap_mode (void) |
| 632 | { |
| 633 | lisp_gpe_main_t *lgm = &lisp_gpe_main; |
| 634 | return lgm->encap_mode; |
| 635 | } |
| 636 | |
Filip Tehlar | f89d185 | 2017-06-12 13:36:02 +0200 | [diff] [blame] | 637 | static clib_error_t * |
| 638 | lisp_gpe_test_send_nsh_packet (u8 * file_name) |
| 639 | { |
| 640 | vlib_frame_t *f; |
| 641 | vlib_buffer_t *b; |
| 642 | lisp_gpe_main_t *lgm = vnet_lisp_gpe_get_main (); |
| 643 | pcap_main_t pm; |
| 644 | clib_error_t *error = 0; |
| 645 | |
| 646 | if (!file_name) |
| 647 | return clib_error_create ("no pcap file specified!"); |
| 648 | |
| 649 | memset (&pm, 0, sizeof (pm)); |
| 650 | pm.file_name = (char *) file_name; |
| 651 | error = pcap_read (&pm); |
| 652 | if (error) |
| 653 | return error; |
| 654 | |
| 655 | u32 bi; |
| 656 | if (vlib_buffer_alloc (lgm->vlib_main, &bi, 1) != 1) |
| 657 | return clib_error_create ("cannot allocate memory!"); |
| 658 | |
| 659 | b = vlib_get_buffer (lgm->vlib_main, bi); |
| 660 | tunnel_lookup_t *nsh_ifaces = &lgm->nsh_ifaces; |
| 661 | uword *hip; |
| 662 | vnet_hw_interface_t *hi; |
| 663 | |
| 664 | hip = hash_get (nsh_ifaces->hw_if_index_by_dp_table, 0); |
| 665 | if (hip == 0) |
| 666 | return clib_error_create ("The NSH 0 interface doesn't exist"); |
| 667 | |
| 668 | hi = vnet_get_hw_interface (lgm->vnet_main, hip[0]); |
| 669 | |
| 670 | vnet_buffer (b)->sw_if_index[VLIB_TX] = hi->sw_if_index; |
| 671 | u8 *p = vlib_buffer_put_uninit (b, vec_len (pm.packets_read[0])); |
| 672 | clib_memcpy (p, pm.packets_read[0], vec_len (pm.packets_read[0])); |
| 673 | vlib_buffer_pull (b, sizeof (ethernet_header_t)); |
| 674 | |
| 675 | vlib_node_t *n = vlib_get_node_by_name (lgm->vlib_main, |
| 676 | (u8 *) "interface-tx"); |
| 677 | f = vlib_get_frame_to_node (lgm->vlib_main, n->index); |
| 678 | u32 *to_next = vlib_frame_vector_args (f); |
| 679 | to_next[0] = bi; |
| 680 | f->n_vectors = 1; |
| 681 | vlib_put_frame_to_node (lgm->vlib_main, n->index, f); |
| 682 | |
| 683 | return error; |
| 684 | } |
| 685 | |
| 686 | static clib_error_t * |
| 687 | lisp_test_nsh_command_fn (vlib_main_t * vm, unformat_input_t * input, |
| 688 | vlib_cli_command_t * cmd) |
| 689 | { |
| 690 | clib_error_t *error = 0; |
| 691 | u8 *file_name = 0; |
| 692 | |
| 693 | while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT) |
| 694 | { |
| 695 | if (unformat (input, "pcap %v", &file_name)) |
| 696 | { |
| 697 | error = lisp_gpe_test_send_nsh_packet (file_name); |
| 698 | goto done; |
| 699 | } |
| 700 | else |
| 701 | { |
| 702 | error = clib_error_create ("unknown input `%U'", |
| 703 | format_unformat_error, input); |
| 704 | goto done; |
| 705 | } |
| 706 | } |
| 707 | |
| 708 | done: |
| 709 | return error; |
| 710 | } |
| 711 | |
| 712 | /* *INDENT-OFF* */ |
| 713 | VLIB_CLI_COMMAND (lisp_test_nsh_command, static) = { |
| 714 | .path = "test one nsh", |
| 715 | .short_help = "test gpe nsh pcap <path-to-pcap-file>", |
| 716 | .function = lisp_test_nsh_command_fn, |
| 717 | }; |
| 718 | /* *INDENT-ON* */ |
| 719 | |
Florin Coras | 220beac | 2016-08-16 23:04:00 +0200 | [diff] [blame] | 720 | VLIB_INIT_FUNCTION (lisp_gpe_init); |
| 721 | |
| 722 | /* |
| 723 | * fd.io coding-style-patch-verification: ON |
| 724 | * |
| 725 | * Local Variables: |
| 726 | * eval: (c-set-style "gnu") |
| 727 | * End: |
| 728 | */ |