FD.io VPP  v16.06
Vector Packet Processing
l2_xcrw.c
Go to the documentation of this file.
1 /*
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 
17 /*
18  * General L2 / L3 cross-connect, used to set up
19  * "L2 interface <--> your-favorite-tunnel-encap" tunnels.
20  *
21  * We set up a typical L2 cross-connect or (future) bridge
22  * to hook L2 interface(s) up to the L3 stack in arbitrary ways.
23  *
24  * Each l2_xcrw adjacency specifies 3 things:
25  *
26  * 1. The next graph node (presumably in the L3 stack) to
27  * process the (L2 -> L3) packet
28  *
29  * 2. A new value for vnet_buffer(b)->sw_if_index[VLIB_TX]
30  * (i.e. a lookup FIB index),
31  *
32  * 3. A rewrite string to apply.
33  *
34  * Example: to cross-connect an L2 interface or (future) bridge
35  * to an mpls-o-gre tunnel, set up the L2 rewrite string as shown in
36  * mpls_gre_rewrite, and use "mpls-post-rewrite" to fix the
37  * GRE IP header checksum and length fields.
38  */
39 
40 typedef struct {
44 
45 /* packet trace format function */
46 static u8 * format_l2_xcrw_trace (u8 * s, va_list * args)
47 {
48  CLIB_UNUSED (vlib_main_t * vm) = va_arg (*args, vlib_main_t *);
49  CLIB_UNUSED (vlib_node_t * node) = va_arg (*args, vlib_node_t *);
50  l2_xcrw_trace_t * t = va_arg (*args, l2_xcrw_trace_t *);
51 
52  s = format (s, "L2_XCRW: next index %d tx_fib_index %d",
53  t->next_index, t->tx_fib_index);
54  return s;
55 }
56 
58 
60 
61 static char * l2_xcrw_error_strings[] = {
62 #define _(sym,string) string,
64 #undef _
65 };
66 
67 static uword
69  vlib_node_runtime_t * node,
70  vlib_frame_t * frame)
71 {
72  u32 n_left_from, * from, * to_next;
73  l2_xcrw_next_t next_index;
75  vlib_node_t *n = vlib_get_node (vm, l2_xcrw_node.index);
76  u32 node_counter_base_index = n->error_heap_index;
77  vlib_error_main_t * em = &vm->error_main;
78 
79  from = vlib_frame_vector_args (frame);
80  n_left_from = frame->n_vectors;
81  next_index = node->cached_next_index;
82 
83  while (n_left_from > 0)
84  {
85  u32 n_left_to_next;
86 
87  vlib_get_next_frame (vm, node, next_index,
88  to_next, n_left_to_next);
89 
90  while (n_left_from >= 4 && n_left_to_next >= 2)
91  {
92  u32 bi0, bi1;
93  vlib_buffer_t * b0, * b1;
94  u32 next0, next1;
95  u32 sw_if_index0, sw_if_index1;
96  l2_xcrw_adjacency_t * adj0, * adj1;
97 
98  /* Prefetch next iteration. */
99  {
100  vlib_buffer_t * p2, * p3;
101 
102  p2 = vlib_get_buffer (vm, from[2]);
103  p3 = vlib_get_buffer (vm, from[3]);
104 
105  vlib_prefetch_buffer_header (p2, LOAD);
106  vlib_prefetch_buffer_header (p3, LOAD);
107 
110  }
111 
112  /* speculatively enqueue b0 and b1 to the current next frame */
113  to_next[0] = bi0 = from[0];
114  to_next[1] = bi1 = from[1];
115  from += 2;
116  to_next += 2;
117  n_left_from -= 2;
118  n_left_to_next -= 2;
119 
120  b0 = vlib_get_buffer (vm, bi0);
121  b1 = vlib_get_buffer (vm, bi1);
122 
123  sw_if_index0 = vnet_buffer(b0)->sw_if_index[VLIB_RX];
124  sw_if_index1 = vnet_buffer(b1)->sw_if_index[VLIB_RX];
125 
126  adj0 = vec_elt_at_index (xcm->adj_by_sw_if_index, sw_if_index0);
127  adj1 = vec_elt_at_index (xcm->adj_by_sw_if_index, sw_if_index1);
128 
129  next0 = adj0->rewrite_header.next_index;
130  vnet_buffer(b0)->sw_if_index[VLIB_TX] =
131  adj0->rewrite_header.sw_if_index;
132 
133  next1 = adj1->rewrite_header.next_index;
134  vnet_buffer(b1)->sw_if_index[VLIB_TX] =
135  adj1->rewrite_header.sw_if_index;
136 
137  em->counters[node_counter_base_index + next1]++;
138 
139  if (PREDICT_TRUE(next0 > 0))
140  {
141  u8 * h0 = vlib_buffer_get_current (b0);
142  vnet_rewrite_one_header (adj0[0], h0,
143  adj0->rewrite_header.data_bytes);
144  vlib_buffer_advance (b0, -adj0->rewrite_header.data_bytes);
145  em->counters[node_counter_base_index + L2_XCRW_ERROR_FWD]++;
146  }
147 
148  if (PREDICT_TRUE(next1 > 0))
149  {
150  u8 * h1 = vlib_buffer_get_current (b1);
151  vnet_rewrite_one_header (adj1[0], h1,
152  adj1->rewrite_header.data_bytes);
153  vlib_buffer_advance (b1, -adj1->rewrite_header.data_bytes);
154  em->counters[node_counter_base_index + L2_XCRW_ERROR_FWD]++;
155  }
156 
157 
159  {
161  && (b0->flags & VLIB_BUFFER_IS_TRACED)))
162  {
163  l2_xcrw_trace_t *t =
164  vlib_add_trace (vm, node, b0, sizeof (*t));
165  t->next_index = next0;
166  t->tx_fib_index = adj0->rewrite_header.sw_if_index;
167  }
169  && (b1->flags & VLIB_BUFFER_IS_TRACED)))
170  {
171  l2_xcrw_trace_t *t =
172  vlib_add_trace (vm, node, b1, sizeof (*t));
173  t->next_index = next1;
174  t->tx_fib_index = adj1->rewrite_header.sw_if_index;
175  }
176  }
177 
178  /* verify speculative enqueues, maybe switch current next frame */
179  vlib_validate_buffer_enqueue_x2 (vm, node, next_index,
180  to_next, n_left_to_next,
181  bi0, bi1, next0, next1);
182  }
183 
184  while (n_left_from > 0 && n_left_to_next > 0)
185  {
186  u32 bi0;
187  vlib_buffer_t * b0;
188  u32 next0;
189  u32 sw_if_index0;
190  l2_xcrw_adjacency_t * adj0;
191 
192  /* speculatively enqueue b0 to the current next frame */
193  bi0 = from[0];
194  to_next[0] = bi0;
195  from += 1;
196  to_next += 1;
197  n_left_from -= 1;
198  n_left_to_next -= 1;
199 
200  b0 = vlib_get_buffer (vm, bi0);
201 
202  sw_if_index0 = vnet_buffer(b0)->sw_if_index[VLIB_RX];
203 
204  adj0 = vec_elt_at_index (xcm->adj_by_sw_if_index, sw_if_index0);
205 
206  next0 = adj0->rewrite_header.next_index;
207  vnet_buffer(b0)->sw_if_index[VLIB_TX] =
208  adj0->rewrite_header.sw_if_index;
209 
210  if (PREDICT_TRUE(next0 > 0))
211  {
212  u8 *h0 = vlib_buffer_get_current (b0);
213  vnet_rewrite_one_header (adj0[0], h0,
214  adj0->rewrite_header.data_bytes);
215  vlib_buffer_advance (b0, -adj0->rewrite_header.data_bytes);
216  em->counters[node_counter_base_index + L2_XCRW_ERROR_FWD]++;
217  }
218 
220  && (b0->flags & VLIB_BUFFER_IS_TRACED)))
221  {
222  l2_xcrw_trace_t *t =
223  vlib_add_trace (vm, node, b0, sizeof (*t));
224  t->next_index = next0;
225  t->tx_fib_index = adj0->rewrite_header.sw_if_index;
226  }
227 
228  /* verify speculative enqueue, maybe switch current next frame */
229  vlib_validate_buffer_enqueue_x1 (vm, node, next_index,
230  to_next, n_left_to_next,
231  bi0, next0);
232  }
233 
234  vlib_put_next_frame (vm, node, next_index, n_left_to_next);
235  }
236 
237  return frame->n_vectors;
238 }
239 
240 VLIB_REGISTER_NODE (l2_xcrw_node, static) = {
241  .function = l2_xcrw_node_fn,
242  .name = "l2-xcrw",
243  .vector_size = sizeof (u32),
244  .format_trace = format_l2_xcrw_trace,
246 
247  .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 
259 {
261 
262  mp->vlib_main = vm;
263  mp->vnet_main = &vnet_main;
265 
266  return 0;
267 }
268 
270 
272  vlib_node_runtime_t * node,
273  vlib_frame_t * frame)
274 {
275  clib_warning ("you shouldn't be here, leaking buffers...");
276  return frame->n_vectors;
277 }
278 
279 static u8 * format_xcrw_name (u8 * s, va_list * args)
280 {
281  u32 dev_instance = va_arg (*args, u32);
282  return format (s, "xcrw%d", dev_instance);
283 }
284 
285 VNET_DEVICE_CLASS (xcrw_device_class,static) = {
286  .name = "Xcrw",
287  .format_device_name = format_xcrw_name,
288  .tx_function = dummy_interface_tx,
289 };
290 
291 /* Create a sham tunnel interface and return its sw_if_index */
292 static u32
294 {
295  vnet_main_t * vnm = vnet_get_main();
296  static u32 instance;
297  u8 address[6];
298  u32 hw_if_index;
300  u32 sw_if_index;
301 
302  /* mac address doesn't really matter */
303  memset (address, 0, sizeof (address));
304  address[2] = 0x12;
305 
306  /* can returns error iff phy != 0 */
308  (vnm,
309  xcrw_device_class.index,
310  instance++,
311  address,
312  &hw_if_index,
313  /* flag change */ 0);
314 
315  hi = vnet_get_hw_interface (vnm, hw_if_index);
316  sw_if_index = hi->sw_if_index;
317  vnet_sw_interface_set_flags (vnm, sw_if_index,
319 
320  /* Output to the sham tunnel invokes the encap node */
321  hi->output_node_index = l2_xcrw_node.index;
322 
323  return sw_if_index;
324 }
325 
327  u32 l2_sw_if_index, u32 tx_fib_index,
328  u8 * rewrite, u32 next_node_index, int is_add)
329 {
330  l2_xcrw_main_t * xcm = &l2_xcrw_main;
332  l2_xcrw_tunnel_t * t;
333  uword * p;
334 
335  if (is_add)
336  {
337 
338  pool_get (xcm->tunnels, t);
339 
340  /* No interface allocated? Do it. Otherwise, set admin up */
341  if (t->tunnel_sw_if_index == 0)
343  else
346 
347  t->l2_sw_if_index = l2_sw_if_index;
348 
350 
352  memset (a, 0, sizeof (*a));
353 
354  a->rewrite_header.sw_if_index = tx_fib_index;
355 
356  /*
357  * Add or find a dynamic disposition for the successor node,
358  * e.g. so we can ship pkts to mpls_post_rewrite...
359  */
360  a->rewrite_header.next_index =
361  vlib_node_add_next (vm, l2_xcrw_node.index, next_node_index);
362 
363  if (vec_len (rewrite))
364  vnet_rewrite_set_data (a[0], rewrite, vec_len(rewrite));
365 
366  set_int_l2_mode (vm, vnm, MODE_L2_XC, t->l2_sw_if_index, 0, 0, 0,
367  t->tunnel_sw_if_index);
369  t->l2_sw_if_index, t - xcm->tunnels);
370  return 0;
371  }
372  else
373  {
374  p = hash_get (xcm->tunnel_index_by_l2_sw_if_index, l2_sw_if_index);
375  if (p == 0)
376  return VNET_API_ERROR_INVALID_SW_IF_INDEX;
377 
378  t = pool_elt_at_index (xcm->tunnels, p[0]);
379 
381  /* Reset adj to drop traffic */
382  memset (a, 0, sizeof (*a));
383 
384  set_int_l2_mode (vm, vnm, MODE_L3, t->l2_sw_if_index, 0, 0, 0, 0);
385 
386  vnet_sw_interface_set_flags (vnm, t->tunnel_sw_if_index, 0 /* down */);
387 
388  hash_unset (xcm->tunnel_index_by_l2_sw_if_index, l2_sw_if_index);
389  pool_put (xcm->tunnels, t);
390  }
391  return 0;
392 }
393 
394 
395 static clib_error_t *
397  unformat_input_t * input,
398  vlib_cli_command_t * cmd)
399 {
400  unformat_input_t _line_input, * line_input = &_line_input;
401  int is_add = 1;
402  int is_ipv6 = 0; /* for fib id -> fib index mapping */
403  u32 tx_fib_id = ~0;
404  u32 tx_fib_index = ~0;
405  u32 next_node_index = ~0;
406  u32 l2_sw_if_index;
407  u8 * rw = 0;
408  vnet_main_t * vnm = vnet_get_main();
409  int rv;
410 
411 
412  if (! unformat_user (input, unformat_line_input, line_input))
413  return 0;
414 
415  if (! unformat (line_input, "%U",
416  unformat_vnet_sw_interface, vnm, &l2_sw_if_index))
417  return clib_error_return (0, "unknown input '%U'",
418  format_unformat_error, line_input);
419 
420  while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
421  {
422  if (unformat (line_input, "next %U",
423  unformat_vlib_node, vm, &next_node_index))
424  ;
425  else if (unformat (line_input, "tx-fib-id %d", &tx_fib_id))
426  ;
427  else if (unformat (line_input, "del"))
428  is_add = 0;
429  else if (unformat (line_input, "ipv6"))
430  is_ipv6 = 1;
431  else if (unformat (line_input, "rw %U",
432  unformat_hex_string, &rw));
433  else
434  break;
435  }
436 
437  if (next_node_index == ~0)
438  return clib_error_return (0, "next node not specified");
439 
440  if (tx_fib_id != ~0)
441  {
442  uword * p;
443 
444  if (is_ipv6)
445  p = hash_get (ip6_main.fib_index_by_table_id, tx_fib_id);
446  else
447  p = hash_get (ip4_main.fib_index_by_table_id, tx_fib_id);
448 
449  if (p == 0)
450  return clib_error_return (0, "nonexistent tx_fib_id %d",
451  tx_fib_id);
452 
453  tx_fib_index = p[0];
454  }
455 
456  rv = vnet_configure_l2_xcrw (vm, vnm, l2_sw_if_index, tx_fib_index,
457  rw, next_node_index, is_add);
458 
459  switch (rv)
460  {
461 
462  case 0:
463  break;
464 
465  case VNET_API_ERROR_INVALID_SW_IF_INDEX:
466  return clib_error_return (0, "%U not cross-connected",
468  vnm, l2_sw_if_index);
469  default:
470  return clib_error_return (0, "vnet_configure_l2_xcrw returned %d",
471  rv);
472  }
473 
474  vec_free (rw);
475 
476  return 0;
477 }
478 
479 VLIB_CLI_COMMAND (set_l2_xcrw_command, static) = {
480  .path = "set interface l2 xcrw",
481  .short_help =
482  "set int l2 xcrw <interface> next <node-name>\n"
483  " [del] [tx-fib-id <id>] [ipv6] rw <hex-bytes>",
484  .function = set_l2_xcrw_command_fn,
485 };
486 
487 static u8 * format_l2xcrw (u8 * s, va_list * args)
488 {
489  vnet_main_t * vnm = va_arg (*args, vnet_main_t *);
490  l2_xcrw_tunnel_t * t = va_arg (*args, l2_xcrw_tunnel_t *);
491  l2_xcrw_main_t * xcm = &l2_xcrw_main;
492  vlib_main_t * vm = vlib_get_main ();
494  u8 * rewrite_string;
495 
496  if (t == 0)
497  {
498  s = format (s, "%-25s%s", "L2 interface", "Tunnel Details");
499  return s;
500  }
501 
502  s = format (s, "%-25U %U ",
505 
507 
508  s = format (s, "next %U ",
510  a->rewrite_header.next_index);
511 
512  if (a->rewrite_header.sw_if_index != ~0)
513  s = format (s, "tx fib index %d ", a->rewrite_header.sw_if_index);
514 
515  if (a->rewrite_header.data_bytes)
516  {
517  rewrite_string = (u8 *)(a + 1);
518  rewrite_string -= a->rewrite_header.data_bytes;
519  s = format (s, "rewrite data: %U ",
520  format_hex_bytes, rewrite_string,
521  a->rewrite_header.data_bytes);
522  }
523 
524  s = format (s, "\n");
525 
526  return s;
527 }
528 
529 
530 static clib_error_t *
532  unformat_input_t * input,
533  vlib_cli_command_t * cmd)
534 {
535  vnet_main_t * vnm = vnet_get_main();
536  l2_xcrw_main_t * xcm = &l2_xcrw_main;
537  l2_xcrw_tunnel_t * t;
538 
539  if (pool_elts (xcm->tunnels) == 0)
540  {
541  vlib_cli_output (vm, "No L2 / L3 rewrite cross-connects configured");
542  return 0;
543  }
544 
545  vlib_cli_output (vm, "%U", format_l2xcrw, 0, 0);
546 
547  pool_foreach (t, xcm->tunnels,
548  ({
549  vlib_cli_output (vm, "%U", format_l2xcrw, vnm, t);
550  }));
551 
552  return 0;
553 }
554 
555 VLIB_CLI_COMMAND (show_l2xcrw_command, static) = {
556  .path = "show l2xcrw",
557  .short_help = "Display L2/L3 rewrite cross-connects",
558  .function = show_l2xcrw_command_fn,
559 };
#define vec_validate(V, I)
Make sure vector is long enough for given index (no header, unspecified alignment) ...
Definition: vec.h:394
void vlib_put_next_frame(vlib_main_t *vm, vlib_node_runtime_t *r, u32 next_index, u32 n_vectors_left)
Definition: main.c:459
#define vnet_rewrite_one_header(rw0, p0, most_likely_size)
Definition: rewrite.h:245
vmrglw vmrglh hi
u32 error_heap_index
Definition: node.h:244
#define hash_set(h, key, value)
Definition: hash.h:237
static char * l2_xcrw_error_strings[]
Definition: l2_xcrw.c:61
#define CLIB_UNUSED(x)
Definition: clib.h:79
uword unformat(unformat_input_t *i, char *fmt,...)
Definition: unformat.c:942
static uword dummy_interface_tx(vlib_main_t *vm, vlib_node_runtime_t *node, vlib_frame_t *frame)
Definition: l2_xcrw.c:271
#define hash_unset(h, key)
Definition: hash.h:243
a
Definition: bitmap.h:393
static void(BVT(clib_bihash)*h, BVT(clib_bihash_value)*v)
always_inline vlib_node_t * vlib_get_node(vlib_main_t *vm, u32 i)
Definition: node_funcs.h:46
bad routing header type(not 4)") sr_error (NO_MORE_SEGMENTS
u32 next_index
Definition: l2_xcrw.c:41
static uword l2_xcrw_node_fn(vlib_main_t *vm, vlib_node_runtime_t *node, vlib_frame_t *frame)
Definition: l2_xcrw.c:68
#define PREDICT_TRUE(x)
Definition: clib.h:98
u32 tx_fib_index
Definition: l2_xcrw.c:42
vlib_main_t * vlib_main
Definition: l2_xcrw.h:58
#define UNFORMAT_END_OF_INPUT
Definition: format.h:142
#define vnet_rewrite_set_data(rw, data, data_bytes)
Definition: rewrite.h:104
struct _vlib_node_registration vlib_node_registration_t
unformat_function_t unformat_vnet_sw_interface
#define pool_get(P, E)
Definition: pool.h:186
format_function_t format_vnet_sw_if_index_name
always_inline void * vlib_buffer_get_current(vlib_buffer_t *b)
Get pointer to current data to process.
Definition: buffer.h:184
always_inline vlib_main_t * vlib_get_main(void)
Definition: global_funcs.h:23
static clib_error_t * show_l2xcrw_command_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
Definition: l2_xcrw.c:531
always_inline uword unformat_check_input(unformat_input_t *i)
Definition: format.h:168
static u8 * format_l2xcrw(u8 *s, va_list *args)
Definition: l2_xcrw.c:487
vnet_main_t * vnet_get_main(void)
Definition: misc.c:45
#define pool_foreach(VAR, POOL, BODY)
Definition: pool.h:328
#define VLIB_INIT_FUNCTION(x)
Definition: init.h:109
unformat_function_t unformat_hex_string
Definition: format.h:285
#define vec_elt_at_index(v, i)
Get vector value at index i checking that i is in bounds.
u8 * format_hex_bytes(u8 *s, va_list *va)
Definition: std-formats.c:79
static u8 * format_xcrw_name(u8 *s, va_list *args)
Definition: l2_xcrw.c:279
clib_error_t * l2_xcrw_init(vlib_main_t *vm)
Definition: l2_xcrw.c:258
always_inline uword pool_elts(void *v)
Definition: pool.h:97
#define clib_warning(format, args...)
Definition: error.h:59
format_function_t format_vlib_next_node_name
Definition: node_funcs.h:962
uword unformat_user(unformat_input_t *input, unformat_function_t *func,...)
Definition: unformat.c:953
always_inline void * vlib_frame_vector_args(vlib_frame_t *f)
Definition: node_funcs.h:202
u32 set_int_l2_mode(vlib_main_t *vm, vnet_main_t *vnet_main, u32 mode, u32 sw_if_index, u32 bd_index, u32 bvi, u32 shg, u32 xc_sw_if_index)
Definition: l2_input.c:509
u32 l2_sw_if_index
Definition: l2_xcrw.h:38
#define hash_get(h, key)
Definition: hash.h:231
#define pool_elt_at_index(p, i)
Definition: pool.h:346
l2_xcrw_tunnel_t * tunnels
Definition: l2_xcrw.h:52
uword * fib_index_by_table_id
Definition: ip4.h:141
vlib_error_main_t error_main
Definition: main.h:124
#define pool_put(P, E)
Definition: pool.h:200
#define PREDICT_FALSE(x)
Definition: clib.h:97
vnet_main_t vnet_main
Definition: misc.c:42
vnet_main_t * vnet_main
Definition: l2_xcrw.h:59
#define vlib_validate_buffer_enqueue_x2(vm, node, next_index, to_next, n_left_to_next, bi0, bi1, next0, next1)
Definition: buffer_node.h:43
#define vlib_validate_buffer_enqueue_x1(vm, node, next_index, to_next, n_left_to_next, bi0, next0)
Definition: buffer_node.h:83
#define vlib_get_next_frame(vm, node, next_index, vectors, n_vectors_left)
Definition: node_funcs.h:265
void vlib_cli_output(vlib_main_t *vm, char *fmt,...)
Definition: cli.c:538
int vnet_configure_l2_xcrw(vlib_main_t *vm, vnet_main_t *vnm, u32 l2_sw_if_index, u32 tx_fib_index, u8 *rewrite, u32 next_node_index, int is_add)
Definition: l2_xcrw.c:326
uword * fib_index_by_table_id
Definition: ip6.h:152
u64 * counters
Definition: error.h:73
u16 n_vectors
Definition: node.h:307
#define CLIB_PREFETCH(addr, size, type)
Definition: cache.h:82
#define vec_free(V)
Free vector&#39;s memory (no header).
Definition: vec.h:298
#define foreach_l2_xcrw_error
Definition: l2_xcrw.h:67
#define ARRAY_LEN(x)
Definition: clib.h:59
l2_xcrw_next_t
Definition: l2_xcrw.h:62
always_inline vnet_hw_interface_t * vnet_get_hw_interface(vnet_main_t *vnm, u32 hw_if_index)
#define VLIB_CLI_COMMAND(x,...)
Definition: cli.h:150
#define hash_create(elts, value_bytes)
Definition: hash.h:615
u16 cached_next_index
Definition: node.h:422
#define VNET_SW_INTERFACE_FLAG_ADMIN_UP
Definition: interface.h:373
unsigned int u32
Definition: types.h:88
u8 * format_unformat_error(u8 *s, va_list *va)
Definition: unformat.c:87
#define vnet_buffer(b)
Definition: buffer.h:300
ip6_main_t ip6_main
Definition: ip6_forward.c:2490
u8 * format(u8 *s, char *fmt,...)
Definition: format.c:405
static u8 * format_l2_xcrw_trace(u8 *s, va_list *args)
Definition: l2_xcrw.c:46
clib_error_t * ethernet_register_interface(vnet_main_t *vnm, u32 dev_class_index, u32 dev_instance, u8 *address, u32 *hw_if_index_return, ethernet_flag_change_function_t flag_change)
Definition: interface.c:157
#define VLIB_NODE_FLAG_TRACE
Definition: node.h:225
always_inline uword vlib_node_add_next(vlib_main_t *vm, uword node, uword next_node)
Definition: node_funcs.h:919
#define MODE_L2_XC
Definition: l2_input.h:210
#define VLIB_BUFFER_IS_TRACED
Definition: buffer.h:91
VNET_DEVICE_CLASS(xcrw_device_class, static)
u64 uword
Definition: types.h:112
Definition: defs.h:46
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
always_inline void vlib_buffer_advance(vlib_buffer_t *b, word l)
Advance current data pointer by the supplied (signed!) amount.
Definition: buffer.h:197
unsigned char u8
Definition: types.h:56
static vlib_node_registration_t l2_xcrw_node
(constructor) VLIB_REGISTER_NODE (l2_xcrw_node)
Definition: l2_xcrw.c:59
static u32 create_xcrw_interface(vlib_main_t *vm)
Definition: l2_xcrw.c:293
unformat_function_t unformat_vlib_node
Definition: node_funcs.h:967
always_inline void * vlib_add_trace(vlib_main_t *vm, vlib_node_runtime_t *r, vlib_buffer_t *b, u32 n_data_bytes)
Definition: trace_funcs.h:55
#define vlib_prefetch_buffer_header(b, type)
Prefetch buffer metadata.
Definition: buffer.h:162
u32 tunnel_sw_if_index
Definition: l2_xcrw.h:41
#define VLIB_REGISTER_NODE(x,...)
Definition: node.h:140
ip4_main_t ip4_main
Definition: ip4_forward.c:1394
u8 data[0]
Packet data.
Definition: buffer.h:150
clib_error_t * vnet_sw_interface_set_flags(vnet_main_t *vnm, u32 sw_if_index, u32 flags)
Definition: interface.c:462
#define clib_error_return(e, args...)
Definition: error.h:112
struct _unformat_input_t unformat_input_t
#define CLIB_CACHE_LINE_BYTES
Definition: cache.h:67
u32 flags
buffer flags: VLIB_BUFFER_IS_TRACED: trace this buffer.
Definition: buffer.h:84
l2_xcrw_adjacency_t * adj_by_sw_if_index
Definition: l2_xcrw.h:49
unformat_function_t unformat_line_input
Definition: format.h:279
always_inline vlib_buffer_t * vlib_get_buffer(vlib_main_t *vm, u32 buffer_index)
Translate buffer index into buffer pointer.
Definition: buffer_funcs.h:69
Definition: defs.h:45
#define MODE_L3
Definition: l2_input.h:208
l2_xcrw_main_t l2_xcrw_main
Definition: l2_xcrw.c:57
uword * tunnel_index_by_l2_sw_if_index
Definition: l2_xcrw.h:55
static clib_error_t * set_l2_xcrw_command_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
Definition: l2_xcrw.c:396