blob: 9edd8b6ba57964f5bb29e420cf507c4494ca5500 [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 * Copyright (c) 2015 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#include <vnet/l2/l2_xcrw.h>
16
Chris Luke16bcf7d2016-09-01 14:31:46 -040017/**
18 * @file
Ed Warnickecb9cada2015-12-08 15:45:58 -070019 * General L2 / L3 cross-connect, used to set up
20 * "L2 interface <--> your-favorite-tunnel-encap" tunnels.
21 *
22 * We set up a typical L2 cross-connect or (future) bridge
23 * to hook L2 interface(s) up to the L3 stack in arbitrary ways.
24 *
25 * Each l2_xcrw adjacency specifies 3 things:
Dave Barach97d8dc22016-08-15 15:31:15 -040026 *
Ed Warnickecb9cada2015-12-08 15:45:58 -070027 * 1. The next graph node (presumably in the L3 stack) to
28 * process the (L2 -> L3) packet
29 *
Dave Barach97d8dc22016-08-15 15:31:15 -040030 * 2. A new value for vnet_buffer(b)->sw_if_index[VLIB_TX]
Ed Warnickecb9cada2015-12-08 15:45:58 -070031 * (i.e. a lookup FIB index),
32 *
33 * 3. A rewrite string to apply.
34 *
35 * Example: to cross-connect an L2 interface or (future) bridge
Dave Barach97d8dc22016-08-15 15:31:15 -040036 * to an mpls-o-gre tunnel, set up the L2 rewrite string as shown in
Ed Warnickecb9cada2015-12-08 15:45:58 -070037 * mpls_gre_rewrite, and use "mpls-post-rewrite" to fix the
38 * GRE IP header checksum and length fields.
39 */
40
Dave Barach97d8dc22016-08-15 15:31:15 -040041typedef struct
42{
Ed Warnickecb9cada2015-12-08 15:45:58 -070043 u32 next_index;
44 u32 tx_fib_index;
45} l2_xcrw_trace_t;
46
47/* packet trace format function */
Dave Barach97d8dc22016-08-15 15:31:15 -040048static u8 *
49format_l2_xcrw_trace (u8 * s, va_list * args)
Ed Warnickecb9cada2015-12-08 15:45:58 -070050{
51 CLIB_UNUSED (vlib_main_t * vm) = va_arg (*args, vlib_main_t *);
52 CLIB_UNUSED (vlib_node_t * node) = va_arg (*args, vlib_node_t *);
Dave Barach97d8dc22016-08-15 15:31:15 -040053 l2_xcrw_trace_t *t = va_arg (*args, l2_xcrw_trace_t *);
54
Ed Warnickecb9cada2015-12-08 15:45:58 -070055 s = format (s, "L2_XCRW: next index %d tx_fib_index %d",
Dave Barach97d8dc22016-08-15 15:31:15 -040056 t->next_index, t->tx_fib_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -070057 return s;
58}
59
Filip Tehlar44f0f712019-03-11 04:26:37 -070060extern l2_xcrw_main_t l2_xcrw_main;
Ed Warnickecb9cada2015-12-08 15:45:58 -070061
Filip Tehlar44f0f712019-03-11 04:26:37 -070062#ifndef CLIB_MARCH_VARIANT
63l2_xcrw_main_t l2_xcrw_main;
64#endif /* CLIB_MARCH_VARIANT */
Ed Warnickecb9cada2015-12-08 15:45:58 -070065
Dave Barach97d8dc22016-08-15 15:31:15 -040066static char *l2_xcrw_error_strings[] = {
Ed Warnickecb9cada2015-12-08 15:45:58 -070067#define _(sym,string) string,
68 foreach_l2_xcrw_error
69#undef _
70};
71
Filip Tehlar44f0f712019-03-11 04:26:37 -070072VLIB_NODE_FN (l2_xcrw_node) (vlib_main_t * vm,
73 vlib_node_runtime_t * node, vlib_frame_t * frame)
Ed Warnickecb9cada2015-12-08 15:45:58 -070074{
Dave Barach97d8dc22016-08-15 15:31:15 -040075 u32 n_left_from, *from, *to_next;
Ed Warnickecb9cada2015-12-08 15:45:58 -070076 l2_xcrw_next_t next_index;
Dave Barach97d8dc22016-08-15 15:31:15 -040077 l2_xcrw_main_t *xcm = &l2_xcrw_main;
Ed Warnickecb9cada2015-12-08 15:45:58 -070078 vlib_node_t *n = vlib_get_node (vm, l2_xcrw_node.index);
79 u32 node_counter_base_index = n->error_heap_index;
Dave Barach97d8dc22016-08-15 15:31:15 -040080 vlib_error_main_t *em = &vm->error_main;
Ed Warnickecb9cada2015-12-08 15:45:58 -070081
82 from = vlib_frame_vector_args (frame);
83 n_left_from = frame->n_vectors;
84 next_index = node->cached_next_index;
85
86 while (n_left_from > 0)
87 {
88 u32 n_left_to_next;
89
Dave Barach97d8dc22016-08-15 15:31:15 -040090 vlib_get_next_frame (vm, node, next_index, to_next, n_left_to_next);
Ed Warnickecb9cada2015-12-08 15:45:58 -070091
92 while (n_left_from >= 4 && n_left_to_next >= 2)
93 {
Dave Barach97d8dc22016-08-15 15:31:15 -040094 u32 bi0, bi1;
95 vlib_buffer_t *b0, *b1;
96 u32 next0, next1;
97 u32 sw_if_index0, sw_if_index1;
98 l2_xcrw_adjacency_t *adj0, *adj1;
99
Ed Warnickecb9cada2015-12-08 15:45:58 -0700100 /* Prefetch next iteration. */
101 {
Dave Barach97d8dc22016-08-15 15:31:15 -0400102 vlib_buffer_t *p2, *p3;
103
Ed Warnickecb9cada2015-12-08 15:45:58 -0700104 p2 = vlib_get_buffer (vm, from[2]);
105 p3 = vlib_get_buffer (vm, from[3]);
Dave Barach97d8dc22016-08-15 15:31:15 -0400106
Ed Warnickecb9cada2015-12-08 15:45:58 -0700107 vlib_prefetch_buffer_header (p2, LOAD);
108 vlib_prefetch_buffer_header (p3, LOAD);
109
Damjan Marionaf7fb042021-07-15 11:54:41 +0200110 clib_prefetch_store (p2->data);
111 clib_prefetch_store (p3->data);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700112 }
113
Dave Barach97d8dc22016-08-15 15:31:15 -0400114 /* speculatively enqueue b0 and b1 to the current next frame */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700115 to_next[0] = bi0 = from[0];
116 to_next[1] = bi1 = from[1];
117 from += 2;
118 to_next += 2;
119 n_left_from -= 2;
120 n_left_to_next -= 2;
121
122 b0 = vlib_get_buffer (vm, bi0);
123 b1 = vlib_get_buffer (vm, bi1);
124
Dave Barach97d8dc22016-08-15 15:31:15 -0400125 sw_if_index0 = vnet_buffer (b0)->sw_if_index[VLIB_RX];
126 sw_if_index1 = vnet_buffer (b1)->sw_if_index[VLIB_RX];
Ed Warnickecb9cada2015-12-08 15:45:58 -0700127
Dave Barach97d8dc22016-08-15 15:31:15 -0400128 adj0 = vec_elt_at_index (xcm->adj_by_sw_if_index, sw_if_index0);
129 adj1 = vec_elt_at_index (xcm->adj_by_sw_if_index, sw_if_index1);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700130
Dave Barach97d8dc22016-08-15 15:31:15 -0400131 next0 = adj0->rewrite_header.next_index;
132 vnet_buffer (b0)->sw_if_index[VLIB_TX] =
133 adj0->rewrite_header.sw_if_index;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700134
Dave Barach97d8dc22016-08-15 15:31:15 -0400135 next1 = adj1->rewrite_header.next_index;
136 vnet_buffer (b1)->sw_if_index[VLIB_TX] =
137 adj1->rewrite_header.sw_if_index;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700138
Dave Barach97d8dc22016-08-15 15:31:15 -0400139 em->counters[node_counter_base_index + next1]++;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700140
Dave Barach97d8dc22016-08-15 15:31:15 -0400141 if (PREDICT_TRUE (next0 > 0))
142 {
143 u8 *h0 = vlib_buffer_get_current (b0);
144 vnet_rewrite_one_header (adj0[0], h0,
145 adj0->rewrite_header.data_bytes);
146 vlib_buffer_advance (b0, -adj0->rewrite_header.data_bytes);
147 em->counters[node_counter_base_index + L2_XCRW_ERROR_FWD]++;
148 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700149
Dave Barach97d8dc22016-08-15 15:31:15 -0400150 if (PREDICT_TRUE (next1 > 0))
151 {
152 u8 *h1 = vlib_buffer_get_current (b1);
153 vnet_rewrite_one_header (adj1[0], h1,
154 adj1->rewrite_header.data_bytes);
155 vlib_buffer_advance (b1, -adj1->rewrite_header.data_bytes);
156 em->counters[node_counter_base_index + L2_XCRW_ERROR_FWD]++;
157 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700158
159
Dave Barach97d8dc22016-08-15 15:31:15 -0400160 if (PREDICT_FALSE ((node->flags & VLIB_NODE_FLAG_TRACE)))
161 {
162 if (PREDICT_FALSE ((node->flags & VLIB_NODE_FLAG_TRACE)
163 && (b0->flags & VLIB_BUFFER_IS_TRACED)))
164 {
165 l2_xcrw_trace_t *t =
166 vlib_add_trace (vm, node, b0, sizeof (*t));
167 t->next_index = next0;
168 t->tx_fib_index = adj0->rewrite_header.sw_if_index;
169 }
170 if (PREDICT_FALSE ((node->flags & VLIB_NODE_FLAG_TRACE)
171 && (b1->flags & VLIB_BUFFER_IS_TRACED)))
172 {
173 l2_xcrw_trace_t *t =
174 vlib_add_trace (vm, node, b1, sizeof (*t));
175 t->next_index = next1;
176 t->tx_fib_index = adj1->rewrite_header.sw_if_index;
177 }
178 }
179
180 /* verify speculative enqueues, maybe switch current next frame */
181 vlib_validate_buffer_enqueue_x2 (vm, node, next_index,
182 to_next, n_left_to_next,
183 bi0, bi1, next0, next1);
184 }
185
Ed Warnickecb9cada2015-12-08 15:45:58 -0700186 while (n_left_from > 0 && n_left_to_next > 0)
187 {
Dave Barach97d8dc22016-08-15 15:31:15 -0400188 u32 bi0;
189 vlib_buffer_t *b0;
190 u32 next0;
191 u32 sw_if_index0;
192 l2_xcrw_adjacency_t *adj0;
193
194 /* speculatively enqueue b0 to the current next frame */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700195 bi0 = from[0];
196 to_next[0] = bi0;
197 from += 1;
198 to_next += 1;
199 n_left_from -= 1;
200 n_left_to_next -= 1;
201
202 b0 = vlib_get_buffer (vm, bi0);
203
Dave Barach97d8dc22016-08-15 15:31:15 -0400204 sw_if_index0 = vnet_buffer (b0)->sw_if_index[VLIB_RX];
Ed Warnickecb9cada2015-12-08 15:45:58 -0700205
Dave Barach97d8dc22016-08-15 15:31:15 -0400206 adj0 = vec_elt_at_index (xcm->adj_by_sw_if_index, sw_if_index0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700207
Dave Barach97d8dc22016-08-15 15:31:15 -0400208 next0 = adj0->rewrite_header.next_index;
209 vnet_buffer (b0)->sw_if_index[VLIB_TX] =
210 adj0->rewrite_header.sw_if_index;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700211
Dave Barach97d8dc22016-08-15 15:31:15 -0400212 if (PREDICT_TRUE (next0 > 0))
213 {
214 u8 *h0 = vlib_buffer_get_current (b0);
215 vnet_rewrite_one_header (adj0[0], h0,
216 adj0->rewrite_header.data_bytes);
217 vlib_buffer_advance (b0, -adj0->rewrite_header.data_bytes);
218 em->counters[node_counter_base_index + L2_XCRW_ERROR_FWD]++;
219 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700220
Dave Barach97d8dc22016-08-15 15:31:15 -0400221 if (PREDICT_FALSE ((node->flags & VLIB_NODE_FLAG_TRACE)
222 && (b0->flags & VLIB_BUFFER_IS_TRACED)))
223 {
224 l2_xcrw_trace_t *t = vlib_add_trace (vm, node, b0, sizeof (*t));
225 t->next_index = next0;
226 t->tx_fib_index = adj0->rewrite_header.sw_if_index;
227 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700228
Dave Barach97d8dc22016-08-15 15:31:15 -0400229 /* verify speculative enqueue, maybe switch current next frame */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700230 vlib_validate_buffer_enqueue_x1 (vm, node, next_index,
231 to_next, n_left_to_next,
232 bi0, next0);
233 }
234
235 vlib_put_next_frame (vm, node, next_index, n_left_to_next);
236 }
237
238 return frame->n_vectors;
239}
240
Filip Tehlar44f0f712019-03-11 04:26:37 -0700241VLIB_REGISTER_NODE (l2_xcrw_node) = {
Ed Warnickecb9cada2015-12-08 15:45:58 -0700242 .name = "l2-xcrw",
243 .vector_size = sizeof (u32),
244 .format_trace = format_l2_xcrw_trace,
245 .type = VLIB_NODE_TYPE_INTERNAL,
Dave Barach97d8dc22016-08-15 15:31:15 -0400246
Ed Warnickecb9cada2015-12-08 15:45:58 -0700247 .n_errors = ARRAY_LEN(l2_xcrw_error_strings),
248 .error_strings = l2_xcrw_error_strings,
249
250 .n_next_nodes = L2_XCRW_N_NEXT,
251
252 /* edit / add dispositions here */
253 .next_nodes = {
254 [L2_XCRW_NEXT_DROP] = "error-drop",
255 },
256};
257
Filip Tehlar44f0f712019-03-11 04:26:37 -0700258#ifndef CLIB_MARCH_VARIANT
259clib_error_t *
260l2_xcrw_init (vlib_main_t * vm)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700261{
Dave Barach97d8dc22016-08-15 15:31:15 -0400262 l2_xcrw_main_t *mp = &l2_xcrw_main;
263
Ed Warnickecb9cada2015-12-08 15:45:58 -0700264 mp->vlib_main = vm;
265 mp->vnet_main = &vnet_main;
Dave Barach97d8dc22016-08-15 15:31:15 -0400266 mp->tunnel_index_by_l2_sw_if_index = hash_create (0, sizeof (uword));
Ed Warnickecb9cada2015-12-08 15:45:58 -0700267
268 return 0;
269}
270
271VLIB_INIT_FUNCTION (l2_xcrw_init);
272
Dave Barach97d8dc22016-08-15 15:31:15 -0400273static u8 *
274format_xcrw_name (u8 * s, va_list * args)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700275{
276 u32 dev_instance = va_arg (*args, u32);
277 return format (s, "xcrw%d", dev_instance);
278}
279
280VNET_DEVICE_CLASS (xcrw_device_class,static) = {
281 .name = "Xcrw",
282 .format_device_name = format_xcrw_name,
Ed Warnickecb9cada2015-12-08 15:45:58 -0700283};
284
285/* Create a sham tunnel interface and return its sw_if_index */
286static u32
287create_xcrw_interface (vlib_main_t * vm)
288{
Dave Barach97d8dc22016-08-15 15:31:15 -0400289 vnet_main_t *vnm = vnet_get_main ();
Damjan Marion5c954c42022-01-06 20:36:14 +0100290 vnet_eth_interface_registration_t eir = {};
Ed Warnickecb9cada2015-12-08 15:45:58 -0700291 static u32 instance;
292 u8 address[6];
293 u32 hw_if_index;
Dave Barach97d8dc22016-08-15 15:31:15 -0400294 vnet_hw_interface_t *hi;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700295 u32 sw_if_index;
296
297 /* mac address doesn't really matter */
Dave Barachb7b92992018-10-17 10:38:51 -0400298 clib_memset (address, 0, sizeof (address));
Ed Warnickecb9cada2015-12-08 15:45:58 -0700299 address[2] = 0x12;
300
Damjan Marion5c954c42022-01-06 20:36:14 +0100301 eir.dev_class_index = xcrw_device_class.index;
302 eir.dev_instance = instance++, eir.address = address;
303 hw_if_index = vnet_eth_register_interface (vnm, &eir);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700304
305 hi = vnet_get_hw_interface (vnm, hw_if_index);
306 sw_if_index = hi->sw_if_index;
Dave Barach97d8dc22016-08-15 15:31:15 -0400307 vnet_sw_interface_set_flags (vnm, sw_if_index,
308 VNET_SW_INTERFACE_FLAG_ADMIN_UP);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700309
310 /* Output to the sham tunnel invokes the encap node */
311 hi->output_node_index = l2_xcrw_node.index;
312
313 return sw_if_index;
314}
315
Dave Barach97d8dc22016-08-15 15:31:15 -0400316int
317vnet_configure_l2_xcrw (vlib_main_t * vm, vnet_main_t * vnm,
318 u32 l2_sw_if_index, u32 tx_fib_index,
319 u8 * rewrite, u32 next_node_index, int is_add)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700320{
Dave Barach97d8dc22016-08-15 15:31:15 -0400321 l2_xcrw_main_t *xcm = &l2_xcrw_main;
322 l2_xcrw_adjacency_t *a;
323 l2_xcrw_tunnel_t *t;
324 uword *p;
325
Ed Warnickecb9cada2015-12-08 15:45:58 -0700326 if (is_add)
327 {
328
329 pool_get (xcm->tunnels, t);
330
331 /* No interface allocated? Do it. Otherwise, set admin up */
332 if (t->tunnel_sw_if_index == 0)
Dave Barach97d8dc22016-08-15 15:31:15 -0400333 t->tunnel_sw_if_index = create_xcrw_interface (vm);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700334 else
Dave Barach97d8dc22016-08-15 15:31:15 -0400335 vnet_sw_interface_set_flags (vnm, t->tunnel_sw_if_index,
336 VNET_SW_INTERFACE_FLAG_ADMIN_UP);
337
Ed Warnickecb9cada2015-12-08 15:45:58 -0700338 t->l2_sw_if_index = l2_sw_if_index;
339
340 vec_validate (xcm->adj_by_sw_if_index, t->l2_sw_if_index);
Dave Barach97d8dc22016-08-15 15:31:15 -0400341
Ed Warnickecb9cada2015-12-08 15:45:58 -0700342 a = vec_elt_at_index (xcm->adj_by_sw_if_index, t->l2_sw_if_index);
Dave Barachb7b92992018-10-17 10:38:51 -0400343 clib_memset (a, 0, sizeof (*a));
Ed Warnickecb9cada2015-12-08 15:45:58 -0700344
345 a->rewrite_header.sw_if_index = tx_fib_index;
346
Dave Barach97d8dc22016-08-15 15:31:15 -0400347 /*
348 * Add or find a dynamic disposition for the successor node,
Ed Warnickecb9cada2015-12-08 15:45:58 -0700349 * e.g. so we can ship pkts to mpls_post_rewrite...
350 */
Dave Barach97d8dc22016-08-15 15:31:15 -0400351 a->rewrite_header.next_index =
352 vlib_node_add_next (vm, l2_xcrw_node.index, next_node_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700353
354 if (vec_len (rewrite))
Dave Barach97d8dc22016-08-15 15:31:15 -0400355 vnet_rewrite_set_data (a[0], rewrite, vec_len (rewrite));
356
Neale Rannsb4743802018-09-05 09:13:57 -0700357 set_int_l2_mode (vm, vnm, MODE_L2_XC, t->l2_sw_if_index, 0,
358 L2_BD_PORT_TYPE_NORMAL, 0, t->tunnel_sw_if_index);
Dave Barach97d8dc22016-08-15 15:31:15 -0400359 hash_set (xcm->tunnel_index_by_l2_sw_if_index,
360 t->l2_sw_if_index, t - xcm->tunnels);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700361 return 0;
362 }
363 else
364 {
365 p = hash_get (xcm->tunnel_index_by_l2_sw_if_index, l2_sw_if_index);
366 if (p == 0)
Dave Barach97d8dc22016-08-15 15:31:15 -0400367 return VNET_API_ERROR_INVALID_SW_IF_INDEX;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700368
369 t = pool_elt_at_index (xcm->tunnels, p[0]);
370
371 a = vec_elt_at_index (xcm->adj_by_sw_if_index, t->l2_sw_if_index);
372 /* Reset adj to drop traffic */
Dave Barachb7b92992018-10-17 10:38:51 -0400373 clib_memset (a, 0, sizeof (*a));
Ed Warnickecb9cada2015-12-08 15:45:58 -0700374
Neale Rannsb4743802018-09-05 09:13:57 -0700375 set_int_l2_mode (vm, vnm, MODE_L3, t->l2_sw_if_index, 0,
376 L2_BD_PORT_TYPE_NORMAL, 0, 0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700377
Dave Barach97d8dc22016-08-15 15:31:15 -0400378 vnet_sw_interface_set_flags (vnm, t->tunnel_sw_if_index, 0 /* down */ );
Ed Warnickecb9cada2015-12-08 15:45:58 -0700379
380 hash_unset (xcm->tunnel_index_by_l2_sw_if_index, l2_sw_if_index);
381 pool_put (xcm->tunnels, t);
382 }
383 return 0;
384}
385
386
387static clib_error_t *
388set_l2_xcrw_command_fn (vlib_main_t * vm,
Dave Barach97d8dc22016-08-15 15:31:15 -0400389 unformat_input_t * input, vlib_cli_command_t * cmd)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700390{
Dave Barach97d8dc22016-08-15 15:31:15 -0400391 unformat_input_t _line_input, *line_input = &_line_input;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700392 int is_add = 1;
Dave Barach97d8dc22016-08-15 15:31:15 -0400393 int is_ipv6 = 0; /* for fib id -> fib index mapping */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700394 u32 tx_fib_id = ~0;
395 u32 tx_fib_index = ~0;
396 u32 next_node_index = ~0;
397 u32 l2_sw_if_index;
Dave Barach97d8dc22016-08-15 15:31:15 -0400398 u8 *rw = 0;
399 vnet_main_t *vnm = vnet_get_main ();
Ed Warnickecb9cada2015-12-08 15:45:58 -0700400 int rv;
Billy McFalla9a20e72017-02-15 11:39:12 -0500401 clib_error_t *error = NULL;
Dave Barach97d8dc22016-08-15 15:31:15 -0400402
403
404 if (!unformat_user (input, unformat_line_input, line_input))
Ed Warnickecb9cada2015-12-08 15:45:58 -0700405 return 0;
Dave Barach97d8dc22016-08-15 15:31:15 -0400406
407 if (!unformat (line_input, "%U",
408 unformat_vnet_sw_interface, vnm, &l2_sw_if_index))
Billy McFalla9a20e72017-02-15 11:39:12 -0500409 {
410 error = clib_error_return (0, "unknown input '%U'",
411 format_unformat_error, line_input);
412 goto done;
413 }
Dave Barach97d8dc22016-08-15 15:31:15 -0400414
Ed Warnickecb9cada2015-12-08 15:45:58 -0700415 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
416 {
Dave Barach97d8dc22016-08-15 15:31:15 -0400417 if (unformat (line_input, "next %U",
418 unformat_vlib_node, vm, &next_node_index))
419 ;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700420 else if (unformat (line_input, "tx-fib-id %d", &tx_fib_id))
Dave Barach97d8dc22016-08-15 15:31:15 -0400421 ;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700422 else if (unformat (line_input, "del"))
Dave Barach97d8dc22016-08-15 15:31:15 -0400423 is_add = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700424 else if (unformat (line_input, "ipv6"))
Dave Barach97d8dc22016-08-15 15:31:15 -0400425 is_ipv6 = 1;
426 else if (unformat (line_input, "rw %U", unformat_hex_string, &rw));
Ed Warnickecb9cada2015-12-08 15:45:58 -0700427 else
Dave Barach97d8dc22016-08-15 15:31:15 -0400428 break;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700429 }
Dave Barach97d8dc22016-08-15 15:31:15 -0400430
Ed Warnickecb9cada2015-12-08 15:45:58 -0700431 if (next_node_index == ~0)
Billy McFalla9a20e72017-02-15 11:39:12 -0500432 {
433 error = clib_error_return (0, "next node not specified");
434 goto done;
435 }
Dave Barach97d8dc22016-08-15 15:31:15 -0400436
Ed Warnickecb9cada2015-12-08 15:45:58 -0700437 if (tx_fib_id != ~0)
438 {
Dave Barach97d8dc22016-08-15 15:31:15 -0400439 uword *p;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700440
441 if (is_ipv6)
Dave Barach97d8dc22016-08-15 15:31:15 -0400442 p = hash_get (ip6_main.fib_index_by_table_id, tx_fib_id);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700443 else
Dave Barach97d8dc22016-08-15 15:31:15 -0400444 p = hash_get (ip4_main.fib_index_by_table_id, tx_fib_id);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700445
446 if (p == 0)
Billy McFalla9a20e72017-02-15 11:39:12 -0500447 {
448 error =
449 clib_error_return (0, "nonexistent tx_fib_id %d", tx_fib_id);
450 goto done;
451 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700452
453 tx_fib_index = p[0];
454 }
455
456 rv = vnet_configure_l2_xcrw (vm, vnm, l2_sw_if_index, tx_fib_index,
Dave Barach97d8dc22016-08-15 15:31:15 -0400457 rw, next_node_index, is_add);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700458
459 switch (rv)
460 {
461
462 case 0:
463 break;
464
465 case VNET_API_ERROR_INVALID_SW_IF_INDEX:
Billy McFalla9a20e72017-02-15 11:39:12 -0500466 error = clib_error_return (0, "%U not cross-connected",
467 format_vnet_sw_if_index_name,
468 vnm, l2_sw_if_index);
469 goto done;
470
Ed Warnickecb9cada2015-12-08 15:45:58 -0700471 default:
Billy McFalla9a20e72017-02-15 11:39:12 -0500472 error = clib_error_return (0, "vnet_configure_l2_xcrw returned %d", rv);
473 goto done;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700474 }
Dave Barach97d8dc22016-08-15 15:31:15 -0400475
Billy McFalla9a20e72017-02-15 11:39:12 -0500476done:
Ed Warnickecb9cada2015-12-08 15:45:58 -0700477 vec_free (rw);
Billy McFalla9a20e72017-02-15 11:39:12 -0500478 unformat_free (line_input);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700479
Billy McFalla9a20e72017-02-15 11:39:12 -0500480 return error;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700481}
482
Billy McFall22aa3e92016-09-09 08:46:40 -0400483/*?
484 * Add or delete a Layer 2 to Layer 3 rewrite cross-connect. This is
485 * used to hook Layer 2 interface(s) up to the Layer 3 stack in
486 * arbitrary ways. For example, cross-connect an L2 interface or
487 * (future) bridge to an mpls-o-gre tunnel. Set up the L2 rewrite
488 * string as shown in mpls_gre_rewrite, and use \"mpls-post-rewrite\"
489 * to fix the GRE IP header checksum and length fields.
490 *
491 * @cliexpar
492 * @todo This is incomplete. This needs a detailed description and a
493 * practical example.
494?*/
Ed Warnickecb9cada2015-12-08 15:45:58 -0700495VLIB_CLI_COMMAND (set_l2_xcrw_command, static) = {
496 .path = "set interface l2 xcrw",
Dave Barach97d8dc22016-08-15 15:31:15 -0400497 .short_help =
Billy McFall22aa3e92016-09-09 08:46:40 -0400498 "set interface l2 xcrw <interface> next <node-name>\n"
Ed Warnickecb9cada2015-12-08 15:45:58 -0700499 " [del] [tx-fib-id <id>] [ipv6] rw <hex-bytes>",
500 .function = set_l2_xcrw_command_fn,
501};
502
Filip Tehlar44f0f712019-03-11 04:26:37 -0700503#endif /* CLIB_MARCH_VARIANT */
504
Dave Barach97d8dc22016-08-15 15:31:15 -0400505static u8 *
506format_l2xcrw (u8 * s, va_list * args)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700507{
Dave Barach97d8dc22016-08-15 15:31:15 -0400508 vnet_main_t *vnm = va_arg (*args, vnet_main_t *);
509 l2_xcrw_tunnel_t *t = va_arg (*args, l2_xcrw_tunnel_t *);
510 l2_xcrw_main_t *xcm = &l2_xcrw_main;
511 vlib_main_t *vm = vlib_get_main ();
512 l2_xcrw_adjacency_t *a;
513 u8 *rewrite_string;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700514
515 if (t == 0)
516 {
517 s = format (s, "%-25s%s", "L2 interface", "Tunnel Details");
518 return s;
519 }
520
Dave Barach97d8dc22016-08-15 15:31:15 -0400521 s = format (s, "%-25U %U ",
522 format_vnet_sw_if_index_name, vnm, t->l2_sw_if_index,
523 format_vnet_sw_if_index_name, vnm, t->tunnel_sw_if_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700524
525 a = vec_elt_at_index (xcm->adj_by_sw_if_index, t->l2_sw_if_index);
Dave Barach97d8dc22016-08-15 15:31:15 -0400526
Ed Warnickecb9cada2015-12-08 15:45:58 -0700527 s = format (s, "next %U ",
Dave Barach97d8dc22016-08-15 15:31:15 -0400528 format_vlib_next_node_name, vm, l2_xcrw_node.index,
529 a->rewrite_header.next_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700530
531 if (a->rewrite_header.sw_if_index != ~0)
532 s = format (s, "tx fib index %d ", a->rewrite_header.sw_if_index);
533
534 if (a->rewrite_header.data_bytes)
535 {
Dave Barach97d8dc22016-08-15 15:31:15 -0400536 rewrite_string = (u8 *) (a + 1);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700537 rewrite_string -= a->rewrite_header.data_bytes;
538 s = format (s, "rewrite data: %U ",
Dave Barach97d8dc22016-08-15 15:31:15 -0400539 format_hex_bytes, rewrite_string,
540 a->rewrite_header.data_bytes);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700541 }
542
543 s = format (s, "\n");
544
545 return s;
546}
547
548
549static clib_error_t *
550show_l2xcrw_command_fn (vlib_main_t * vm,
Dave Barach97d8dc22016-08-15 15:31:15 -0400551 unformat_input_t * input, vlib_cli_command_t * cmd)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700552{
Dave Barach97d8dc22016-08-15 15:31:15 -0400553 vnet_main_t *vnm = vnet_get_main ();
554 l2_xcrw_main_t *xcm = &l2_xcrw_main;
555 l2_xcrw_tunnel_t *t;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700556
557 if (pool_elts (xcm->tunnels) == 0)
558 {
559 vlib_cli_output (vm, "No L2 / L3 rewrite cross-connects configured");
560 return 0;
561 }
562
563 vlib_cli_output (vm, "%U", format_l2xcrw, 0, 0);
564
Damjan Marionb2c31b62020-12-13 21:47:40 +0100565 pool_foreach (t, xcm->tunnels)
566 {
Ed Warnickecb9cada2015-12-08 15:45:58 -0700567 vlib_cli_output (vm, "%U", format_l2xcrw, vnm, t);
Damjan Marionb2c31b62020-12-13 21:47:40 +0100568 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700569
570 return 0;
571}
572
Billy McFall22aa3e92016-09-09 08:46:40 -0400573/*?
574 * Display a Layer 2 to Layer 3 rewrite cross-connect. This is used to
575 * hook Layer 2 interface(s) up to the Layer 3 stack in arbitrary ways.
576 *
577 * @todo This is incomplete. This needs a detailed description and a
578 * practical example.
579?*/
Ed Warnickecb9cada2015-12-08 15:45:58 -0700580VLIB_CLI_COMMAND (show_l2xcrw_command, static) = {
581 .path = "show l2xcrw",
Billy McFall22aa3e92016-09-09 08:46:40 -0400582 .short_help = "show l2xcrw",
Ed Warnickecb9cada2015-12-08 15:45:58 -0700583 .function = show_l2xcrw_command_fn,
584};
Dave Barach97d8dc22016-08-15 15:31:15 -0400585
586/*
587 * fd.io coding-style-patch-verification: ON
588 *
589 * Local Variables:
590 * eval: (c-set-style "gnu")
591 * End:
592 */