/*
* Copyright (c) 2015 Cisco and/or its affiliates.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef included_nsh_h
#define included_nsh_h
#include <vnet/vnet.h>
#include <nsh/nsh_packet.h>
#include <vnet/ip/ip4_packet.h>
typedef struct {
u16 class;
u8 type;
u8 pad;
} nsh_option_map_by_key_t;
typedef struct {
u32 option_id;
} nsh_option_map_t;
#define MAX_METADATA_LEN 62
/** Note:
* rewrite and rewrite_size used to support varied nsh header
*/
typedef struct {
/* Required for pool_get_aligned */
CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
nsh_base_header_t nsh_base;
union {
nsh_md1_data_t md1_data;
nsh_md2_data_t md2_data;
} md;
u8 tlvs_len; /* configured md2 metadata's length, unit: byte */
u8 * tlvs_data; /* configured md2 metadata, network order */
/** Rewrite string. network order
* contains base header and metadata */
u8 * rewrite;
u8 rewrite_size; /* unit: byte */
} nsh_entry_t;
typedef struct {
u8 is_add;
nsh_entry_t nsh_entry;
} nsh_add_del_entry_args_t;
typedef struct {
/* Required for pool_get_aligned */
CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
/** Key for nsh_header_t entry: 24bit NSP 8bit NSI */
u32 nsp_nsi;
/** Key for nsh_header_t entry to map to. : 24bit NSP 8bit NSI
* This may be ~0 if next action is to decap to NSH next protocol
* Note the following heuristic:
* if nsp_nsi == mapped_nsp_nsi then use-case is like SFC SFF
* if nsp_nsi != mapped_nsp_nsi then use-case is like SFC SF
* Note: these are heuristics. Rules about NSI decrement are out of scope
*/
u32 mapped_nsp_nsi;
/* NSH Header action: swap, push and pop */
u32 nsh_action;
/** vnet intfc hw_if_index */
u32 nsh_hw_if;
/* vnet intfc sw_if_index */
u32 nsh_sw_if;
/* encap if index */
u32 sw_if_index;
u32 rx_sw_if_index;
u32 next_node;
u32 adj_index;
} nsh_map_t;
typedef struct {
u8 is_add;
nsh_map_t map;
} nsh_add_del_map_args_t;
typedef struct {
u32 transport_type; /* 1:vxlan; */
u32 transport_index; /* transport's sw_if_index */
} nsh_proxy_session_by_key_t;
typedef struct {
/* 24bit NSP 8bit NSI */
u32 nsp_nsi;
} nsh_proxy_session_t;
#define MAX_MD2_OPTIONS 256
typedef struct {
/* API message ID base */
u16 msg_id_base;
/* vector of nsh_header entry instances */
nsh_entry_t *nsh_entries;
/* hash lookup nsh header by key: {u32: nsp_nsi} */
uword * nsh_entry_by_key;
/* vector of nsh_mappings */
nsh_map_t *nsh_mappings;
/* hash lookup nsh mapping by key: {u32: nsp_nsi} */
uword * nsh_mapping_by_key;
uword * nsh_mapping_by_mapped_key; // for use in NSHSFC
/* vector of nsh_proxy */
nsh_proxy_session_t *nsh_proxy_sessions;
/* hash lookup nsh_proxy by key */
uword * nsh_proxy_session_by_key;
/** Free vlib hw_if_indices */
u32 * free_nsh_tunnel_hw_if_indices;
/** Mapping from sw_if_index to tunnel index */
u32 * tunnel_index_by_sw_if_index;
/* vector of nsh_option_map */
nsh_option_map_t * nsh_option_mappings;
/* hash lookup nsh_option_map by key */
uword * nsh_option_map_by_key;
/* Array of function pointers to process MD-Type 2 handling routines */
/*
* For API or CLI configuration and construct the rewrite buffer, invokes add_options() function.
* In the encap node, i.e. when performing PUSH nsh header, invokes options() function.
* In the swap node, i.e. when performing SWAP nsh header, invokes swap_options() function.
* In the decap node, i.e. when performing POP nsh header, invokes pop_options() function.
*/
u8 options_size[MAX_MD2_OPTIONS]; /* sum of header and metadata */
int (*add_options[MAX_MD2_OPTIONS]) (u8 * opt,
u8 * opt_size);
int (*options[MAX_MD2_OPTIONS]) (vlib_buffer_t * b,
nsh_tlv_header_t * opt);
int (*swap_options[MAX_MD2_OPTIONS]) (vlib_buffer_t * b,
nsh_tlv_header_t * old_opt,
nsh_tlv_header_t * new_opt);
int (*pop_options[MAX_MD2_OPTIONS]) (vlib_buffer_t * b,
nsh_tlv_header_t * opt);
u8 *(*trace[MAX_MD2_OPTIONS]) (u8 * s, nsh_tlv_header_t * opt);
uword decap_v4_next_override;
/* Feature arc indices */
u8 input_feature_arc_index;
u8 output_feature_arc_index;
u32 nsh_input_node_index;
u32 nsh_proxy_node_index;
u32 nsh_classifier_node_index;
/* convenience */
vlib_main_t * vlib_main;
vnet_main_t * vnet_main;
} nsh_main_t;
extern nsh_main_t nsh_main;
extern vlib_node_registration_t nsh_aware_vnf_proxy_node;
extern vlib_node_registration_t nsh_eth_output_node;
typedef struct {
u8 trace_data[256];
} nsh_input_trace_t;
u8 * format_nsh_input_map_trace (u8 * s, va_list * args);
u8 * format_nsh_header_with_length (u8 * s, va_list * args);
/* Helper macros used in nsh.c and nsh_test.c */
#define foreach_copy_nsh_base_hdr_field \
_(ver_o_c) \
_(length) \
_(md_type) \
_(next_protocol) \
_(nsp_nsi)
/* Statistics (not really errors) */
#define foreach_nsh_node_error \
_(MAPPED, "NSH header found and mapped") \
_(NO_MAPPING, "no mapping for nsh key") \
_(NO_ENTRY, "no entry for nsh key") \
_(NO_PROXY, "no proxy for transport key") \
_(INVALID_NEXT_PROTOCOL, "invalid next protocol") \
_(INVALID_OPTIONS, "invalid md2 options") \
_(INVALID_TTL, "ttl equals zero") \
typedef enum {
#define _(sym,str) NSH_NODE_ERROR_##sym,
foreach_nsh_node_error
#undef _
NSH_NODE_N_ERROR,
} nsh_input_error_t;
#define foreach_nsh_node_next \
_(DROP, "error-drop") \
_(ENCAP_GRE4, "gre4-input" ) \
_(ENCAP_GRE6, "gre6-input" ) \
_(ENCAP_VXLANGPE, "vxlan-gpe-encap" ) \
_(ENCAP_VXLAN4, "vxlan4-encap" ) \
_(ENCAP_VXLAN6, "vxlan6-encap" ) \
_(DECAP_ETH_INPUT, "ethernet-input" ) \
_(ENCAP_LISP_GPE, "interface-output" ) \
_(ENCAP_ETHERNET, "nsh-eth-output") \
/* _(DECAP_IP4_INPUT, "ip4-input") \ */
/* _(DECAP_IP6_INPUT, "ip6-input" ) \ */
typedef enum {
#define _(s,n) NSH_NODE_NEXT_##s,
foreach_nsh_node_next
#undef _
NSH_NODE_N_NEXT,
} nsh_node_next_t;
typedef enum {
NSH_ACTION_SWAP,
NSH_ACTION_PUSH,
NSH_ACTION_POP,
} nsh_action_type;
typedef enum {
NSH_INPUT_TYPE,
NSH_PROXY_TYPE,
NSH_CLASSIFIER_TYPE,
NSH_AWARE_VNF_PROXY_TYPE,
} nsh_entity_type;
#define VNET_SW_INTERFACE_FLAG_ADMIN_DOWN 0
/* md2 class and type definition */
#define NSH_MD2_IOAM_CLASS 0x9
#define NSH_MD2_IOAM_OPTION_TYPE_TRACE 0x3B
#define NSH_MD2_IOAM_OPTION_TYPE_PROOF_OF_TRANSIT 0x3C
#define NSH_MD2_IOAM_TRACE_DUMMY_LEN 0x8
#define MAX_NSH_HEADER_LEN 256
#define MAX_NSH_OPTION_LEN 128
int
nsh_md2_register_option (u16 class,
u8 type,
u8 option_size,
int add_options (u8 * opt,
u8 * opt_size),
int options(vlib_buffer_t * b,
nsh_tlv_header_t * opt),
int swap_options (vlib_buffer_t * b,
nsh_tlv_header_t * old_opt,
nsh_tlv_header_t * new_opt),
int pop_options (vlib_buffer_t * b,
nsh_tlv_header_t * opt),
u8 * trace (u8 * s,
nsh_tlv_header_t * opt));
typedef struct _nsh_main_placeholder
{
u8 output_feature_arc_index;
} nsh_main_placeholder_t;
int
nsh_add_del_map (nsh_add_del_map_args_t * a, u32 * map_indexp);
int
nsh_add_del_proxy_session (nsh_add_del_map_args_t * a);
nsh_option_map_t *
nsh_md2_lookup_option (u16 class, u8 type);
int
nsh_add_del_entry (nsh_add_del_entry_args_t * a, u32 * entry_indexp);
u8 *
format_nsh_node_map_trace (u8 * s, va_list * args);
u8 *
format_nsh_header (u8 * s, va_list * args);
clib_error_t *
nsh_api_init (vlib_main_t * vm, nsh_main_t * nm);
#endif /* included_nsh_h */