blob: 231c7e3108facee1c99330ee5dca17846553af3c [file] [log] [blame]
Ole Troan298c6952018-03-08 12:30:43 +01001/*
2 * Copyright (c) 2018 Cisco and/or its affiliates.
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 */
15
16#include "ipip.h"
17#include <vppinfra/error.h>
18#include <vnet/vnet.h>
Neale Ranns61502112018-08-22 00:21:14 -070019#include <vnet/fib/fib_table.h>
Ole Troan298c6952018-03-08 12:30:43 +010020
21static clib_error_t *create_ipip_tunnel_command_fn(vlib_main_t *vm,
22 unformat_input_t *input,
23 vlib_cli_command_t *cmd) {
24 unformat_input_t _line_input, *line_input = &_line_input;
25 ip46_address_t src = ip46_address_initializer, dst = ip46_address_initializer;
26 u32 instance = ~0;
27 u32 fib_index = 0;
Neale Ranns61502112018-08-22 00:21:14 -070028 u32 table_id = 0;
Ole Troan298c6952018-03-08 12:30:43 +010029 int rv;
30 u32 num_m_args = 0;
31 u32 sw_if_index;
32 clib_error_t *error = NULL;
33 bool ip4_set = false, ip6_set = false;
Neale Ranns14053c92019-12-29 23:55:18 +000034 tunnel_mode_t mode = TUNNEL_MODE_P2P;
Ole Troan298c6952018-03-08 12:30:43 +010035
36 /* Get a line of input. */
37 if (!unformat_user(input, unformat_line_input, line_input))
38 return 0;
39
40 while (unformat_check_input(line_input) != UNFORMAT_END_OF_INPUT) {
41 if (unformat(line_input, "instance %d", &instance))
42 ;
43 else if (unformat(line_input, "src %U", unformat_ip4_address, &src.ip4)) {
44 num_m_args++;
45 ip4_set = true;
46 } else if (unformat(line_input, "dst %U", unformat_ip4_address, &dst.ip4)) {
47 num_m_args++;
48 ip4_set = true;
49 } else if (unformat(line_input, "src %U", unformat_ip6_address, &src.ip6)) {
50 num_m_args++;
51 ip6_set = true;
52 } else if (unformat(line_input, "dst %U", unformat_ip6_address, &dst.ip6)) {
53 num_m_args++;
54 ip6_set = true;
Neale Ranns14053c92019-12-29 23:55:18 +000055 } else if (unformat(line_input, "%U", unformat_tunnel_mode, &mode)) {
56 num_m_args++;
Neale Ranns61502112018-08-22 00:21:14 -070057 } else if (unformat(line_input, "outer-table-id %d", &table_id))
Ole Troan298c6952018-03-08 12:30:43 +010058 ;
59 else {
60 error = clib_error_return(0, "unknown input `%U'", format_unformat_error,
61 line_input);
62 goto done;
63 }
64 }
65
66 if (num_m_args < 2) {
67 error = clib_error_return(0, "mandatory argument(s) missing");
68 goto done;
69 }
70 if (ip4_set && ip6_set) {
71 error = clib_error_return(0, "source and destination must be of same address family");
72 goto done;
73 }
74
Neale Ranns61502112018-08-22 00:21:14 -070075 fib_index = fib_table_find(fib_ip_proto(ip6_set), table_id);
Ole Troan298c6952018-03-08 12:30:43 +010076
Neale Ranns61502112018-08-22 00:21:14 -070077 if (~0 == fib_index)
78 {
79 rv = VNET_API_ERROR_NO_SUCH_FIB;
80 }
81 else
82 {
83 rv = ipip_add_tunnel(ip6_set ? IPIP_TRANSPORT_IP6 : IPIP_TRANSPORT_IP4,
84 instance,
85 &src,
86 &dst,
87 fib_index,
Neale Ranns59ff9182019-12-29 23:55:18 +000088 TUNNEL_ENCAP_DECAP_FLAG_NONE,
Neale Ranns95346962019-11-25 13:04:44 +000089 IP_DSCP_CS0,
Neale Ranns14053c92019-12-29 23:55:18 +000090 mode,
Neale Ranns61502112018-08-22 00:21:14 -070091 &sw_if_index);
92 }
93
94 switch (rv) {
Ole Troan298c6952018-03-08 12:30:43 +010095 case 0:
96 vlib_cli_output(vm, "%U\n", format_vnet_sw_if_index_name, vnet_get_main(),
97 sw_if_index);
98 break;
99 case VNET_API_ERROR_IF_ALREADY_EXISTS:
100 error = clib_error_return(0, "IPIP tunnel already exists...");
101 goto done;
102 case VNET_API_ERROR_NO_SUCH_FIB:
103 error = clib_error_return(0, "outer fib ID %d doesn't exist\n", fib_index);
104 goto done;
105 case VNET_API_ERROR_NO_SUCH_ENTRY:
106 error = clib_error_return(0, "IPIP tunnel doesn't exist");
107 goto done;
108 case VNET_API_ERROR_INSTANCE_IN_USE:
109 error = clib_error_return(0, "Instance is in use");
110 goto done;
Neale Ranns14053c92019-12-29 23:55:18 +0000111 case VNET_API_ERROR_INVALID_DST_ADDRESS:
112 error = clib_error_return(0, "destination IP address when mode is multi-point");
113 goto done;
Ole Troan298c6952018-03-08 12:30:43 +0100114 default:
115 error = clib_error_return(0, "vnet_ipip_add_del_tunnel returned %d", rv);
116 goto done;
117 }
118
119done:
120 unformat_free(line_input);
121
122 return error;
123}
124
125static clib_error_t *delete_ipip_tunnel_command_fn(vlib_main_t *vm,
126 unformat_input_t *input,
127 vlib_cli_command_t *cmd) {
128 unformat_input_t _line_input, *line_input = &_line_input;
129 int rv;
130 u32 num_m_args = 0;
131 u32 sw_if_index = ~0;
132 clib_error_t *error = NULL;
133
134 /* Get a line of input. */
135 if (!unformat_user(input, unformat_line_input, line_input))
136 return 0;
137
138 while (unformat_check_input(line_input) != UNFORMAT_END_OF_INPUT) {
139 if (unformat(line_input, "sw_if_index %d", &sw_if_index))
140 num_m_args++;
141 else {
142 error = clib_error_return(0, "unknown input `%U'", format_unformat_error,
143 line_input);
144 goto done;
145 }
146 }
147
148 if (num_m_args < 1) {
149 error = clib_error_return(0, "mandatory argument(s) missing");
150 goto done;
151 }
152
153 rv = ipip_del_tunnel(sw_if_index);
154 printf("RV %d\n", rv);
155
156done:
157 unformat_free(line_input);
158
159 return error;
160}
161
162/* *INDENT-OFF* */
163VLIB_CLI_COMMAND(create_ipip_tunnel_command, static) = {
164 .path = "create ipip tunnel",
165 .short_help = "create ipip tunnel src <addr> dst <addr> [instance <n>] "
Neale Rannsae0d46e2020-09-22 13:20:27 +0000166 "[outer-table-id <ID>] [p2mp]",
Ole Troan298c6952018-03-08 12:30:43 +0100167 .function = create_ipip_tunnel_command_fn,
168};
169VLIB_CLI_COMMAND(delete_ipip_tunnel_command, static) = {
170 .path = "delete ipip tunnel",
Ignas Bacius3d93ad92019-10-10 16:14:47 +0300171 .short_help = "delete ipip tunnel sw_if_index <sw_if_index>",
Ole Troan298c6952018-03-08 12:30:43 +0100172 .function = delete_ipip_tunnel_command_fn,
173};
174/* *INDENT-ON* */
175
176static u8 *format_ipip_tunnel(u8 *s, va_list *args) {
177 ipip_tunnel_t *t = va_arg(*args, ipip_tunnel_t *);
178
179 ip46_type_t type = (t->transport == IPIP_TRANSPORT_IP4) ? IP46_TYPE_IP4 : IP46_TYPE_IP6;
Neale Ranns61502112018-08-22 00:21:14 -0700180 u32 table_id;
181
182 table_id = fib_table_get_table_id(t->fib_index,
183 fib_proto_from_ip46(type));
Ole Troan298c6952018-03-08 12:30:43 +0100184 switch (t->mode) {
185 case IPIP_MODE_6RD:
Neale Ranns95346962019-11-25 13:04:44 +0000186 s = format(s, "[%d] 6rd src %U ip6-pfx %U/%d ",
Ole Troan298c6952018-03-08 12:30:43 +0100187 t->dev_instance,
188 format_ip46_address, &t->tunnel_src, type,
Neale Ranns95346962019-11-25 13:04:44 +0000189 format_ip6_address, &t->sixrd.ip6_prefix, t->sixrd.ip6_prefix_len);
Ole Troan298c6952018-03-08 12:30:43 +0100190 break;
191 case IPIP_MODE_P2P:
Neale Ranns95346962019-11-25 13:04:44 +0000192 s = format(s, "[%d] instance %d src %U dst %U ",
Ole Troan298c6952018-03-08 12:30:43 +0100193 t->dev_instance, t->user_instance,
194 format_ip46_address, &t->tunnel_src, type,
Neale Ranns95346962019-11-25 13:04:44 +0000195 format_ip46_address, &t->tunnel_dst, type);
Ole Troan298c6952018-03-08 12:30:43 +0100196 break;
Neale Ranns14053c92019-12-29 23:55:18 +0000197 case IPIP_MODE_P2MP:
198 s = format(s, "[%d] instance %d p2mp src %U ",
199 t->dev_instance, t->user_instance,
200 format_ip46_address, &t->tunnel_src, type);
201 break;
Ole Troan298c6952018-03-08 12:30:43 +0100202 }
203
Neale Ranns95346962019-11-25 13:04:44 +0000204 s = format(s, "table-ID %d sw-if-idx %d flags [%U] dscp %U",
205 table_id, t->sw_if_index,
Neale Ranns59ff9182019-12-29 23:55:18 +0000206 format_tunnel_encap_decap_flags, t->flags,
Neale Ranns95346962019-11-25 13:04:44 +0000207 format_ip_dscp, t->dscp);
208
Ole Troan298c6952018-03-08 12:30:43 +0100209 return s;
210}
211
212static clib_error_t *show_ipip_tunnel_command_fn(vlib_main_t *vm,
213 unformat_input_t *input,
214 vlib_cli_command_t *cmd) {
215 ipip_main_t *gm = &ipip_main;
216 ipip_tunnel_t *t;
217 u32 ti = ~0;
218
219 if (pool_elts(gm->tunnels) == 0)
220 vlib_cli_output(vm, "No IPIP tunnels configured...");
221
222 while (unformat_check_input(input) != UNFORMAT_END_OF_INPUT) {
223 if (unformat(input, "%d", &ti))
224 ;
225 else
226 break;
227 }
228
229 if (ti == ~0) {
230 /* *INDENT-OFF* */
231 pool_foreach(t, gm->tunnels,
232 ({vlib_cli_output(vm, "%U", format_ipip_tunnel, t); }));
233 /* *INDENT-ON* */
234 } else {
Neale Ranns8b276df2020-11-09 10:13:24 +0000235 if (pool_is_free_index (gm->tunnels, ti))
236 return clib_error_return(0, "unknown index:%d", ti);
Ole Troan298c6952018-03-08 12:30:43 +0100237 t = pool_elt_at_index(gm->tunnels, ti);
238 if (t)
239 vlib_cli_output(vm, "%U", format_ipip_tunnel, t);
240 }
241 return 0;
242}
243
244/* *INDENT-OFF* */
245VLIB_CLI_COMMAND(show_ipip_tunnel_command, static) = {
246 .path = "show ipip tunnel",
247 .function = show_ipip_tunnel_command_fn,
248};
249/* *INDENT-ON* */
250
Neale Ranns14053c92019-12-29 23:55:18 +0000251static u8 *
252format_ipip_tunnel_key (u8 *s, va_list *args)
253{
254 ipip_tunnel_key_t *t = va_arg(*args, ipip_tunnel_key_t *);
255
256 s = format (s, "src:%U dst:%U fib:%d transport:%d mode:%d",
257 format_ip46_address, &t->src, IP46_TYPE_ANY,
258 format_ip46_address, &t->dst, IP46_TYPE_ANY,
259 t->fib_index, t->transport, t->mode);
260
261 return (s);
262}
263
264static clib_error_t *
265ipip_tunnel_hash_show (vlib_main_t * vm,
266 unformat_input_t * input,
267 vlib_cli_command_t * cmd)
268{
269 ipip_main_t *im = &ipip_main;
270 ipip_tunnel_key_t *key;
271 u32 index;
272
273 /* *INDENT-OFF* */
274 hash_foreach(key, index, im->tunnel_by_key,
275 ({
276 vlib_cli_output (vm, " %U -> %d", format_ipip_tunnel_key, key, index);
277 }));
278 /* *INDENT-ON* */
279
280 return NULL;
281}
282
283/**
284 * show IPSEC tunnel protection hash tables
285 */
286/* *INDENT-OFF* */
287VLIB_CLI_COMMAND (ipip_tunnel_hash_show_node, static) =
288{
289 .path = "show ipip tunnel-hash",
290 .function = ipip_tunnel_hash_show,
291 .short_help = "show ipip tunnel-hash",
292};
293/* *INDENT-ON* */
294
Ole Troan298c6952018-03-08 12:30:43 +0100295static clib_error_t *create_sixrd_tunnel_command_fn(vlib_main_t *vm,
296 unformat_input_t *input,
297 vlib_cli_command_t *cmd) {
298 unformat_input_t _line_input, *line_input = &_line_input;
299 ip4_address_t ip4_prefix;
300 ip6_address_t ip6_prefix;
301 ip4_address_t ip4_src;
302 u32 ip6_prefix_len = 0, ip4_prefix_len = 0, sixrd_tunnel_index;
303 u32 num_m_args = 0;
304 /* Optional arguments */
Neale Ranns61502112018-08-22 00:21:14 -0700305 u32 ip4_table_id = 0, ip4_fib_index;
306 u32 ip6_table_id = 0, ip6_fib_index;
Ole Troan298c6952018-03-08 12:30:43 +0100307 clib_error_t *error = 0;
308 bool security_check = false;
Neale Ranns61502112018-08-22 00:21:14 -0700309 int rv;
Ole Troan298c6952018-03-08 12:30:43 +0100310
311 /* Get a line of input. */
312 if (!unformat_user(input, unformat_line_input, line_input))
313 return 0;
314 while (unformat_check_input(line_input) != UNFORMAT_END_OF_INPUT) {
315 if (unformat(line_input, "security-check"))
316 security_check = true;
317 else if (unformat(line_input, "ip6-pfx %U/%d", unformat_ip6_address,
318 &ip6_prefix, &ip6_prefix_len))
319 num_m_args++;
320 else if (unformat(line_input, "ip4-pfx %U/%d", unformat_ip4_address,
321 &ip4_prefix, &ip4_prefix_len))
322 num_m_args++;
323 else if (unformat(line_input, "ip4-src %U", unformat_ip4_address, &ip4_src))
324 num_m_args++;
Neale Ranns61502112018-08-22 00:21:14 -0700325 else if (unformat(line_input, "ip4-table-id %d", &ip4_table_id))
326 ;
327 else if (unformat(line_input, "ip6-table-id %d", &ip6_table_id))
Ole Troan298c6952018-03-08 12:30:43 +0100328 ;
329 else {
330 error = clib_error_return(0, "unknown input `%U'", format_unformat_error,
331 line_input);
332 goto done;
333 }
334 }
335
336 if (num_m_args < 3) {
337 error = clib_error_return(0, "mandatory argument(s) missing");
338 goto done;
339 }
Neale Ranns61502112018-08-22 00:21:14 -0700340 ip4_fib_index = fib_table_find(FIB_PROTOCOL_IP4, ip4_table_id);
341 ip6_fib_index = fib_table_find(FIB_PROTOCOL_IP6, ip6_table_id);
Ole Troan298c6952018-03-08 12:30:43 +0100342
Neale Ranns61502112018-08-22 00:21:14 -0700343 if (~0 == ip4_fib_index)
344 {
345 error = clib_error_return(0, "No such IP4 table %d", ip4_table_id);
346 rv = VNET_API_ERROR_NO_SUCH_FIB;
347 }
348 else if (~0 == ip6_fib_index)
349 {
350 error = clib_error_return(0, "No such IP6 table %d", ip6_table_id);
351 rv = VNET_API_ERROR_NO_SUCH_FIB;
352 }
353 else
354 {
355 rv = sixrd_add_tunnel(&ip6_prefix, ip6_prefix_len, &ip4_prefix,
356 ip4_prefix_len, &ip4_src, security_check,
357 ip4_fib_index, ip6_fib_index,
358 &sixrd_tunnel_index);
359
360 if (rv)
361 error = clib_error_return(0, "adding tunnel failed %d", rv);
362 }
363
364done:
Ole Troan298c6952018-03-08 12:30:43 +0100365 unformat_free(line_input);
366
367 return error;
368}
369
370static clib_error_t *delete_sixrd_tunnel_command_fn(vlib_main_t *vm,
371 unformat_input_t *input,
372 vlib_cli_command_t *cmd) {
373 unformat_input_t _line_input, *line_input = &_line_input;
374 u32 num_m_args = 0;
375 /* Optional arguments */
376 clib_error_t *error = 0;
377 u32 sw_if_index = ~0;
378
379 /* Get a line of input. */
380 if (!unformat_user(input, unformat_line_input, line_input))
381 return 0;
382 while (unformat_check_input(line_input) != UNFORMAT_END_OF_INPUT) {
383 if (unformat(line_input, "sw_if_index %d", &sw_if_index))
384 num_m_args++;
385 else {
386 error = clib_error_return(0, "unknown input `%U'", format_unformat_error,
387 line_input);
388 goto done;
389 }
390 }
391
392 if (num_m_args < 1) {
393 error = clib_error_return(0, "mandatory argument(s) missing");
394 goto done;
395 }
396 int rv = sixrd_del_tunnel(sw_if_index);
397 printf("RV %d\n", rv);
398
399done:
400 unformat_free(line_input);
401
402 return error;
403}
404
405/* *INDENT-OFF* */
406VLIB_CLI_COMMAND(create_sixrd_tunnel_command, static) = {
407 .path = "create 6rd tunnel",
408 .short_help = "create 6rd tunnel ip6-pfx <ip6-pfx> ip4-pfx <ip4-pfx> "
Ignas Bacius3d93ad92019-10-10 16:14:47 +0300409 "ip4-src <ip4-addr> ip4-table-id <ID> ip6-table-id <ID> "
BenoƮt Ganne1b52ca92019-04-19 10:12:42 +0200410 "[security-check]",
Ole Troan298c6952018-03-08 12:30:43 +0100411 .function = create_sixrd_tunnel_command_fn,
412};
413VLIB_CLI_COMMAND(delete_sixrd_tunnel_command, static) = {
414 .path = "delete 6rd tunnel",
Ignas Bacius3d93ad92019-10-10 16:14:47 +0300415 .short_help = "delete 6rd tunnel sw_if_index <sw_if_index>",
Ole Troan298c6952018-03-08 12:30:43 +0100416 .function = delete_sixrd_tunnel_command_fn,
417};
418/* *INDENT-ON* */