blob: 479a6260ea328cb82de22c7a7379f9147ce4f359 [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
Florin Corase127a7e2016-02-18 22:20:01 +01002 * Copyright (c) 2016 Cisco and/or its affiliates.
Ed Warnickecb9cada2015-12-08 15:45:58 -07003 * 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 Corasff0bf132016-09-05 19:30:35 +020015/**
16 * @file
17 * @brief Common utility functions for IPv4, IPv6 and L2 LISP-GPE tunnels.
18 *
19 */
Florin Corase127a7e2016-02-18 22:20:01 +010020
Ed Warnickecb9cada2015-12-08 15:45:58 -070021#include <vnet/lisp-gpe/lisp_gpe.h>
Neale Ranns5e575b12016-10-03 09:40:25 +010022#include <vnet/lisp-gpe/lisp_gpe_fwd_entry.h>
Neale Ranns0bfe5d82016-08-25 15:29:12 +010023#include <vnet/lisp-gpe/lisp_gpe_adjacency.h>
Neale Ranns5e575b12016-10-03 09:40:25 +010024#include <vnet/lisp-gpe/lisp_gpe_tenant.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070025
Florin Corasff0bf132016-09-05 19:30:35 +020026/** LISP-GPE global state */
Ed Warnickecb9cada2015-12-08 15:45:58 -070027lisp_gpe_main_t lisp_gpe_main;
28
Florin Coras1a1adc72016-07-22 01:45:30 +020029
Florin Corasff0bf132016-09-05 19:30:35 +020030/** CLI command to add/del forwarding entry. */
Ed Warnickecb9cada2015-12-08 15:45:58 -070031static clib_error_t *
Florin Corase127a7e2016-02-18 22:20:01 +010032lisp_gpe_add_del_fwd_entry_command_fn (vlib_main_t * vm,
Florin Coras220beac2016-08-16 23:04:00 +020033 unformat_input_t * input,
34 vlib_cli_command_t * cmd)
Ed Warnickecb9cada2015-12-08 15:45:58 -070035{
Florin Coras220beac2016-08-16 23:04:00 +020036 unformat_input_t _line_input, *line_input = &_line_input;
Ed Warnickecb9cada2015-12-08 15:45:58 -070037 u8 is_add = 1;
Florin Corasbb5c22f2016-08-02 02:31:03 +020038 ip_address_t lloc, rloc;
Florin Coras220beac2016-08-16 23:04:00 +020039 clib_error_t *error = 0;
40 gid_address_t _reid, *reid = &_reid, _leid, *leid = &_leid;
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010041 u8 reid_set = 0, leid_set = 0, is_negative = 0, dp_table_set = 0,
42 vni_set = 0;
43 u32 vni = 0, dp_table = 0, action = ~0, w;
Florin Coras220beac2016-08-16 23:04:00 +020044 locator_pair_t pair, *pairs = 0;
Andrej Kozemcak8ebb2a12016-06-07 12:25:20 +020045 int rv;
Florin Corase127a7e2016-02-18 22:20:01 +010046
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010047 memset (leid, 0, sizeof (*leid));
48 memset (reid, 0, sizeof (*reid));
49
Ed Warnickecb9cada2015-12-08 15:45:58 -070050 /* Get a line of input. */
Florin Coras220beac2016-08-16 23:04:00 +020051 if (!unformat_user (input, unformat_line_input, line_input))
Ed Warnickecb9cada2015-12-08 15:45:58 -070052 return 0;
53
Florin Corase127a7e2016-02-18 22:20:01 +010054 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
55 {
56 if (unformat (line_input, "del"))
Florin Coras220beac2016-08-16 23:04:00 +020057 is_add = 0;
Florin Corase127a7e2016-02-18 22:20:01 +010058 else if (unformat (line_input, "add"))
Florin Coras220beac2016-08-16 23:04:00 +020059 is_add = 1;
60 else if (unformat (line_input, "leid %U", unformat_gid_address, leid))
61 {
62 leid_set = 1;
63 }
64 else if (unformat (line_input, "reid %U", unformat_gid_address, reid))
65 {
66 reid_set = 1;
67 }
Florin Coras03c4f992016-07-19 15:27:58 +020068 else if (unformat (line_input, "vni %u", &vni))
Florin Coras220beac2016-08-16 23:04:00 +020069 {
70 gid_address_vni (leid) = vni;
71 gid_address_vni (reid) = vni;
72 vni_set = 1;
73 }
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010074 else if (unformat (line_input, "vrf %u", &dp_table))
Florin Coras220beac2016-08-16 23:04:00 +020075 {
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010076 dp_table_set = 1;
Florin Coras220beac2016-08-16 23:04:00 +020077 }
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010078 else if (unformat (line_input, "bd %u", &dp_table))
Neale Ranns5e575b12016-10-03 09:40:25 +010079 {
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010080 dp_table_set = 1;
Neale Ranns5e575b12016-10-03 09:40:25 +010081 }
Florin Coras03c4f992016-07-19 15:27:58 +020082 else if (unformat (line_input, "negative action %U",
Florin Coras220beac2016-08-16 23:04:00 +020083 unformat_negative_mapping_action, &action))
84 {
85 is_negative = 1;
86 }
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010087 else if (unformat (line_input, "loc-pair %U %U w %d",
Florin Coras220beac2016-08-16 23:04:00 +020088 unformat_ip_address, &lloc,
Filip Tehlarc3af7bf2017-01-13 14:13:09 +010089 unformat_ip_address, &rloc, &w))
Florin Coras220beac2016-08-16 23:04:00 +020090 {
91 pair.lcl_loc = lloc;
92 pair.rmt_loc = rloc;
Florin Coras220beac2016-08-16 23:04:00 +020093 pair.weight = w;
Florin Coras42e480d2017-01-16 00:57:02 -080094 pair.priority = 0;
Florin Coras220beac2016-08-16 23:04:00 +020095 vec_add1 (pairs, pair);
96 }
Florin Corase127a7e2016-02-18 22:20:01 +010097 else
Florin Coras220beac2016-08-16 23:04:00 +020098 {
99 error = unformat_parse_error (line_input);
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100100 vlib_cli_output (vm, "parse error: '%U'",
101 format_unformat_error, line_input);
Florin Coras220beac2016-08-16 23:04:00 +0200102 goto done;
103 }
Florin Corase127a7e2016-02-18 22:20:01 +0100104 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700105
Florin Coras03c4f992016-07-19 15:27:58 +0200106 if (!reid_set)
Florin Corase127a7e2016-02-18 22:20:01 +0100107 {
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100108 vlib_cli_output (vm, "remote eid must be set!");
Florin Corase127a7e2016-02-18 22:20:01 +0100109 goto done;
110 }
111
Florin Corasb69111e2017-02-13 23:55:27 -0800112 if (gid_address_type (reid) != GID_ADDR_NSH && (!vni_set || !dp_table_set))
113 {
114 vlib_cli_output (vm, "vni and vrf/bd must be set!");
115 goto done;
116 }
117
Florin Coras03c4f992016-07-19 15:27:58 +0200118 if (is_negative)
Florin Corase127a7e2016-02-18 22:20:01 +0100119 {
Florin Coras03c4f992016-07-19 15:27:58 +0200120 if (~0 == action)
Florin Coras220beac2016-08-16 23:04:00 +0200121 {
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100122 vlib_cli_output (vm, "no action set for negative tunnel!");
Florin Coras220beac2016-08-16 23:04:00 +0200123 goto done;
124 }
Florin Coras03c4f992016-07-19 15:27:58 +0200125 }
126 else
127 {
Florin Corasbb5c22f2016-08-02 02:31:03 +0200128 if (vec_len (pairs) == 0)
Florin Coras220beac2016-08-16 23:04:00 +0200129 {
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100130 vlib_cli_output (vm, "expected ip4/ip6 locators");
Florin Coras220beac2016-08-16 23:04:00 +0200131 goto done;
132 }
Florin Corase127a7e2016-02-18 22:20:01 +0100133 }
134
Florin Coras03c4f992016-07-19 15:27:58 +0200135 if (!leid_set)
136 {
137 /* if leid not set, make sure it's the same AFI like reid */
Florin Coras220beac2016-08-16 23:04:00 +0200138 gid_address_type (leid) = gid_address_type (reid);
Florin Coras03c4f992016-07-19 15:27:58 +0200139 if (GID_ADDR_IP_PREFIX == gid_address_type (reid))
Florin Coras220beac2016-08-16 23:04:00 +0200140 gid_address_ip_version (leid) = gid_address_ip_version (reid);
Florin Coras03c4f992016-07-19 15:27:58 +0200141 }
142
143 /* add fwd entry */
Florin Coras220beac2016-08-16 23:04:00 +0200144 vnet_lisp_gpe_add_del_fwd_entry_args_t _a, *a = &_a;
145 memset (a, 0, sizeof (a[0]));
Florin Coras03c4f992016-07-19 15:27:58 +0200146
147 a->is_add = is_add;
Florin Coras82bf7cd2016-09-26 18:59:44 +0300148 a->is_negative = is_negative;
Florin Coras03c4f992016-07-19 15:27:58 +0200149 a->vni = vni;
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100150 a->table_id = dp_table;
Florin Coras220beac2016-08-16 23:04:00 +0200151 gid_address_copy (&a->lcl_eid, leid);
152 gid_address_copy (&a->rmt_eid, reid);
Florin Corasbb5c22f2016-08-02 02:31:03 +0200153 a->locator_pairs = pairs;
Florin Corasce1b4c72017-01-26 14:25:34 -0800154 a->action = action;
Florin Coras03c4f992016-07-19 15:27:58 +0200155
156 rv = vnet_lisp_gpe_add_del_fwd_entry (a, 0);
157 if (0 != rv)
158 {
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100159 vlib_cli_output (vm, "failed to %s gpe tunnel!",
160 is_add ? "add" : "delete");
Florin Coras03c4f992016-07-19 15:27:58 +0200161 }
162
Florin Coras220beac2016-08-16 23:04:00 +0200163done:
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100164 unformat_free (line_input);
Florin Coras220beac2016-08-16 23:04:00 +0200165 vec_free (pairs);
Florin Corase127a7e2016-02-18 22:20:01 +0100166 return error;
167}
168
Florin Coras220beac2016-08-16 23:04:00 +0200169/* *INDENT-OFF* */
Florin Corased09a052016-05-06 14:22:40 +0200170VLIB_CLI_COMMAND (lisp_gpe_add_del_fwd_entry_command, static) = {
Filip Tehlar82786c42017-02-20 15:20:37 +0100171 .path = "gpe entry",
172 .short_help = "gpe entry add/del vni <vni> vrf/bd <id> [leid <leid>]"
Filip Tehlarc3af7bf2017-01-13 14:13:09 +0100173 "reid <reid> [loc-pair <lloc> <rloc> w <weight>] "
Florin Corasff0bf132016-09-05 19:30:35 +0200174 "[negative action <action>]",
Florin Corase127a7e2016-02-18 22:20:01 +0100175 .function = lisp_gpe_add_del_fwd_entry_command_fn,
176};
Florin Coras220beac2016-08-16 23:04:00 +0200177/* *INDENT-ON* */
Florin Corase127a7e2016-02-18 22:20:01 +0100178
Florin Corasff0bf132016-09-05 19:30:35 +0200179/** Check if LISP-GPE is enabled. */
Andrej Kozemcaka9edd852016-05-02 12:14:33 +0200180u8
Florin Coras220beac2016-08-16 23:04:00 +0200181vnet_lisp_gpe_enable_disable_status (void)
Andrej Kozemcaka9edd852016-05-02 12:14:33 +0200182{
Florin Coras220beac2016-08-16 23:04:00 +0200183 lisp_gpe_main_t *lgm = &lisp_gpe_main;
Andrej Kozemcaka9edd852016-05-02 12:14:33 +0200184
185 return lgm->is_en;
186}
187
Florin Corasff0bf132016-09-05 19:30:35 +0200188/** Enable/disable LISP-GPE. */
Florin Coras577c3552016-04-21 00:45:40 +0200189clib_error_t *
190vnet_lisp_gpe_enable_disable (vnet_lisp_gpe_enable_disable_args_t * a)
Florin Corase127a7e2016-02-18 22:20:01 +0100191{
Florin Coras220beac2016-08-16 23:04:00 +0200192 lisp_gpe_main_t *lgm = &lisp_gpe_main;
Florin Corase127a7e2016-02-18 22:20:01 +0100193
Florin Coras577c3552016-04-21 00:45:40 +0200194 if (a->is_en)
Florin Corase127a7e2016-02-18 22:20:01 +0100195 {
Andrej Kozemcaka9edd852016-05-02 12:14:33 +0200196 lgm->is_en = 1;
Florin Corase127a7e2016-02-18 22:20:01 +0100197 }
198 else
199 {
Neale Ranns0bfe5d82016-08-25 15:29:12 +0100200 /* remove all entries */
Neale Ranns5e575b12016-10-03 09:40:25 +0100201 vnet_lisp_gpe_fwd_entry_flush ();
Florin Coras577c3552016-04-21 00:45:40 +0200202
Florin Coras1a1adc72016-07-22 01:45:30 +0200203 /* disable all l3 ifaces */
Neale Ranns5e575b12016-10-03 09:40:25 +0100204 lisp_gpe_tenant_flush ();
Florin Coras220beac2016-08-16 23:04:00 +0200205
Andrej Kozemcaka9edd852016-05-02 12:14:33 +0200206 lgm->is_en = 0;
Florin Corase127a7e2016-02-18 22:20:01 +0100207 }
Florin Coras577c3552016-04-21 00:45:40 +0200208
209 return 0;
Florin Corase127a7e2016-02-18 22:20:01 +0100210}
211
Florin Corasff0bf132016-09-05 19:30:35 +0200212/** CLI command to enable/disable LISP-GPE. */
Florin Corase127a7e2016-02-18 22:20:01 +0100213static clib_error_t *
Florin Coras220beac2016-08-16 23:04:00 +0200214lisp_gpe_enable_disable_command_fn (vlib_main_t * vm,
215 unformat_input_t * input,
216 vlib_cli_command_t * cmd)
Florin Corase127a7e2016-02-18 22:20:01 +0100217{
Florin Coras220beac2016-08-16 23:04:00 +0200218 unformat_input_t _line_input, *line_input = &_line_input;
Florin Coras577c3552016-04-21 00:45:40 +0200219 u8 is_en = 1;
Florin Coras220beac2016-08-16 23:04:00 +0200220 vnet_lisp_gpe_enable_disable_args_t _a, *a = &_a;
Billy McFalla9a20e72017-02-15 11:39:12 -0500221 clib_error_t *error = NULL;
Florin Corase127a7e2016-02-18 22:20:01 +0100222
223 /* Get a line of input. */
Florin Coras220beac2016-08-16 23:04:00 +0200224 if (!unformat_user (input, unformat_line_input, line_input))
Florin Corase127a7e2016-02-18 22:20:01 +0100225 return 0;
226
227 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
228 {
Florin Coras577c3552016-04-21 00:45:40 +0200229 if (unformat (line_input, "enable"))
Florin Coras220beac2016-08-16 23:04:00 +0200230 is_en = 1;
Florin Coras577c3552016-04-21 00:45:40 +0200231 else if (unformat (line_input, "disable"))
Florin Coras220beac2016-08-16 23:04:00 +0200232 is_en = 0;
Florin Corase127a7e2016-02-18 22:20:01 +0100233 else
Florin Coras220beac2016-08-16 23:04:00 +0200234 {
Billy McFalla9a20e72017-02-15 11:39:12 -0500235 error = clib_error_return (0, "parse error: '%U'",
236 format_unformat_error, line_input);
237 goto done;
Florin Coras220beac2016-08-16 23:04:00 +0200238 }
Florin Corase127a7e2016-02-18 22:20:01 +0100239 }
Florin Coras577c3552016-04-21 00:45:40 +0200240 a->is_en = is_en;
Billy McFalla9a20e72017-02-15 11:39:12 -0500241 error = vnet_lisp_gpe_enable_disable (a);
242
243done:
244 unformat_free (line_input);
245
246 return error;
Florin Corase127a7e2016-02-18 22:20:01 +0100247}
248
Florin Coras220beac2016-08-16 23:04:00 +0200249/* *INDENT-OFF* */
Florin Coras577c3552016-04-21 00:45:40 +0200250VLIB_CLI_COMMAND (enable_disable_lisp_gpe_command, static) = {
Filip Tehlar82786c42017-02-20 15:20:37 +0100251 .path = "gpe",
252 .short_help = "gpe [enable|disable]",
Florin Coras577c3552016-04-21 00:45:40 +0200253 .function = lisp_gpe_enable_disable_command_fn,
Florin Corase127a7e2016-02-18 22:20:01 +0100254};
Florin Coras220beac2016-08-16 23:04:00 +0200255/* *INDENT-ON* */
Florin Corase127a7e2016-02-18 22:20:01 +0100256
Florin Corasff0bf132016-09-05 19:30:35 +0200257/** CLI command to show LISP-GPE interfaces. */
Filip Tehlar53f09e32016-05-19 14:25:44 +0200258static clib_error_t *
259lisp_show_iface_command_fn (vlib_main_t * vm,
Florin Coras220beac2016-08-16 23:04:00 +0200260 unformat_input_t * input,
261 vlib_cli_command_t * cmd)
Filip Tehlar53f09e32016-05-19 14:25:44 +0200262{
Florin Coras220beac2016-08-16 23:04:00 +0200263 lisp_gpe_main_t *lgm = &lisp_gpe_main;
264 hash_pair_t *p;
Filip Tehlar53f09e32016-05-19 14:25:44 +0200265
266 vlib_cli_output (vm, "%=10s%=12s", "vrf", "hw_if_index");
Florin Coras220beac2016-08-16 23:04:00 +0200267
268 /* *INDENT-OFF* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200269 hash_foreach_pair (p, lgm->l3_ifaces.hw_if_index_by_dp_table, ({
Filip Tehlar53f09e32016-05-19 14:25:44 +0200270 vlib_cli_output (vm, "%=10d%=10d", p->key, p->value[0]);
271 }));
Florin Coras220beac2016-08-16 23:04:00 +0200272 /* *INDENT-ON* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200273
274 if (0 != lgm->l2_ifaces.hw_if_index_by_dp_table)
275 {
276 vlib_cli_output (vm, "%=10s%=12s", "bd_id", "hw_if_index");
Florin Coras220beac2016-08-16 23:04:00 +0200277 /* *INDENT-OFF* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200278 hash_foreach_pair (p, lgm->l2_ifaces.hw_if_index_by_dp_table, ({
279 vlib_cli_output (vm, "%=10d%=10d", p->key, p->value[0]);
280 }));
Florin Coras220beac2016-08-16 23:04:00 +0200281 /* *INDENT-ON* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200282 }
Filip Tehlar53f09e32016-05-19 14:25:44 +0200283 return 0;
284}
285
Florin Coras220beac2016-08-16 23:04:00 +0200286/* *INDENT-OFF* */
Filip Tehlar53f09e32016-05-19 14:25:44 +0200287VLIB_CLI_COMMAND (lisp_show_iface_command) = {
Filip Tehlar82786c42017-02-20 15:20:37 +0100288 .path = "show gpe interface",
289 .short_help = "show gpe interface",
Filip Tehlar53f09e32016-05-19 14:25:44 +0200290 .function = lisp_show_iface_command_fn,
291};
Florin Coras220beac2016-08-16 23:04:00 +0200292/* *INDENT-ON* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200293
Florin Corasff0bf132016-09-05 19:30:35 +0200294/** Format LISP-GPE status. */
Florin Coras1a1adc72016-07-22 01:45:30 +0200295u8 *
296format_vnet_lisp_gpe_status (u8 * s, va_list * args)
297{
Florin Coras220beac2016-08-16 23:04:00 +0200298 lisp_gpe_main_t *lgm = &lisp_gpe_main;
Florin Coras1a1adc72016-07-22 01:45:30 +0200299 return format (s, "%s", lgm->is_en ? "enabled" : "disabled");
300}
301
Florin Corasff0bf132016-09-05 19:30:35 +0200302/** LISP-GPE init function. */
Florin Corase127a7e2016-02-18 22:20:01 +0100303clib_error_t *
Florin Coras220beac2016-08-16 23:04:00 +0200304lisp_gpe_init (vlib_main_t * vm)
Florin Corase127a7e2016-02-18 22:20:01 +0100305{
Florin Coras220beac2016-08-16 23:04:00 +0200306 lisp_gpe_main_t *lgm = &lisp_gpe_main;
307 clib_error_t *error = 0;
Florin Corase127a7e2016-02-18 22:20:01 +0100308
309 if ((error = vlib_call_init_function (vm, ip_main_init)))
310 return error;
311
312 if ((error = vlib_call_init_function (vm, ip4_lookup_init)))
313 return error;
314
Florin Coras220beac2016-08-16 23:04:00 +0200315 lgm->vnet_main = vnet_get_main ();
Florin Corase127a7e2016-02-18 22:20:01 +0100316 lgm->vlib_main = vm;
317 lgm->im4 = &ip4_main;
Florin Coras02655bd2016-04-26 00:17:24 +0200318 lgm->im6 = &ip6_main;
319 lgm->lm4 = &ip4_main.lookup_main;
320 lgm->lm6 = &ip6_main.lookup_main;
Florin Coras577c3552016-04-21 00:45:40 +0200321
Neale Ranns5e575b12016-10-03 09:40:25 +0100322 lgm->lisp_gpe_fwd_entries =
323 hash_create_mem (0, sizeof (lisp_gpe_fwd_entry_key_t), sizeof (uword));
Florin Coras1a1adc72016-07-22 01:45:30 +0200324
Florin Coras220beac2016-08-16 23:04:00 +0200325 udp_register_dst_port (vm, UDP_DST_PORT_lisp_gpe,
326 lisp_gpe_ip4_input_node.index, 1 /* is_ip4 */ );
Florin Coras02655bd2016-04-26 00:17:24 +0200327 udp_register_dst_port (vm, UDP_DST_PORT_lisp_gpe6,
Florin Coras220beac2016-08-16 23:04:00 +0200328 lisp_gpe_ip6_input_node.index, 0 /* is_ip4 */ );
Florin Coras263440e2017-02-22 23:38:08 -0800329
Ed Warnickecb9cada2015-12-08 15:45:58 -0700330 return 0;
331}
332
Florin Coras220beac2016-08-16 23:04:00 +0200333VLIB_INIT_FUNCTION (lisp_gpe_init);
334
335/*
336 * fd.io coding-style-patch-verification: ON
337 *
338 * Local Variables:
339 * eval: (c-set-style "gnu")
340 * End:
341 */