blob: 1f8afdaefb8d10f1396bdc476eb84ea6408cc0c8 [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;
Florin Corase127a7e2016-02-18 22:20:01 +0100221
222 /* Get a line of input. */
Florin Coras220beac2016-08-16 23:04:00 +0200223 if (!unformat_user (input, unformat_line_input, line_input))
Florin Corase127a7e2016-02-18 22:20:01 +0100224 return 0;
225
226 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
227 {
Florin Coras577c3552016-04-21 00:45:40 +0200228 if (unformat (line_input, "enable"))
Florin Coras220beac2016-08-16 23:04:00 +0200229 is_en = 1;
Florin Coras577c3552016-04-21 00:45:40 +0200230 else if (unformat (line_input, "disable"))
Florin Coras220beac2016-08-16 23:04:00 +0200231 is_en = 0;
Florin Corase127a7e2016-02-18 22:20:01 +0100232 else
Florin Coras220beac2016-08-16 23:04:00 +0200233 {
234 return clib_error_return (0, "parse error: '%U'",
235 format_unformat_error, line_input);
236 }
Florin Corase127a7e2016-02-18 22:20:01 +0100237 }
Florin Coras577c3552016-04-21 00:45:40 +0200238 a->is_en = is_en;
239 return vnet_lisp_gpe_enable_disable (a);
Florin Corase127a7e2016-02-18 22:20:01 +0100240}
241
Florin Coras220beac2016-08-16 23:04:00 +0200242/* *INDENT-OFF* */
Florin Coras577c3552016-04-21 00:45:40 +0200243VLIB_CLI_COMMAND (enable_disable_lisp_gpe_command, static) = {
Filip Tehlar82786c42017-02-20 15:20:37 +0100244 .path = "gpe",
245 .short_help = "gpe [enable|disable]",
Florin Coras577c3552016-04-21 00:45:40 +0200246 .function = lisp_gpe_enable_disable_command_fn,
Florin Corase127a7e2016-02-18 22:20:01 +0100247};
Florin Coras220beac2016-08-16 23:04:00 +0200248/* *INDENT-ON* */
Florin Corase127a7e2016-02-18 22:20:01 +0100249
Florin Corasff0bf132016-09-05 19:30:35 +0200250/** CLI command to show LISP-GPE interfaces. */
Filip Tehlar53f09e32016-05-19 14:25:44 +0200251static clib_error_t *
252lisp_show_iface_command_fn (vlib_main_t * vm,
Florin Coras220beac2016-08-16 23:04:00 +0200253 unformat_input_t * input,
254 vlib_cli_command_t * cmd)
Filip Tehlar53f09e32016-05-19 14:25:44 +0200255{
Florin Coras220beac2016-08-16 23:04:00 +0200256 lisp_gpe_main_t *lgm = &lisp_gpe_main;
257 hash_pair_t *p;
Filip Tehlar53f09e32016-05-19 14:25:44 +0200258
259 vlib_cli_output (vm, "%=10s%=12s", "vrf", "hw_if_index");
Florin Coras220beac2016-08-16 23:04:00 +0200260
261 /* *INDENT-OFF* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200262 hash_foreach_pair (p, lgm->l3_ifaces.hw_if_index_by_dp_table, ({
Filip Tehlar53f09e32016-05-19 14:25:44 +0200263 vlib_cli_output (vm, "%=10d%=10d", p->key, p->value[0]);
264 }));
Florin Coras220beac2016-08-16 23:04:00 +0200265 /* *INDENT-ON* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200266
267 if (0 != lgm->l2_ifaces.hw_if_index_by_dp_table)
268 {
269 vlib_cli_output (vm, "%=10s%=12s", "bd_id", "hw_if_index");
Florin Coras220beac2016-08-16 23:04:00 +0200270 /* *INDENT-OFF* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200271 hash_foreach_pair (p, lgm->l2_ifaces.hw_if_index_by_dp_table, ({
272 vlib_cli_output (vm, "%=10d%=10d", p->key, p->value[0]);
273 }));
Florin Coras220beac2016-08-16 23:04:00 +0200274 /* *INDENT-ON* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200275 }
Filip Tehlar53f09e32016-05-19 14:25:44 +0200276 return 0;
277}
278
Florin Coras220beac2016-08-16 23:04:00 +0200279/* *INDENT-OFF* */
Filip Tehlar53f09e32016-05-19 14:25:44 +0200280VLIB_CLI_COMMAND (lisp_show_iface_command) = {
Filip Tehlar82786c42017-02-20 15:20:37 +0100281 .path = "show gpe interface",
282 .short_help = "show gpe interface",
Filip Tehlar53f09e32016-05-19 14:25:44 +0200283 .function = lisp_show_iface_command_fn,
284};
Florin Coras220beac2016-08-16 23:04:00 +0200285/* *INDENT-ON* */
Florin Coras1a1adc72016-07-22 01:45:30 +0200286
Florin Corasff0bf132016-09-05 19:30:35 +0200287/** Format LISP-GPE status. */
Florin Coras1a1adc72016-07-22 01:45:30 +0200288u8 *
289format_vnet_lisp_gpe_status (u8 * s, va_list * args)
290{
Florin Coras220beac2016-08-16 23:04:00 +0200291 lisp_gpe_main_t *lgm = &lisp_gpe_main;
Florin Coras1a1adc72016-07-22 01:45:30 +0200292 return format (s, "%s", lgm->is_en ? "enabled" : "disabled");
293}
294
Florin Corasff0bf132016-09-05 19:30:35 +0200295/** LISP-GPE init function. */
Florin Corase127a7e2016-02-18 22:20:01 +0100296clib_error_t *
Florin Coras220beac2016-08-16 23:04:00 +0200297lisp_gpe_init (vlib_main_t * vm)
Florin Corase127a7e2016-02-18 22:20:01 +0100298{
Florin Coras220beac2016-08-16 23:04:00 +0200299 lisp_gpe_main_t *lgm = &lisp_gpe_main;
300 clib_error_t *error = 0;
Florin Corase127a7e2016-02-18 22:20:01 +0100301
302 if ((error = vlib_call_init_function (vm, ip_main_init)))
303 return error;
304
305 if ((error = vlib_call_init_function (vm, ip4_lookup_init)))
306 return error;
307
Florin Coras220beac2016-08-16 23:04:00 +0200308 lgm->vnet_main = vnet_get_main ();
Florin Corase127a7e2016-02-18 22:20:01 +0100309 lgm->vlib_main = vm;
310 lgm->im4 = &ip4_main;
Florin Coras02655bd2016-04-26 00:17:24 +0200311 lgm->im6 = &ip6_main;
312 lgm->lm4 = &ip4_main.lookup_main;
313 lgm->lm6 = &ip6_main.lookup_main;
Florin Coras577c3552016-04-21 00:45:40 +0200314
Neale Ranns5e575b12016-10-03 09:40:25 +0100315 lgm->lisp_gpe_fwd_entries =
316 hash_create_mem (0, sizeof (lisp_gpe_fwd_entry_key_t), sizeof (uword));
Florin Coras1a1adc72016-07-22 01:45:30 +0200317
Florin Coras220beac2016-08-16 23:04:00 +0200318 udp_register_dst_port (vm, UDP_DST_PORT_lisp_gpe,
319 lisp_gpe_ip4_input_node.index, 1 /* is_ip4 */ );
Florin Coras02655bd2016-04-26 00:17:24 +0200320 udp_register_dst_port (vm, UDP_DST_PORT_lisp_gpe6,
Florin Coras220beac2016-08-16 23:04:00 +0200321 lisp_gpe_ip6_input_node.index, 0 /* is_ip4 */ );
Ed Warnickecb9cada2015-12-08 15:45:58 -0700322 return 0;
323}
324
Florin Coras220beac2016-08-16 23:04:00 +0200325VLIB_INIT_FUNCTION (lisp_gpe_init);
326
327/*
328 * fd.io coding-style-patch-verification: ON
329 *
330 * Local Variables:
331 * eval: (c-set-style "gnu")
332 * End:
333 */