55 case VXLAN_GPE_PROTOCOL_IP4:
58 case VXLAN_GPE_PROTOCOL_IP6:
61 case VXLAN_GPE_PROTOCOL_ETHERNET:
62 s =
format (s,
"protocol ethernet");
64 case VXLAN_GPE_PROTOCOL_NSH:
65 s =
format (s,
"protocol nsh");
89 s =
format (s,
"[%d] lcl %U rmt %U vni %d fib-idx %d sw-if-idx %d ",
120 u32 dev_instance = va_arg (*args,
u32);
121 return format (s,
"vxlan_gpe_tunnel%d", dev_instance);
167 u32 dev_instance = va_arg (*args,
u32);
168 s =
format (s,
"unimplemented dev %u", dev_instance);
250 #define foreach_gpe_copy_field \ 253 _(mcast_sw_if_index) \ 257 #define foreach_copy_ipv4 { \ 258 _(local.ip4.as_u32) \ 259 _(remote.ip4.as_u32) \ 262 #define foreach_copy_ipv6 { \ 263 _(local.ip6.as_u64[0]) \ 264 _(local.ip6.as_u64[1]) \ 265 _(remote.ip6.as_u64[0]) \ 266 _(remote.ip6.as_u64[1]) \ 280 u8 protocol_override,
uword encap_next_node)
284 ip4_vxlan_gpe_header_t *h0;
287 len =
sizeof (*h0) + extension_size;
292 h0 = (ip4_vxlan_gpe_header_t *) rw;
306 h0->udp.src_port = clib_host_to_net_u16 (4790);
307 h0->udp.dst_port = clib_host_to_net_u16 (UDP_DST_PORT_VXLAN_GPE);
312 if (protocol_override)
314 h0->vxlan.protocol = protocol_override;
320 t->
rewrite_size =
sizeof (ip4_vxlan_gpe_header_t) + extension_size;
321 h0->vxlan.vni_res = clib_host_to_net_u32 (t->
vni << 8);
338 u8 protocol_override,
uword encap_next_node)
342 ip6_vxlan_gpe_header_t *h0;
345 len =
sizeof (*h0) + extension_size;
350 h0 = (ip6_vxlan_gpe_header_t *) rw;
355 clib_host_to_net_u32 (6 << 28);
365 h0->udp.src_port = clib_host_to_net_u16 (4790);
366 h0->udp.dst_port = clib_host_to_net_u16 (UDP_DST_PORT_VXLAN_GPE);
371 if (protocol_override)
377 h0->vxlan.protocol = protocol_override;
379 t->
rewrite_size =
sizeof (ip4_vxlan_gpe_header_t) + extension_size;
380 h0->vxlan.vni_res = clib_host_to_net_u32 (t->
vni << 8);
397 static inline mcast_shared_t
403 return (mcast_shared_t)
412 mcast_shared_t new_ep = {
413 .mcast_adj_index = ai,
414 .mfib_entry_index = mfei,
448 u32 hw_if_index = ~0;
451 vxlan4_gpe_tunnel_key_t key4, *key4_copy;
452 vxlan6_gpe_tunnel_key_t key6, *key6_copy;
457 key4.local = a->
local.ip4.as_u32;
458 key4.remote = a->
remote.ip4.as_u32;
459 key4.vni = clib_host_to_net_u32 (a->
vni << 8);
466 key6.local.as_u64[0] = a->
local.ip6.as_u64[0];
467 key6.local.as_u64[1] = a->
local.ip6.as_u64[1];
468 key6.remote.as_u64[0] = a->
remote.ip6.as_u64[0];
469 key6.remote.as_u64[1] = a->
remote.ip6.as_u64[1];
470 key6.vni = clib_host_to_net_u32 (a->
vni << 8);
481 return VNET_API_ERROR_TUNNEL_EXIST;
488 #define _(x) t->x = a->x; 557 (vnm, vxlan_gpe_device_class.index, t - ngm->
tunnels,
558 vxlan_gpe_hw_class.index, t - ngm->
tunnels);
619 .frp_sw_if_index = 0xffffffff,
627 .fp_len = (is_ip6 ? 128 : 32),
628 .fp_grp_addr = tun_remote_pfx.
fp_addr,
680 return VNET_API_ERROR_NO_SUCH_ENTRY;
739 ip46_address_t local, remote;
746 u32 encap_fib_index = 0;
747 u32 decap_fib_index = 0;
765 else if (
unformat (line_input,
"local %U",
771 else if (
unformat (line_input,
"remote %U",
777 else if (
unformat (line_input,
"local %U",
783 else if (
unformat (line_input,
"remote %U",
789 else if (
unformat (line_input,
"group %U %U",
794 grp_set = remote_set = 1;
797 else if (
unformat (line_input,
"group %U %U",
802 grp_set = remote_set = 1;
805 else if (
unformat (line_input,
"encap-vrf-id %d", &tmp))
812 if (encap_fib_index == ~0)
819 else if (
unformat (line_input,
"decap-vrf-id %d", &tmp))
826 if (decap_fib_index == ~0)
833 else if (
unformat (line_input,
"vni %d", &vni))
835 else if (
unformat (line_input,
"next-ip4"))
836 protocol = VXLAN_GPE_PROTOCOL_IP4;
837 else if (
unformat (line_input,
"next-ip6"))
838 protocol = VXLAN_GPE_PROTOCOL_IP6;
839 else if (
unformat (line_input,
"next-ethernet"))
840 protocol = VXLAN_GPE_PROTOCOL_ETHERNET;
841 else if (
unformat (line_input,
"next-nsh"))
842 protocol = VXLAN_GPE_PROTOCOL_NSH;
875 if (grp_set && mcast_sw_if_index == ~0)
880 if (ipv4_set && ipv6_set)
886 if ((ipv4_set && memcmp (&local.ip4, &remote.ip4, sizeof (local.ip4)) == 0)
888 && memcmp (&local.ip6, &remote.ip6, sizeof (local.ip6)) == 0))
906 #define _(x) a->x = x; 923 case VNET_API_ERROR_INVALID_DECAP_NEXT:
927 case VNET_API_ERROR_TUNNEL_EXIST:
931 case VNET_API_ERROR_NO_SUCH_ENTRY:
937 (0,
"vnet_vxlan_gpe_add_del_tunnel returned %d", rv);
968 .path =
"create vxlan-gpe tunnel",
970 "create vxlan-gpe tunnel local <local-addr> " 971 " {remote <remote-addr>|group <mcast-addr> <intf-name>}" 972 " vni <nn> [next-ip4][next-ip6][next-ethernet][next-nsh]" 973 " [encap-vrf-id <nn>] [decap-vrf-id <nn>] [del]\n",
1002 vlib_cli_output (vm,
"%U", format_vxlan_gpe_tunnel, t);
1020 .path =
"show vxlan-gpe",
1030 sw_if_index, is_enable, 0, 0);
1033 sw_if_index, is_enable, 0, 0);
1057 else if (
unformat (line_input,
"del"))
1066 if (~0 == sw_if_index)
1132 .path =
"set interface ip vxlan-gpe-bypass",
1134 .short_help =
"set interface ip vxlan-gpe-bypass <interface> [del]",
1189 .path =
"set interface ip6 vxlan-gpe-bypass",
1191 .short_help =
"set interface ip6 vxlan-gpe-bypass <interface> [del]",
1198 .arc_name =
"ip4-unicast",
1199 .node_name =
"ip4-vxlan-gpe-bypass",
1205 .arc_name =
"ip6-unicast",
1206 .node_name =
"ip6-vxlan-gpe-bypass",
1235 sizeof (ip46_address_t),
1236 sizeof (mcast_shared_t));
1241 VXLAN_GPE_INPUT_NEXT_IP4_INPUT);
1243 VXLAN_GPE_INPUT_NEXT_IP6_INPUT);
1245 VXLAN_GPE_INPUT_NEXT_L2_INPUT);
#define vec_validate(V, I)
Make sure vector is long enough for given index (no header, unspecified alignment) ...
void vnet_set_interface_output_node(vnet_main_t *vnm, u32 hw_if_index, u32 node_index)
Set interface output node - for interface registered without its output/tx nodes created because its ...
void dpo_stack_from_node(u32 child_node_index, dpo_id_t *dpo, const dpo_id_t *parent)
Stack one DPO object on another, and thus establish a child parent relationship.
uword * vxlan6_gpe_tunnel_by_key
lookup IPv6 VXLAN GPE tunnel by key
u32 sibling_index
The tunnel is a child of the FIB entry for its destination.
fib_node_index_t fib_entry_track(u32 fib_index, const fib_prefix_t *prefix, fib_node_type_t child_type, index_t child_index, u32 *sibling)
Trackers are used on FIB entries by objects that which to track the changing state of the entry...
Contribute an object that is to be used to forward IP6 packets.
l2_input_config_t * configs
uword vtep_addr_ref(vtep_table_t *t, u32 fib_index, ip46_address_t *ip)
static fib_node_back_walk_rc_t vxlan_gpe_tunnel_back_walk(fib_node_t *node, fib_node_back_walk_ctx_t *ctx)
Function definition to backwalk a FIB node - Here we will restack the new dpo of VXLAN_GPE DIP to enc...
static void vxlan_gpe_tunnel_last_lock_gone(fib_node_t *node)
Function definition to inform the FIB node that its last lock has gone.
clib_error_t * vxlan_gpe_init(vlib_main_t *vm)
Feature init function for VXLAN GPE.
static uword ip46_address_is_multicast(const ip46_address_t *a)
#define hash_unset(h, key)
vl_api_wireguard_peer_flags_t flags
A representation of a path as described by a route producer.
vnet_main_t * vnet_get_main(void)
vlib_node_registration_t vxlan4_gpe_input_node
(constructor) VLIB_REGISTER_NODE (vxlan4_gpe_input_node)
vnet_interface_main_t interface_main
void fib_node_init(fib_node_t *node, fib_node_type_t type)
fib_node_index_t mfib_table_entry_path_update(u32 fib_index, const mfib_prefix_t *prefix, mfib_source_t source, const fib_route_path_t *rpath)
Add n paths to an entry (aka route) in the FIB.
u32 frp_mitf_flags
MFIB interface flags.
static mcast_shared_t mcast_shared_get(ip46_address_t *ip)
#define foreach_copy_ipv4
#define clib_memcpy_fast(a, b, c)
clib_memset(h->entries, 0, sizeof(h->entries[0]) *entries)
u32 * tunnel_index_by_sw_if_index
Mapping from sw_if_index to tunnel index.
enum fib_node_back_walk_rc_t_ fib_node_back_walk_rc_t
Return code from a back walk function.
void fib_entry_contribute_forwarding(fib_node_index_t fib_entry_index, fib_forward_chain_type_t fct, dpo_id_t *dpo)
static vnet_hw_interface_t * vnet_get_hw_interface(vnet_main_t *vnm, u32 hw_if_index)
#define vec_add1(V, E)
Add 1 element to end of vector (unspecified alignment).
int vxlan6_gpe_rewrite(vxlan_gpe_tunnel_t *t, u32 extension_size, u8 protocol_override, uword encap_next_node)
Calculate IPv6 VXLAN GPE rewrite header.
vlib_node_registration_t vxlan6_gpe_input_node
(constructor) VLIB_REGISTER_NODE (vxlan6_gpe_input_node)
Contribute an object that is to be used to forward IP4 packets.
#define hash_set_mem(h, key, value)
#define STRUCT_OFFSET_OF(t, f)
void fib_node_deinit(fib_node_t *node)
static vnet_sw_interface_t * vnet_get_sw_interface(vnet_main_t *vnm, u32 sw_if_index)
unformat_function_t unformat_vnet_sw_interface
#define vec_validate_aligned(V, I, A)
Make sure vector is long enough for given index (no header, specified alignment)
int vnet_vxlan_gpe_add_del_tunnel(vnet_vxlan_gpe_add_del_tunnel_args_t *a, u32 *sw_if_indexp)
Add or Del a VXLAN GPE tunnel.
#define VXLAN_GPE_FLAGS_P
static u8 ip46_address_is_ip4(const ip46_address_t *ip46)
format_function_t format_vnet_sw_if_index_name
vnet_main_t * vnet_main
State convenience vnet_main_t.
enum fib_protocol_t_ fib_protocol_t
Protocol Type.
#define VXLAN_GPE_VERSION
void fib_node_register_type(fib_node_type_t type, const fib_node_vft_t *vft)
fib_node_register_type
#define VXLAN_GPE_FLAGS_I
vl_api_ip_proto_t protocol
#define foreach_gpe_copy_field
vnet_flood_class_t flood_class
static fib_node_t * vxlan_gpe_tunnel_fib_node_get(fib_node_index_t index)
Function definition to get a FIB node from its index.
u8 * format_vxlan_gpe_tunnel(u8 *s, va_list *args)
Format function for VXLAN GPE tunnel.
#define pool_foreach(VAR, POOL, BODY)
Iterate through pool.
u32 frp_sw_if_index
The interface.
#define VLIB_INIT_FUNCTION(x)
ip46_address_t local
tunnel local address
static u8 * format_vxlan_gpe_header_with_length(u8 *s, va_list *args)
Formatting function for tracing VXLAN GPE with length.
void mfib_table_entry_delete_index(fib_node_index_t mfib_entry_index, mfib_source_t source)
Delete a FIB entry.
vlib_combined_counter_main_t * combined_sw_if_counters
vl_api_interface_index_t mcast_sw_if_index
u8 * format_vxlan_gpe_encap_trace(u8 *s, va_list *args)
Trace of packets encapsulated in VXLAN GPE.
Aggregate type for a prefix.
#define clib_error_return(e, args...)
void adj_unlock(adj_index_t adj_index)
Release a reference counting lock on the adjacency.
#define ALWAYS_ASSERT(truth)
u32 fib_table_find(fib_protocol_t proto, u32 table_id)
Get the index of the FIB for a Table-ID.
void vxlan_gpe_register_decap_protocol(u8 protocol_id, uword next_node_index)
u32 vnet_register_interface(vnet_main_t *vnm, u32 dev_class_index, u32 dev_instance, u32 hw_class_index, u32 hw_instance)
fib_node_index_t fib_entry_index
u8 * rewrite
Rewrite string.
VNET_FEATURE_INIT(ip4_vxlan_gpe_bypass, static)
void vnet_int_vxlan_gpe_bypass_mode(u32 sw_if_index, u8 is_ip6, u8 is_enable)
The identity of a DPO is a combination of its type and its instance number/index of objects of that t...
#define hash_create_mem(elts, key_bytes, value_bytes)
int vxlan4_gpe_rewrite(vxlan_gpe_tunnel_t *t, u32 extension_size, u8 protocol_override, uword encap_next_node)
Calculate IPv4 VXLAN GPE rewrite header.
#define pool_elt_at_index(p, i)
Returns pointer to element at given index.
static void vlib_zero_combined_counter(vlib_combined_counter_main_t *cm, u32 index)
Clear a combined counter Clears the set of per-thread counters.
uword vtep_addr_unref(vtep_table_t *t, u32 fib_index, ip46_address_t *ip)
ip46_address_t fp_addr
The address type is not deriveable from the fp_addr member.
Struct for VXLAN GPE tunnel.
typedef CLIB_PACKED(union { struct { fib_node_index_t mfib_entry_index;adj_index_t mcast_adj_index;};u64 as_u64;})
#define pool_put(P, E)
Free an object E in pool P.
static void mcast_shared_remove(ip46_address_t *remote)
vxlan_gpe_main_t vxlan_gpe_main
static vxlan_gpe_tunnel_t * vxlan_gpe_tunnel_from_fib_node(fib_node_t *node)
vlib_main_t * vlib_main
State convenience vlib_main_t.
vnet_sw_interface_flags_t flags
u8 protocol
encapsulated protocol
vlib_simple_counter_main_t * sw_if_counters
static clib_error_t * set_ip4_vxlan_gpe_bypass(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
fib_node_type_t fn_type
The node's type.
An node in the FIB graph.
ip46_address_t remote
tunnel remote address
u32 vni
VXLAN GPE VNI in HOST byte order, shifted left 8 bits.
format_function_t format_ip46_address
#define pool_get_aligned(P, E, A)
Allocate an object E from a pool P with alignment A.
static clib_error_t * set_ip6_vxlan_gpe_bypass(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
Struct for VXLAN GPE add/del args.
VNET_DEVICE_CLASS(vxlan_gpe_device_class, static)
#define vec_free(V)
Free vector's memory (no header).
static void vnet_interface_counter_unlock(vnet_interface_main_t *im)
fib_node_t node
Linkage into the FIB object graph.
u32 fib_node_index_t
A typedef of a node index.
u32 adj_index_t
An index for adjacencies.
static u8 * format_vxlan_gpe_name(u8 *s, va_list *args)
Naming for VXLAN GPE tunnel.
void fib_prefix_from_ip46_addr(const ip46_address_t *addr, fib_prefix_t *pfx)
Host prefix from ip.
void dpo_set(dpo_id_t *dpo, dpo_type_t type, dpo_proto_t proto, index_t index)
Set/create a DPO ID The DPO will be locked.
Aggregate type for a prefix.
u8 * default_build_rewrite(vnet_main_t *vnm, u32 sw_if_index, vnet_link_t link_type, const void *dst_address)
Return a complete, zero-length (aka placeholder) rewrite.
vlib_main_t vlib_node_runtime_t * node
Context passed between object during a back walk.
#define VLIB_CLI_COMMAND(x,...)
static void vnet_interface_counter_lock(vnet_interface_main_t *im)
static clib_error_t * vxlan_gpe_add_del_tunnel_command_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
void fib_entry_untrack(fib_node_index_t fei, u32 sibling)
Stop tracking a FIB entry.
void vlib_cli_output(vlib_main_t *vm, char *fmt,...)
VNET_HW_INTERFACE_CLASS(vxlan_gpe_hw_class)
static clib_error_t * vxlan_gpe_interface_admin_up_down(vnet_main_t *vnm, u32 hw_if_index, u32 flags)
CLI function for VXLAN GPE admin up/down.
u8 rewrite_size
rewrite size for dynamic plugins like iOAM
Struct for VXLAN GPE node state.
enum fib_forward_chain_type_t_ fib_forward_chain_type_t
FIB output chain type.
fib_route_path_flags_t frp_flags
flags on the path
static void hash_unset_mem_free(uword **h, const void *key)
static void vlib_zero_simple_counter(vlib_simple_counter_main_t *cm, u32 index)
Clear a simple counter Clears the set of per-thread u16 counters, and the u64 counter.
dpo_proto_t fib_proto_to_dpo(fib_protocol_t fib_proto)
static u8 * format_decap_next(u8 *s, va_list *args)
static void * clib_mem_alloc(uword size)
#define VNET_FEATURES(...)
static vtep_table_t vtep_table_create()
vxlan_gpe_tunnel_t * tunnels
vector of encap tunnel instances
fib_protocol_t fp_proto
protocol type
uword * vxlan4_gpe_tunnel_by_key
lookup IPv4 VXLAN GPE tunnel by key
u32 decap_fib_index
FIB indices - inner IP packet lookup here.
index_t dpoi_index
the index of objects of that type
bool udp_is_valid_dst_port(udp_dst_port_t dst_port, u8 is_ip4)
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
static fib_protocol_t fib_ip_proto(bool is_ip6)
Convert from boolean is_ip6 to FIB protocol.
clib_error_t * vnet_hw_interface_set_flags(vnet_main_t *vnm, u32 hw_if_index, vnet_hw_interface_flags_t flags)
static clib_error_t * show_vxlan_gpe_tunnel_command_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI function for showing VXLAN GPE tunnels.
#define DPO_INVALID
An initialiser for DPOs declared on the stack.
u32 hw_if_index
vnet intfc hw_if_index
static void vxlan_gpe_tunnel_restack_dpo(vxlan_gpe_tunnel_t *t)
#define hash_get_mem(h, key)
A FIB graph nodes virtual function table.
u32 encap_fib_index
FIB indices - tunnel partner lookup here.
clib_error_t * vnet_sw_interface_set_flags(vnet_main_t *vnm, u32 sw_if_index, vnet_sw_interface_flags_t flags)
uword encap_next_node
Next node after VxLAN-GPE encap.
u32 sw_if_index
vnet intfc sw_if_index
adj_index_t adj_mcast_add_or_lock(fib_protocol_t proto, vnet_link_t link_type, u32 sw_if_index)
Mcast Adjacency.
void dpo_reset(dpo_id_t *dpo)
reset a DPO ID The DPO will be unlocked.
vnet_link_t fib_proto_to_link(fib_protocol_t proto)
Convert from a protocol to a link type.
static void mcast_shared_add(ip46_address_t *remote, fib_node_index_t mfei, adj_index_t ai)
static void hash_set_mem_alloc(uword **h, const void *key, uword v)
void udp_register_dst_port(vlib_main_t *vm, udp_dst_port_t dst_port, u32 node_index, u8 is_ip4)
vlib_node_registration_t vxlan_gpe_encap_node
(constructor) VLIB_REGISTER_NODE (vxlan_gpe_encap_node)
#define vec_validate_init_empty(V, I, INIT)
Make sure vector is long enough for given index and initialize empty space (no header, unspecified alignment)
#define CLIB_CACHE_LINE_BYTES
#define VXLAN_GPE_TUNNEL_IS_IPV4
Flags for vxlan_gpe_tunnel_t.
static clib_error_t * set_ip_vxlan_gpe_bypass(u32 is_ip6, unformat_input_t *input, vlib_cli_command_t *cmd)
#define foreach_copy_ipv6
static u16 ip4_header_checksum(ip4_header_t *i)
const ip46_address_t zero_addr
vl_api_interface_index_t sw_if_index
int vnet_feature_enable_disable(const char *arc_name, const char *node_name, u32 sw_if_index, int enable_disable, void *feature_config, u32 n_feature_config_bytes)
u32 * free_vxlan_gpe_tunnel_hw_if_indices
Free vlib hw_if_indices.
static uword pool_elts(void *v)
Number of active elements in a pool.