L3 cross connect

- all packets input on interface X are load-balanced over the set of
paths provided.

Change-Id: Ic27cb88c4cd5d6d3462570632daff7a43d5a652d
Signed-off-by: Neale Ranns <nranns@cisco.com>
This commit is contained in:
Neale Ranns
2019-05-22 13:26:39 +00:00
committed by Damjan Marion
parent e6be702362
commit 59fa121f89
11 changed files with 1434 additions and 4 deletions
+26
View File
@@ -0,0 +1,26 @@
# Copyright (c) 2018 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.
add_vpp_plugin(l3xc
SOURCES
l3xc.c
l3xc_api.c
l3xc_node.c
API_FILES
l3xc.api
INSTALL_HEADERS
l3xc_all_api_h.h
l3xc_msg_enum.h
)
+98
View File
@@ -0,0 +1,98 @@
/* Hey Emacs use -*- mode: C -*- */
/*
* 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
This file defines the vpp control-plane API messages
used to control the L3XC plugin
*/
option version = "1.0.0";
import "vnet/fib/fib_types.api";
/** \brief Get the plugin version
@param client_index - opaque cookie to identify the sender
@param context - sender context, to match reply w/ request
*/
define l3xc_plugin_get_version
{
u32 client_index;
u32 context;
};
/** \brief Reply to get the plugin version
@param context - returned sender context, to match reply w/ request
@param major - Incremented every time a known breaking behavior change is introduced
@param minor - Incremented with small changes, may be used to avoid buggy versions
*/
define l3xc_plugin_get_version_reply
{
u32 context;
u32 major;
u32 minor;
};
/** \brief A description of an L3XC policy
@param input interface of the x-connect
@param n_paths Number of paths
@param paths The set of forwarding paths.
*/
typeonly define l3xc
{
u32 sw_if_index;
u8 is_ip6;
u8 n_paths;
vl_api_fib_path_t paths[n_paths];
};
define l3xc_update
{
u32 client_index;
u32 context;
vl_api_l3xc_t l3xc;
};
define l3xc_update_reply
{
u32 context;
i32 retval;
u32 stats_index;
};
autoreply define l3xc_del
{
u32 client_index;
u32 context;
u32 sw_if_index;
u8 is_ip6;
};
/** \brief Dump all L3XC policies
*/
define l3xc_dump
{
u32 client_index;
u32 context;
u32 sw_if_index;
};
/** \brief description returned in the dump
*/
define l3xc_details
{
u32 context;
vl_api_l3xc_t l3xc;
};
File diff suppressed because it is too large Load Diff
+118
View File
@@ -0,0 +1,118 @@
/*
* Copyright (c) 2019 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.
*/
/**
* A L3 cross connect will send all traffic that is received on the input
* interface to the [set of] paths requested.
* It is a much more memory efficient solution than using a separate IP table
* for each input interface and much faster than an ABF match all rule.
*/
#ifndef __L3XC_H__
#define __L3XC_H__
#include <vnet/fib/fib_node.h>
#define L3XC_PLUGIN_VERSION_MAJOR 1
#define L3XC_PLUGIN_VERSION_MINOR 0
/**
*/
typedef struct l3xc_t_
{
CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
/**
* Linkage into the FIB graph
*/
fib_node_t l3xc_node;
/**
* The path-list describing how to forward in case of a match
*/
fib_node_index_t l3xc_pl;
fib_protocol_t l3xc_proto;
/**
* Sibling index on the path-list
*/
u32 l3xc_sibling;
/**
* The input interface
*/
u32 l3xc_sw_if_index;
/**
* DPO for forwarding
*/
dpo_id_t l3xc_dpo;
} l3xc_t;
/**
* Create or update an L3XC Policy
*
* @param sw_if_index_index the input interface
* @param rpaths The set of paths to add to the forwarding set
* @return error code
*/
extern int l3xc_update (u32 sw_if_index,
u8 is_ip6, const fib_route_path_t * rpaths);
/**
* Delete an L3XC.
*
* @param sw_if_index_index the input interface
*/
extern int l3xc_delete (u32 sw_if_index, u8 is_ip6);
/**
* Callback function invoked during a walk of all policies
*/
typedef int (*l3xc_walk_cb_t) (index_t l3xci, void *ctx);
/**
* Walk/visit each of the L3XC policies
*/
extern void l3xc_walk (l3xc_walk_cb_t cb, void *ctx);
/**
* Find a L3 XC object from an interfce and FIB protocol
*/
extern index_t l3xc_find (u32 sw_if_index, fib_protocol_t fproto);
/**
* Data-plane functions
*/
extern l3xc_t *l3xc_pool;
static_always_inline l3xc_t *
l3xc_get (u32 index)
{
return (pool_elt_at_index (l3xc_pool, index));
}
extern vlib_node_registration_t l3xc_ip4_node;
extern vlib_node_registration_t l3xc_ip6_node;
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/
#endif
+16
View File
@@ -0,0 +1,16 @@
/*
* Copyright (c) 2017 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.
*/
/* Include the generated file, see BUILT_SOURCES in Makefile.am */
#include <l3xc/l3xc.api.h>
+296
View File
@@ -0,0 +1,296 @@
/*
* 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.
*/
#include <stddef.h>
#include <vnet/vnet.h>
#include <vnet/plugin/plugin.h>
#include <l3xc/l3xc.h>
#include <vnet/mpls/mpls_types.h>
#include <vnet/fib/fib_path_list.h>
#include <vnet/fib/fib_api.h>
#include <vpp/app/version.h>
#include <vlibapi/api.h>
#include <vlibmemory/api.h>
/* define message IDs */
#include <l3xc/l3xc_msg_enum.h>
/* define message structures */
#define vl_typedefs
#include <l3xc/l3xc_all_api_h.h>
#undef vl_typedefs
/* define generated endian-swappers */
#define vl_endianfun
#include <l3xc/l3xc_all_api_h.h>
#undef vl_endianfun
/* instantiate all the print functions we know about */
#define vl_print(handle, ...) vlib_cli_output (handle, __VA_ARGS__)
#define vl_printfun
#include <l3xc/l3xc_all_api_h.h>
#undef vl_printfun
/* Get the API version number */
#define vl_api_version(n,v) static u32 api_version=(v);
#include <l3xc/l3xc_all_api_h.h>
#undef vl_api_version
/**
* Base message ID fot the plugin
*/
static u32 l3xc_base_msg_id;
#include <vlibapi/api_helper_macros.h>
/* List of message types that this plugin understands */
#define foreach_l3xc_plugin_api_msg \
_(L3XC_PLUGIN_GET_VERSION, l3xc_plugin_get_version) \
_(L3XC_UPDATE, l3xc_update) \
_(L3XC_DEL, l3xc_del) \
_(L3XC_DUMP, l3xc_dump)
static void
vl_api_l3xc_plugin_get_version_t_handler (vl_api_l3xc_plugin_get_version_t *
mp)
{
vl_api_l3xc_plugin_get_version_reply_t *rmp;
vl_api_registration_t *rp;
rp = vl_api_client_index_to_registration (mp->client_index);
if (rp == 0)
return;
rmp = vl_msg_api_alloc (sizeof (*rmp));
rmp->_vl_msg_id =
ntohs (VL_API_L3XC_PLUGIN_GET_VERSION_REPLY + l3xc_base_msg_id);
rmp->context = mp->context;
rmp->major = htonl (L3XC_PLUGIN_VERSION_MAJOR);
rmp->minor = htonl (L3XC_PLUGIN_VERSION_MINOR);
vl_api_send_msg (rp, (u8 *) rmp);
}
static void
vl_api_l3xc_update_t_handler (vl_api_l3xc_update_t * mp)
{
vl_api_l3xc_update_reply_t *rmp;
fib_route_path_t *paths = NULL, *path;
int rv = 0;
u8 pi;
VALIDATE_SW_IF_INDEX (&mp->l3xc);
if (0 == mp->l3xc.n_paths)
{
rv = VNET_API_ERROR_INVALID_VALUE;
goto done;
}
vec_validate (paths, mp->l3xc.n_paths - 1);
for (pi = 0; pi < mp->l3xc.n_paths; pi++)
{
path = &paths[pi];
rv = fib_path_api_parse (&mp->l3xc.paths[pi], path);
if (0 != rv)
{
goto done;
}
}
rv = l3xc_update (ntohl (mp->l3xc.sw_if_index), mp->l3xc.is_ip6, paths);
done:
vec_free (paths);
BAD_SW_IF_INDEX_LABEL;
/* *INDENT-OFF* */
REPLY_MACRO2 (VL_API_L3XC_UPDATE_REPLY + l3xc_base_msg_id,
({
rmp->stats_index = 0;
}))
/* *INDENT-ON* */
}
static void
vl_api_l3xc_del_t_handler (vl_api_l3xc_del_t * mp)
{
vl_api_l3xc_del_reply_t *rmp;
int rv = 0;
VALIDATE_SW_IF_INDEX (mp);
rv = l3xc_delete (ntohl (mp->sw_if_index), mp->is_ip6);
BAD_SW_IF_INDEX_LABEL;
REPLY_MACRO (VL_API_L3XC_DEL_REPLY + l3xc_base_msg_id);
}
typedef struct l3xc_dump_walk_ctx_t_
{
vl_api_registration_t *rp;
u32 context;
} l3xc_dump_walk_ctx_t;
static int
l3xc_send_details (u32 l3xci, void *args)
{
fib_route_path_encode_t *api_rpaths = NULL, *api_rpath;
vl_api_l3xc_details_t *mp;
l3xc_dump_walk_ctx_t *ctx;
vl_api_fib_path_t *fp;
size_t msg_size;
l3xc_t *l3xc;
u8 n_paths;
ctx = args;
l3xc = l3xc_get (l3xci);
n_paths = fib_path_list_get_n_paths (l3xc->l3xc_pl);
msg_size = sizeof (*mp) + sizeof (mp->l3xc.paths[0]) * n_paths;
mp = vl_msg_api_alloc (msg_size);
clib_memset (mp, 0, msg_size);
mp->_vl_msg_id = ntohs (VL_API_L3XC_DETAILS + l3xc_base_msg_id);
/* fill in the message */
mp->context = ctx->context;
mp->l3xc.n_paths = n_paths;
mp->l3xc.sw_if_index = htonl (l3xc->l3xc_sw_if_index);
fib_path_list_walk_w_ext (l3xc->l3xc_pl, NULL, fib_path_encode,
&api_rpaths);
fp = mp->l3xc.paths;
vec_foreach (api_rpath, api_rpaths)
{
fib_api_path_encode (api_rpath, fp);
fp++;
}
vl_api_send_msg (ctx->rp, (u8 *) mp);
return (1);
}
static void
vl_api_l3xc_dump_t_handler (vl_api_l3xc_dump_t * mp)
{
vl_api_registration_t *rp;
u32 sw_if_index;
rp = vl_api_client_index_to_registration (mp->client_index);
if (rp == 0)
return;
l3xc_dump_walk_ctx_t ctx = {
.rp = rp,
.context = mp->context,
};
sw_if_index = ntohl (mp->sw_if_index);
if (~0 == sw_if_index)
l3xc_walk (l3xc_send_details, &ctx);
else
{
fib_protocol_t fproto;
index_t l3xci;
FOR_EACH_FIB_IP_PROTOCOL (fproto)
{
l3xci = l3xc_find (sw_if_index, fproto);
if (INDEX_INVALID != l3xci)
l3xc_send_details (l3xci, &ctx);
}
}
}
#define vl_msg_name_crc_list
#include <l3xc/l3xc_all_api_h.h>
#undef vl_msg_name_crc_list
/* Set up the API message handling tables */
static clib_error_t *
l3xc_plugin_api_hookup (vlib_main_t * vm)
{
#define _(N,n) \
vl_msg_api_set_handlers((VL_API_##N + l3xc_base_msg_id), \
#n, \
vl_api_##n##_t_handler, \
vl_noop_handler, \
vl_api_##n##_t_endian, \
vl_api_##n##_t_print, \
sizeof(vl_api_##n##_t), 1);
foreach_l3xc_plugin_api_msg;
#undef _
return 0;
}
static void
setup_message_id_table (api_main_t * apim)
{
#define _(id,n,crc) \
vl_msg_api_add_msg_name_crc (apim, #n "_" #crc, id + l3xc_base_msg_id);
foreach_vl_msg_name_crc_l3xc;
#undef _
}
static clib_error_t *
l3xc_api_init (vlib_main_t * vm)
{
clib_error_t *error = 0;
u8 *name = format (0, "l3xc_%08x%c", api_version, 0);
/* Ask for a correctly-sized block of API message decode slots */
l3xc_base_msg_id = vl_msg_api_get_msg_ids ((char *) name,
VL_MSG_FIRST_AVAILABLE);
error = l3xc_plugin_api_hookup (vm);
/* Add our API messages to the global name_crc hash table */
setup_message_id_table (&api_main);
vec_free (name);
return error;
}
VLIB_INIT_FUNCTION (l3xc_api_init);
/* *INDENT-OFF* */
VLIB_PLUGIN_REGISTER () = {
.version = VPP_BUILD_VER,
.description = "L3 Cross-Connect (L3XC)",
};
/* *INDENT-ON* */
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/
+19
View File
@@ -0,0 +1,19 @@
/*
* l3xc_error.def: L3XC errors
*
* Copyright (c) 2012 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.
*/
l3xc_error (NONE, "no match")
l3xc_error (MATCHED, "matched")
+28
View File
@@ -0,0 +1,28 @@
/*
* 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.
*/
#ifndef included_l3xc_msg_enum_h
#define included_l3xc_msg_enum_h
#include <vppinfra/byte_order.h>
#define vl_msg_id(n,h) n,
typedef enum {
#include <l3xc/l3xc_all_api_h.h>
/* We'll want to know how many messages IDs we need... */
VL_MSG_FIRST_AVAILABLE,
} vl_msg_id_t;
#undef vl_msg_id
#endif
+256
View File
@@ -0,0 +1,256 @@
/*
* Copyright (c) 2019 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.
*/
#include <plugins/l3xc/l3xc.h>
#include <vnet/feature/feature.h>
typedef enum l3xc_next_t_
{
L3XC_NEXT_DROP,
L3XC_N_NEXT,
} l3xc_next_t;
typedef struct l3xc_input_trace_t_
{
index_t l3xci;
index_t lbi;
} l3xc_input_trace_t;
typedef enum
{
#define l3xc_error(n,s) L3XC_ERROR_##n,
#include "l3xc_error.def"
#undef l3xc_error
L3XC_N_ERROR,
} l3xc_error_t;
always_inline uword
l3xc_input_inline (vlib_main_t * vm,
vlib_node_runtime_t * node,
vlib_frame_t * frame, fib_protocol_t fproto)
{
vlib_buffer_t *bufs[VLIB_FRAME_SIZE], **b;
u16 nexts[VLIB_FRAME_SIZE], *next;
u32 n_left, *from;
from = vlib_frame_vector_args (frame);
n_left = frame->n_vectors;
b = bufs;
next = nexts;
vlib_get_buffers (vm, from, bufs, n_left);
while (n_left >= 8)
{
const l3xc_t *l3xc0, *l3xc1, *l3xc2, *l3xc3;
u32 l3xci0, l3xci1, l3xci2, l3xci3;
u32 next_u32;
/* Prefetch next iteration. */
{
vlib_prefetch_buffer_header (b[4], LOAD);
vlib_prefetch_buffer_header (b[5], LOAD);
vlib_prefetch_buffer_header (b[6], LOAD);
vlib_prefetch_buffer_header (b[7], LOAD);
}
l3xci0 =
*(u32 *) vnet_feature_next_with_data (&next_u32, b[0],
sizeof (l3xci0));
l3xci1 =
*(u32 *) vnet_feature_next_with_data (&next_u32, b[1],
sizeof (l3xci1));
l3xci2 =
*(u32 *) vnet_feature_next_with_data (&next_u32, b[2],
sizeof (l3xci2));
l3xci3 =
*(u32 *) vnet_feature_next_with_data (&next_u32, b[3],
sizeof (l3xci3));
l3xc0 = l3xc_get (l3xci0);
l3xc1 = l3xc_get (l3xci1);
l3xc2 = l3xc_get (l3xci2);
l3xc3 = l3xc_get (l3xci3);
next[0] = l3xc0->l3xc_dpo.dpoi_next_node;
next[1] = l3xc1->l3xc_dpo.dpoi_next_node;
next[2] = l3xc2->l3xc_dpo.dpoi_next_node;
next[3] = l3xc3->l3xc_dpo.dpoi_next_node;
vnet_buffer (b[0])->ip.adj_index[VLIB_TX] = l3xc0->l3xc_dpo.dpoi_index;
vnet_buffer (b[1])->ip.adj_index[VLIB_TX] = l3xc1->l3xc_dpo.dpoi_index;
vnet_buffer (b[2])->ip.adj_index[VLIB_TX] = l3xc2->l3xc_dpo.dpoi_index;
vnet_buffer (b[3])->ip.adj_index[VLIB_TX] = l3xc3->l3xc_dpo.dpoi_index;
if (PREDICT_FALSE ((node->flags & VLIB_NODE_FLAG_TRACE)))
{
if (PREDICT_FALSE (b[0]->flags & VLIB_BUFFER_IS_TRACED))
{
l3xc_input_trace_t *tr;
tr = vlib_add_trace (vm, node, b[0], sizeof (*tr));
tr->l3xci = l3xci0;
tr->lbi = vnet_buffer (b[0])->ip.adj_index[VLIB_TX];
}
if (PREDICT_FALSE (b[1]->flags & VLIB_BUFFER_IS_TRACED))
{
l3xc_input_trace_t *tr;
tr = vlib_add_trace (vm, node, b[1], sizeof (*tr));
tr->l3xci = l3xci1;
tr->lbi = vnet_buffer (b[1])->ip.adj_index[VLIB_TX];
}
if (PREDICT_FALSE (b[2]->flags & VLIB_BUFFER_IS_TRACED))
{
l3xc_input_trace_t *tr;
tr = vlib_add_trace (vm, node, b[2], sizeof (*tr));
tr->l3xci = l3xci2;
tr->lbi = vnet_buffer (b[2])->ip.adj_index[VLIB_TX];
}
if (PREDICT_FALSE (b[3]->flags & VLIB_BUFFER_IS_TRACED))
{
l3xc_input_trace_t *tr;
tr = vlib_add_trace (vm, node, b[3], sizeof (*tr));
tr->l3xci = l3xci3;
tr->lbi = vnet_buffer (b[3])->ip.adj_index[VLIB_TX];
}
}
b += 4;
next += 4;
n_left -= 4;
}
while (n_left > 0)
{
u32 l3xci0, next_u32;
const l3xc_t *l3xc0;
l3xci0 =
*(u32 *) vnet_feature_next_with_data (&next_u32, b[0],
sizeof (l3xci0));
l3xc0 = l3xc_get (l3xci0);
next[0] = l3xc0->l3xc_dpo.dpoi_next_node;
vnet_buffer (b[0])->ip.adj_index[VLIB_TX] = l3xc0->l3xc_dpo.dpoi_index;
if (PREDICT_FALSE (b[0]->flags & VLIB_BUFFER_IS_TRACED))
{
l3xc_input_trace_t *tr;
tr = vlib_add_trace (vm, node, b[0], sizeof (*tr));
tr->l3xci = l3xci0;
tr->lbi = vnet_buffer (b[0])->ip.adj_index[VLIB_TX];
}
b += 1;
next += 1;
n_left -= 1;
}
vlib_buffer_enqueue_to_next (vm, node, from, nexts, frame->n_vectors);
return frame->n_vectors;
}
static uword
l3xc_input_ip4 (vlib_main_t * vm,
vlib_node_runtime_t * node, vlib_frame_t * frame)
{
return l3xc_input_inline (vm, node, frame, FIB_PROTOCOL_IP4);
}
static uword
l3xc_input_ip6 (vlib_main_t * vm,
vlib_node_runtime_t * node, vlib_frame_t * frame)
{
return l3xc_input_inline (vm, node, frame, FIB_PROTOCOL_IP6);
}
static u8 *
format_l3xc_input_trace (u8 * s, va_list * args)
{
CLIB_UNUSED (vlib_main_t * vm) = va_arg (*args, vlib_main_t *);
CLIB_UNUSED (vlib_node_t * node) = va_arg (*args, vlib_node_t *);
l3xc_input_trace_t *t = va_arg (*args, l3xc_input_trace_t *);
s = format (s, "l3xc-index:%d lb-index:%d", t->l3xci, t->lbi);
return s;
}
static char *l3xc_error_strings[] = {
#define l3xc_error(n,s) s,
#include "l3xc_error.def"
#undef l3xc_error
};
/* *INDENT-OFF* */
VLIB_REGISTER_NODE (l3xc_ip4_node) =
{
.function = l3xc_input_ip4,
.name = "l3xc-input-ip4",
.vector_size = sizeof (u32),
.format_trace = format_l3xc_input_trace,
.type = VLIB_NODE_TYPE_INTERNAL,
.n_errors = L3XC_N_ERROR,
.error_strings = l3xc_error_strings,
.n_next_nodes = L3XC_N_NEXT,
.next_nodes =
{
[L3XC_NEXT_DROP] = "error-drop",
}
};
VLIB_REGISTER_NODE (l3xc_ip6_node) =
{
.function = l3xc_input_ip6,
.name = "l3xc-input-ip6",
.vector_size = sizeof (u32),
.format_trace = format_l3xc_input_trace,
.type = VLIB_NODE_TYPE_INTERNAL,
.n_errors = 0,
.n_next_nodes = L3XC_N_NEXT,
.next_nodes =
{
[L3XC_NEXT_DROP] = "error-drop",
}
};
VNET_FEATURE_INIT (l3xc_ip4_feat, static) =
{
.arc_name = "ip4-unicast",
.node_name = "l3xc-input-ip4",
.runs_after = VNET_FEATURES ("acl-plugin-in-ip4-fa"),
};
VNET_FEATURE_INIT (l3xc_ip6_feat, static) =
{
.arc_name = "ip6-unicast",
.node_name = "l3xc-input-ip6",
.runs_after = VNET_FEATURES ("acl-plugin-in-ip6-fa"),
};
/* *INDENT-ON* */
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/
+8 -4
View File
@@ -25,6 +25,7 @@
#include <vnet/fib/fib_walk.h>
#include <vnet/fib/fib_urpf_list.h>
#include <vnet/fib/fib_path_ext.h>
#include <vnet/fib/fib_table.h>
/**
* The magic number of child entries that make a path-list popular.
@@ -364,10 +365,12 @@ fib_path_list_mk_lb (fib_path_list_t *path_list,
dpo_id_t *dpo,
fib_path_list_fwd_flags_t flags)
{
load_balance_path_t *nhs;
fib_node_index_t *path_index;
load_balance_path_t *nhs;
dpo_proto_t dproto;
nhs = NULL;
dproto = fib_forw_chain_type_to_dpo_proto(fct);
/*
* We gather the DPOs from resolved paths.
@@ -388,10 +391,11 @@ fib_path_list_mk_lb (fib_path_list_t *path_list,
*/
dpo_set(dpo,
DPO_LOAD_BALANCE,
fib_forw_chain_type_to_dpo_proto(fct),
dproto,
load_balance_create(vec_len(nhs),
fib_forw_chain_type_to_dpo_proto(fct),
0 /* FIXME FLOW HASH */));
dproto,
fib_table_get_default_flow_hash_config(
dpo_proto_to_fib(dproto))));
load_balance_multipath_update(dpo, nhs,
fib_path_list_fwd_flags_2_load_balance(flags));
+174
View File
@@ -0,0 +1,174 @@
#!/usr/bin/env python
from socket import inet_pton, inet_ntop, AF_INET, AF_INET6
import unittest
from framework import VppTestCase, VppTestRunner
from vpp_ip import DpoProto
from vpp_ip_route import VppIpRoute, VppRoutePath, VppMplsLabel, VppIpTable
from scapy.packet import Raw
from scapy.layers.l2 import Ether
from scapy.layers.inet import IP, UDP
from scapy.layers.inet6 import IPv6
from vpp_object import VppObject
NUM_PKTS = 67
def find_l3xc(test, sw_if_index, dump_sw_if_index=None):
if not dump_sw_if_index:
dump_sw_if_index = sw_if_index
xcs = test.vapi.l3xc_dump(dump_sw_if_index)
for xc in xcs:
if sw_if_index == xc.l3xc.sw_if_index:
return True
return False
class VppL3xc(VppObject):
def __init__(self, test, intf, paths, is_ip6=False):
self._test = test
self.intf = intf
self.is_ip6 = is_ip6
self.paths = paths
def encode_paths(self):
br_paths = []
for p in self.paths:
lstack = []
for l in p.nh_labels:
if type(l) == VppMplsLabel:
lstack.append(l.encode())
else:
lstack.append({'label': l, 'ttl': 255})
n_labels = len(lstack)
while (len(lstack) < 16):
lstack.append({})
br_paths.append({'next_hop': p.nh_addr,
'weight': 1,
'afi': p.proto,
'sw_if_index': p.nh_itf,
'preference': 0,
'table_id': p.nh_table_id,
'next_hop_id': p.next_hop_id,
'is_udp_encap': p.is_udp_encap,
'n_labels': n_labels,
'label_stack': lstack})
return br_paths
def add_vpp_config(self):
self._test.vapi.l3xc_update(
l3xc={
'is_ip6': self.is_ip6,
'sw_if_index': self.intf.sw_if_index,
'n_paths': len(self.paths),
'paths': self.encode_paths()
})
self._test.registry.register(self, self._test.logger)
def remove_vpp_config(self):
self._test.vapi.l3xc_del(
is_ip6=self.is_ip6,
sw_if_index=self.intf.sw_if_index)
def query_vpp_config(self):
return find_l3xc(self._test, self.intf.sw_if_index)
def object_id(self):
return ("l3xc-%d" % self.intf.sw_if_index)
class TestL3xc(VppTestCase):
""" L3XC Test Case """
@classmethod
def setUpClass(cls):
super(TestL3xc, cls).setUpClass()
@classmethod
def tearDownClass(cls):
super(TestL3xc, cls).tearDownClass()
def setUp(self):
super(TestL3xc, self).setUp()
self.create_pg_interfaces(range(6))
for i in self.pg_interfaces:
i.admin_up()
i.config_ip4()
i.resolve_arp()
i.config_ip6()
i.resolve_ndp()
def tearDown(self):
for i in self.pg_interfaces:
i.unconfig_ip4()
i.unconfig_ip6()
i.ip6_disable()
i.admin_down()
super(TestL3xc, self).tearDown()
def send_and_expect_load_balancing(self, input, pkts, outputs):
self.pg_send(input, pkts)
rxs = []
for oo in outputs:
rx = oo._get_capture(1)
self.assertNotEqual(0, len(rx))
for r in rx:
rxs.append(r)
return rxs
def test_l3xc4(self):
""" IPv4 X-Connect """
#
# x-connect pg0 to pg1 and pg2 to pg3->5
#
l3xc_1 = VppL3xc(self, self.pg0,
[VppRoutePath(self.pg1.remote_ip4,
self.pg1.sw_if_index)])
l3xc_1.add_vpp_config()
l3xc_2 = VppL3xc(self, self.pg2,
[VppRoutePath(self.pg3.remote_ip4,
self.pg3.sw_if_index),
VppRoutePath(self.pg4.remote_ip4,
self.pg4.sw_if_index),
VppRoutePath(self.pg5.remote_ip4,
self.pg5.sw_if_index)])
l3xc_2.add_vpp_config()
self.assertTrue(find_l3xc(self, self.pg2.sw_if_index, 0xffffffff))
self.logger.info(self.vapi.cli("sh l3xc"))
#
# fire in packets. If it's forwarded then the L3XC was successful,
# since default routing will drop it
#
p_1 = (Ether(src=self.pg0.remote_mac,
dst=self.pg0.local_mac) /
IP(src="1.1.1.1", dst="1.1.1.2") /
UDP(sport=1234, dport=1234) /
Raw('\xa5' * 100))
# self.send_and_expect(self.pg0, p_1*NUM_PKTS, self.pg1)
p_2 = []
for ii in range(NUM_PKTS):
p_2.append(Ether(src=self.pg0.remote_mac,
dst=self.pg0.local_mac) /
IP(src="1.1.1.1", dst="1.1.1.2") /
UDP(sport=1000 + ii, dport=1234) /
Raw('\xa5' * 100))
self.send_and_expect_load_balancing(self.pg2, p_2,
[self.pg3, self.pg4, self.pg5])
l3xc_2.remove_vpp_config()
self.send_and_assert_no_replies(self.pg2, p_2)
if __name__ == '__main__':
unittest.main(testRunner=VppTestRunner)