34 l2_rw_entry_t *e = va_arg (*args, l2_rw_entry_t *);
36 s =
format (s,
"%d - mask:%U value:%U\n",
40 e->value, e->rewrite_n_vectors * sizeof (
u32x4));
42 format (s,
" hits:%d skip_bytes:%d", e->hit_count,
43 e->skip_n_vectors * sizeof (
u32x4));
50 l2_rw_config_t *
c = va_arg (*args, l2_rw_config_t *);
51 return format (s,
"table-index:%d miss-index:%d",
52 c->table_index, c->miss_index);
62 return format (s,
"l2-rw: sw_if_index %d, table %d, entry %d",
74 rw->
configs[sw_if_index].table_index = ~0;
75 rw->
configs[sw_if_index].miss_index = ~0;
79 return &rw->
configs[sw_if_index];
87 u32x4 *d = ((
u32x4 *) h) + rwe->skip_n_vectors;
88 switch (rwe->rewrite_n_vectors)
91 d[4] = (d[4] & ~rwe->mask[4]) | rwe->value[4];
94 d[3] = (d[3] & ~rwe->mask[3]) | rwe->value[3];
97 d[2] = (d[2] & ~rwe->mask[2]) | rwe->value[2];
100 d[1] = (d[1] & ~rwe->mask[1]) | rwe->value[1];
103 d[0] = (d[0] & ~rwe->mask[0]) | rwe->value[0];
111 u64 *d = ((
u64 *) h) + rwe->skip_n_vectors * 2;
112 switch (rwe->rewrite_n_vectors)
116 (d[8] & ~(((
u64 *) rwe->mask)[8])) | (((
u64 *) rwe->value)[8]);
118 (d[9] & ~(((
u64 *) rwe->mask)[9])) | (((
u64 *) rwe->value)[9]);
122 (d[6] & ~(((
u64 *) rwe->mask)[6])) | (((
u64 *) rwe->value)[6]);
124 (d[7] & ~(((
u64 *) rwe->mask)[7])) | (((
u64 *) rwe->value)[7]);
128 (d[4] & ~(((
u64 *) rwe->mask)[4])) | (((
u64 *) rwe->value)[4]);
130 (d[5] & ~(((
u64 *) rwe->mask)[5])) | (((
u64 *) rwe->value)[5]);
134 (d[2] & ~(((
u64 *) rwe->mask)[2])) | (((
u64 *) rwe->value)[2]);
136 (d[3] & ~(((
u64 *) rwe->mask)[3])) | (((
u64 *) rwe->value)[3]);
140 (d[0] & ~(((
u64 *) rwe->mask)[0])) | (((
u64 *) rwe->value)[0]);
142 (d[1] & ~(((
u64 *) rwe->mask)[1])) | (((
u64 *) rwe->value)[1]);
155 u32 n_left_from, *from, *to_next, next_index;
158 u32 prefetch_size = 0;
164 while (n_left_from > 0)
171 while (n_left_from >= 4 && n_left_to_next >= 2)
173 u32 bi0, next0, sw_if_index0, feature_bitmap0, rwe_index0;
174 u32 bi1, next1, sw_if_index1, feature_bitmap1, rwe_index1;
177 l2_rw_config_t *config0, *config1;
180 vnet_classify_entry_t *e0, *e1;
181 l2_rw_entry_t *rwe0, *rwe1;
236 rwe_index0 = e0 ? e0->opaque_index : config0->miss_index;
237 rwe_index1 = e1 ? e1->opaque_index : config1->miss_index;
239 if (rwe_index0 != ~0)
244 if (rwe_index1 != ~0)
268 vnet_buffer (b0)->l2.feature_bitmap & ~L2INPUT_FEAT_RW;
270 vnet_buffer (b1)->l2.feature_bitmap & ~L2INPUT_FEAT_RW;
271 vnet_buffer (b0)->l2.feature_bitmap = feature_bitmap0;
272 vnet_buffer (b1)->l2.feature_bitmap = feature_bitmap1;
279 to_next, n_left_to_next,
280 bi0, bi1, next0, next1);
283 while (n_left_from > 0 && n_left_to_next > 0)
285 u32 bi0, next0, sw_if_index0, feature_bitmap0, rwe_index0;
288 l2_rw_config_t *config0;
291 vnet_classify_entry_t *e0;
318 rwe_index0 = e0 ? e0->opaque_index : config0->miss_index;
320 if (rwe_index0 != ~0)
336 vnet_buffer (b0)->l2.feature_bitmap & ~L2INPUT_FEAT_RW;
337 vnet_buffer (b0)->l2.feature_bitmap = feature_bitmap0;
342 to_next, n_left_to_next,
356 l2_rw_entry_t *e = 0;
380 e->skip_n_vectors = skip /
sizeof (
u32x4);
381 skip -= e->skip_n_vectors *
sizeof (
u32x4);
382 e->rewrite_n_vectors = (skip + len - 1) /
sizeof (
u32x4) + 1;
384 memset (e->mask, 0, e->rewrite_n_vectors * sizeof (
u32x4));
386 memset (e->value, 0, e->rewrite_n_vectors * sizeof (
u32x4));
392 for (i = 0; i < e->rewrite_n_vectors; i++)
394 e->value[
i] &= e->mask[
i];
412 if (
unformat (input,
"index %d", &index))
418 else if (
unformat (input,
"skip %d", &skip))
442 .path =
"l2 rewrite entry",
444 "l2 rewrite entry [index <index>] [mask <hex-mask>] [value <hex-value>] [skip <n_bytes>] [del]",
455 c->table_index = table_index;
456 c->miss_index = miss_index;
457 u32 feature_bitmap = (table_index == ~0) ? 0 : L2INPUT_FEAT_RW;
461 if (c->table_index == ~0)
472 u32 table_index = ~0;
473 u32 sw_if_index = ~0;
483 if (
unformat (input,
"table %d", &table_index))
485 else if (
unformat (input,
"miss-index %d", &miss_index))
491 if (sw_if_index == ~0)
493 "You must specify an interface 'iface <interface>'",
506 .path =
"set interface l2 rewrite",
508 "set interface l2 rewrite <interface> [table <table index>] [miss-index <entry-index>]",
525 vlib_cli_output (vm,
"sw_if_index:%d %U\n", i, format_l2_rw_config, &rw->configs[i]);
533 .path =
"show l2 rewrite interfaces",
535 "show l2 rewrite interfaces",
551 vlib_cli_output (vm,
"%U\n", format_l2_rw_entry, e);
559 .path =
"show l2 rewrite entries",
561 "show l2 rewrite entries",
569 u32 mask = L2INPUT_FEAT_RW;
582 !
unformat (input,
"%d", &bridge_domain))
601 .path =
"set bridge-domain rewrite",
603 "set bridge-domain rewrite <bridge-domain> [disable]",
631 #define foreach_l2_rw_error \ 632 _(UNKNOWN, "Unknown error") 636 #define _(sym,str) L2_RW_ERROR_##sym, 643 #define _(sym,string) string, 652 .vector_size =
sizeof (
u32),
657 .runtime_data_bytes = 0,
u64 vnet_classify_hash_packet(vnet_classify_table_t *t, u8 *h)
#define vec_validate(V, I)
Make sure vector is long enough for given index (no header, unspecified alignment) ...
void vlib_put_next_frame(vlib_main_t *vm, vlib_node_runtime_t *r, u32 next_index, u32 n_vectors_left)
Release pointer to next frame vector data.
sll srl srl sll sra u16x4 i
static vlib_main_t * vlib_get_main(void)
bad routing header type(not 4)") sr_error (NO_MORE_SEGMENTS
static clib_error_t * l2_rw_show_interfaces_cli_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
static l2_rw_config_t * l2_rw_get_config(u32 sw_if_index)
static u8 * format_l2_rw_entry(u8 *s, va_list *args)
static f64 vlib_time_now(vlib_main_t *vm)
static clib_error_t * l2_rw_show_entries_cli_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
struct _vlib_node_registration vlib_node_registration_t
static uword * clib_bitmap_set(uword *ai, uword i, uword value)
Sets the ith bit of a bitmap to new_value Removes trailing zeros from the bitmap. ...
unformat_function_t unformat_vnet_sw_interface
#define pool_get(P, E)
Allocate an object E from a pool P (unspecified alignment).
vlib_node_registration_t l2_rw_node
(constructor) VLIB_REGISTER_NODE (l2_rw_node)
static u8 * format_l2_rw_config(u8 *s, va_list *args)
#define foreach_l2_rw_error
vnet_main_t * vnet_get_main(void)
#define static_always_inline
#define pool_foreach(VAR, POOL, BODY)
Iterate through pool.
#define VLIB_INIT_FUNCTION(x)
static void * vlib_buffer_get_current(vlib_buffer_t *b)
Get pointer to current data to process.
static char * l2_rw_error_strings[]
int l2_rw_enable_disable(u32 bridge_domain, u8 disable)
#define clib_bitmap_foreach(i, ai, body)
Macro to iterate across set bits in a bitmap.
static clib_error_t * l2_rw_set_cli_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
#define pool_elt_at_index(p, i)
Returns pointer to element at given index.
int l2_rw_mod_entry(u32 *index, u8 *mask, u8 *value, u32 len, u32 skip, u8 is_del)
static clib_error_t * l2_rw_entry_cli_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
#define vec_alloc_aligned(V, N, A)
Allocate space for N more elements (no header, given alignment)
#define pool_put(P, E)
Free an object E in pool P.
static_always_inline void l2_rw_rewrite(l2_rw_entry_t *rwe, u8 *h)
#define vlib_validate_buffer_enqueue_x2(vm, node, next_index, to_next, n_left_to_next, bi0, bi1, next0, next1)
Finish enqueueing two buffers forward in the graph.
static u32 feat_bitmap_get_next_node_index(u32 *next_nodes, u32 bitmap)
Return the graph node index for the feature corresponding to the first set bit in the bitmap...
#define vlib_validate_buffer_enqueue_x1(vm, node, next_index, to_next, n_left_to_next, bi0, next0)
Finish enqueueing one buffer forward in the graph.
#define vlib_get_next_frame(vm, node, next_index, vectors, n_vectors_left)
Get pointer to next frame vector data by (vlib_node_runtime_t, next_index).
void vlib_cli_output(vlib_main_t *vm, char *fmt,...)
static u8 * format_l2_rw_trace(u8 *s, va_list *args)
static uword l2_rw_node_fn(vlib_main_t *vm, vlib_node_runtime_t *node, vlib_frame_t *frame)
#define clib_bitmap_alloc(v, n_bits)
Allocate a bitmap with the supplied number of bits.
#define CLIB_PREFETCH(addr, size, type)
static void feat_bitmap_init_next_nodes(vlib_main_t *vm, u32 node_index, u32 num_features, char **feat_names, u32 *next_nodes)
Initialize the feature next-node indexes of a graph node.
#define clib_memcpy(a, b, c)
#define pool_is_free_index(P, I)
Use free bitmap to query whether given index is free.
static uword clib_bitmap_get(uword *ai, uword i)
Gets the ith bit value from a bitmap.
struct _vnet_classify_main vnet_classify_main_t
static clib_error_t * l2_rw_init(vlib_main_t *vm)
static clib_error_t * l2_rw_interface_cli_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
u32 feat_next_node_index[32]
vnet_classify_main_t vnet_classify_main
#define VLIB_BUFFER_IS_TRACED
static void * vlib_add_trace(vlib_main_t *vm, vlib_node_runtime_t *r, vlib_buffer_t *b, u32 n_data_bytes)
static uword clib_bitmap_count_set_bits(uword *ai)
Return the number of set bits in a bitmap.
VLIB_CLI_COMMAND(set_interface_ip_source_and_port_range_check_command, static)
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
static void * vlib_frame_vector_args(vlib_frame_t *f)
Get pointer to frame vector data.
int l2_rw_interface_set_table(u32 sw_if_index, u32 table_index, u32 miss_index)
#define vlib_prefetch_buffer_header(b, type)
Prefetch buffer metadata.
#define VLIB_NODE_FUNCTION_MULTIARCH(node, fn)
#define VLIB_REGISTER_NODE(x,...)
#define clib_error_return(e, args...)
u32 flags
buffer flags: VLIB_BUFFER_IS_TRACED: trace this buffer.
static vlib_buffer_t * vlib_get_buffer(vlib_main_t *vm, u32 buffer_index)
Translate buffer index into buffer pointer.
vnet_classify_entry_t * vnet_classify_find_entry(vnet_classify_table_t *t, u8 *h, u64 hash, f64 now)
static uword pool_elts(void *v)
Number of active elements in a pool.