/* * Copyright (c) 2016 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. */ /** * @file * @brief LISP-GPE definitions. */ #ifndef included_vnet_lisp_gpe_h #define included_vnet_lisp_gpe_h #include #include #include #include #include #include #include #include #include #include #include /** IP4-UDP-LISP encap header */ /* *INDENT-OFF* */ typedef CLIB_PACKED (struct { ip4_header_t ip4; /* 20 bytes */ udp_header_t udp; /* 8 bytes */ lisp_gpe_header_t lisp; /* 8 bytes */ }) ip4_udp_lisp_gpe_header_t; /* *INDENT-ON* */ /** IP6-UDP-LISP encap header */ /* *INDENT-OFF* */ typedef CLIB_PACKED (struct { ip6_header_t ip6; /* 40 bytes */ udp_header_t udp; /* 8 bytes */ lisp_gpe_header_t lisp; /* 8 bytes */ }) ip6_udp_lisp_gpe_header_t; /* *INDENT-ON* */ /** LISP-GPE tunnel structure */ typedef struct { /** tunnel src and dst addresses */ locator_pair_t *locator_pairs; /** locator-pairs with best priority become sub-tunnels */ u32 *sub_tunnels; /** decap next index */ u32 decap_next_index; /* TODO remove */ ip_address_t src, dst; /** FIB indices */ u32 encap_fib_index; /* tunnel partner lookup here */ u32 decap_fib_index; /* inner IP lookup here */ /** index of the source address lookup FIB */ u32 src_fib_index; /** vnet intfc hw/sw_if_index */ u32 hw_if_index; u32 sw_if_index; /** L2 path-list */ fib_node_index_t l2_path_list; /** action for 'negative' tunnels */ u8 action; /** LISP header fields in HOST byte order */ u8 flags; u8 ver_res; u8 res; u8 next_protocol; u32 vni; } lisp_gpe_tunnel_t; /** * @brief A path on which to forward lisp traffic */ typedef struct lisp_fwd_path_t_ { /** * The adjacency constructed for the locator pair */ index_t lisp_adj; /** * Priority. Only the paths with the best priority will be installed in FIB */ u8 priority; /** * [UE]CMP weigt for the path */ u8 weight; } lisp_fwd_path_t; /** * @brief A Forwarding entry can be 'normal' or 'negative' * Negative implies we deliberately want to add a FIB entry for an EID * that results in 'spcial' behaviour determined by an 'action'. * @normal' means send it down some tunnels. */ typedef enum lisp_fwd_entry_type_t_ { LISP_FWD_ENTRY_TYPE_NORMAL, LISP_FWD_ENTRY_TYPE_NEGATIVE, } lisp_fwd_entry_type_t; typedef enum { NO_ACTION, FORWARD_NATIVE, SEND_MAP_REQUEST, DROP } negative_fwd_actions_e; /** * LISP-GPE fwd entry key */ typedef struct lisp_gpe_fwd_entry_key_t_ { dp_address_t rmt; dp_address_t lcl; u32 vni; } lisp_gpe_fwd_entry_key_t; /** * @brief A LISP Forwarding Entry * * A forwarding entry is from a locai EID to a remote EID over a set of rloc pairs */ typedef struct lisp_fwd_entry_t_ { /** * The Entry's key: {lEID,r-EID,vni} */ lisp_gpe_fwd_entry_key_t *key; /** * The VRF (in the case of L3) or Bridge-Domain (for L2) index */ union { u32 eid_table_id; u32 eid_bd_index; }; /** * The forwarding entry type */ lisp_fwd_entry_type_t type; union { /** * @brief When the type is 'normal' * The RLOC pair that form the route's paths. i.e. where to send * packets for this route. */ lisp_fwd_path_t *paths; /** * @brief When the type is negative. The action to take. */ negative_fwd_actions_e action; }; /** * The FIB index for the overlay, i.e. the FIB in which the EIDs * are present */ u32 eid_fib_index; /** * The SRC-FIB index for created for anding source-route entries */ u32 src_fib_index; } lisp_fwd_entry_t; #define foreach_lisp_gpe_ip_input_next \ _(DROP, "error-drop") \ _(IP4_INPUT, "ip4-input") \ _(IP6_INPUT, "ip6-input") \ _(L2_INPUT, "l2-input") /** Enum of possible next nodes post LISP-GPE decap */ typedef enum { #define _(s,n) LISP_GPE_INPUT_NEXT_##s, foreach_lisp_gpe_ip_input_next #undef _ LISP_GPE_INPUT_N_NEXT, } lisp_gpe_input_next_t; typedef enum { #define lisp_gpe_error(n,s) LISP_GPE_ERROR_##n, #include #undef lisp_gpe_error LISP_GPE_N_ERROR, } lisp_gpe_error_t; typedef struct tunnel_lookup { /** Lookup lisp-gpe interfaces by dp table (eg. vrf/bridge index) */ uword *hw_if_index_by_dp_table; /** lookup decap tunnel termination sw_if_index by vni and vice versa */ uword *sw_if_index_by_vni; // FIXME - Need this? uword *vni_by_sw_if_index; } tunnel_lookup_t; /** LISP-GPE global state*/ typedef struct lisp_gpe_main { /** pool of encap tunnel instances */ lisp_gpe_tunnel_t *tunnels; /** Free vlib hw_if_indices */ u32 *free_tunnel_hw_if_indices; u8 is_en; /* L3 data structures * ================== */ tunnel_lookup_t l3_ifaces; /* L2 data structures * ================== */ /** L2 LISP FIB */ BVT (clib_bihash) l2_fib; tunnel_lookup_t l2_ifaces; /** Load-balance for a miss in the table */ index_t l2_lb_miss; index_t l2_lb_cp_lkup; /** convenience */ vlib_main_t *vlib_main; vnet_main_t *vnet_main; ip4_main_t *im4; ip6_main_t *im6; ip_lookup_main_t *lm4; ip_lookup_main_t *lm6; } lisp_gpe_main_t; /** LISP-GPE global state*/ lisp_gpe_main_t lisp_gpe_main; always_inline lisp_gpe_main_t * vnet_lisp_gpe_get_main () { return &lisp_gpe_main; } extern vlib_node_registration_t lisp_gpe_ip4_input_node; extern vlib_node_registration_t lisp_gpe_ip6_input_node; extern vnet_hw_interface_class_t lisp_gpe_hw_class; u8 *format_lisp_gpe_header_with_length (u8 * s, va_list * args); /** Arguments to add an L2/L3 LISP-GPE interface*/ typedef struct { u8 is_add; union { /** vrf */ u32 table_id; /** bridge domain */ u16 bd_id; /** generic access */ u32 dp_table; }; u8 is_l2; /** virtual network identifier in host byte order */ u32 vni; } vnet_lisp_gpe_add_del_iface_args_t; /** Read LISP-GPE status */ u8 vnet_lisp_gpe_enable_disable_status (void); /** Add/del LISP-GPE interface. */ int vnet_lisp_gpe_add_del_iface (vnet_lisp_gpe_add_del_iface_args_t * a, u32 * hw_if_indexp); typedef struct { u8 is_en; } vnet_lisp_gpe_enable_disable_args_t; clib_error_t * vnet_lisp_gpe_enable_disable (vnet_lisp_gpe_enable_disable_args_t * a); /** */ typedef struct { u8 is_add; /** type of mapping */ u8 is_negative; /** action for negative mappings */ negative_fwd_actions_e action; /** local eid */ gid_address_t lcl_eid; /** remote eid */ gid_address_t rmt_eid; /** vector of locator pairs */ locator_pair_t *locator_pairs; /** FIB index to lookup remote locator at encap */ u32 encap_fib_index; /** FIB index to lookup inner IP at decap */ u32 decap_fib_index; /* TODO remove */ u32 decap_next_index; /** VNI/tenant id in HOST byte order */ u32 vni; /** vrf or bd where fwd entry should be inserted */ union { /** table (vrf) id */ u32 table_id; /** bridge domain id */ u16 bd_id; /** generic access */ u32 dp_table; }; } vnet_lisp_gpe_add_del_fwd_entry_args_t; int vnet_lisp_gpe_add_del_fwd_entry (vnet_lisp_gpe_add_del_fwd_entry_args_t * a, u32 * hw_if_indexp); extern void ip_src_fib_add_route (u32 src_fib_index, const ip_prefix_t * src_prefix, const lisp_fwd_path_t * paths); extern void ip_src_dst_fib_del_route (u32 src_fib_index, const ip_prefix_t * src_prefix, u32 dst_table_id, const ip_prefix_t * dst_prefix); extern void ip_src_fib_add_route_w_dpo (u32 src_fib_index, const ip_prefix_t * src_prefix, const dpo_id_t * src_dpo); extern u32 ip_dst_fib_add_route (u32 dst_table_id, const ip_prefix_t * dst_prefix); extern fib_route_path_t *lisp_gpe_mk_paths_for_sub_tunnels (lisp_gpe_tunnel_t * t); #define foreach_lgpe_ip4_lookup_next \ _(DROP, "error-drop") \ _(LISP_CP_LOOKUP, "lisp-cp-lookup") typedef enum lgpe_ip4_lookup_next { #define _(sym,str) LGPE_IP4_LOOKUP_NEXT_##sym, foreach_lgpe_ip4_lookup_next #undef _ LGPE_IP4_LOOKUP_N_NEXT, } lgpe_ip4_lookup_next_t; #define foreach_lgpe_ip6_lookup_next \ _(DROP, "error-drop") \ _(LISP_CP_LOOKUP, "lisp-cp-lookup") typedef enum lgpe_ip6_lookup_next { #define _(sym,str) LGPE_IP6_LOOKUP_NEXT_##sym, foreach_lgpe_ip6_lookup_next #undef _ LGPE_IP6_LOOKUP_N_NEXT, } lgpe_ip6_lookup_next_t; u8 *format_vnet_lisp_gpe_status (u8 * s, va_list * args); #define L2_FIB_DEFAULT_HASH_NUM_BUCKETS (64 * 1024) #define L2_FIB_DEFAULT_HASH_MEMORY_SIZE (32<<20) u32 lisp_l2_fib_lookup (lisp_gpe_main_t * lgm, u16 bd_index, u8 src_mac[8], u8 dst_mac[8]); #endif /* included_vnet_lisp_gpe_h */ /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */