mirror of https://gitee.com/openkylin/linux.git
mlxsw: spectrum_router: Make nexthops typed
In the router, some next hops may reference an encapsulating netdevice, such as GRE or IPIP. To properly offload these next hops, mlxsw needs to keep track of whether a given next hop is a regular Ethernet entry, or an IP-in-IP tunneling entry. To facilitate this book-keeping, add a type field to struct mlxsw_sp_nexthop. There is, as of this patch, only one next hop type: MLXSW_SP_NEXTHOP_TYPE_ETH. Follow-up patches will introduce the IP-in-IP variant. There are several places where next hops are initialized in the IPv4 path. Instead of replicating the logic at every one of them, factor it out to a function mlxsw_sp_nexthop4_type_init(). The corresponding fini is actually protocol-neutral, so put it to mlxsw_sp_nexthop_type_fini(), but create a corresponding protocoled _fini function that dispatches to the protocol-neutral one. The IPv6 path is simpler, but for symmetry with IPv4, create the same suite of functions with corresponding logic. Signed-off-by: Petr Machata <petrm@mellanox.com> Reviewed-by: Ido Schimmel <idosch@mellanox.com> Signed-off-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
f6050ee6f4
commit
35225e4740
|
@ -1652,6 +1652,10 @@ static void mlxsw_sp_neigh_rif_gone_sync(struct mlxsw_sp *mlxsw_sp,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum mlxsw_sp_nexthop_type {
|
||||||
|
MLXSW_SP_NEXTHOP_TYPE_ETH,
|
||||||
|
};
|
||||||
|
|
||||||
struct mlxsw_sp_nexthop_key {
|
struct mlxsw_sp_nexthop_key {
|
||||||
struct fib_nh *fib_nh;
|
struct fib_nh *fib_nh;
|
||||||
};
|
};
|
||||||
|
@ -1676,7 +1680,10 @@ struct mlxsw_sp_nexthop {
|
||||||
update:1; /* set indicates that MAC of this neigh should be
|
update:1; /* set indicates that MAC of this neigh should be
|
||||||
* updated in HW
|
* updated in HW
|
||||||
*/
|
*/
|
||||||
struct mlxsw_sp_neigh_entry *neigh_entry;
|
enum mlxsw_sp_nexthop_type type;
|
||||||
|
union {
|
||||||
|
struct mlxsw_sp_neigh_entry *neigh_entry;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlxsw_sp_nexthop_group {
|
struct mlxsw_sp_nexthop_group {
|
||||||
|
@ -1964,9 +1971,9 @@ static int mlxsw_sp_nexthop_mac_update(struct mlxsw_sp *mlxsw_sp, u32 adj_index,
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
mlxsw_sp_nexthop_group_mac_update(struct mlxsw_sp *mlxsw_sp,
|
mlxsw_sp_nexthop_group_update(struct mlxsw_sp *mlxsw_sp,
|
||||||
struct mlxsw_sp_nexthop_group *nh_grp,
|
struct mlxsw_sp_nexthop_group *nh_grp,
|
||||||
bool reallocate)
|
bool reallocate)
|
||||||
{
|
{
|
||||||
u32 adj_index = nh_grp->adj_index; /* base */
|
u32 adj_index = nh_grp->adj_index; /* base */
|
||||||
struct mlxsw_sp_nexthop *nh;
|
struct mlxsw_sp_nexthop *nh;
|
||||||
|
@ -1982,8 +1989,12 @@ mlxsw_sp_nexthop_group_mac_update(struct mlxsw_sp *mlxsw_sp,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nh->update || reallocate) {
|
if (nh->update || reallocate) {
|
||||||
err = mlxsw_sp_nexthop_mac_update(mlxsw_sp,
|
switch (nh->type) {
|
||||||
adj_index, nh);
|
case MLXSW_SP_NEXTHOP_TYPE_ETH:
|
||||||
|
err = mlxsw_sp_nexthop_mac_update
|
||||||
|
(mlxsw_sp, adj_index, nh);
|
||||||
|
break;
|
||||||
|
}
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
nh->update = 0;
|
nh->update = 0;
|
||||||
|
@ -2071,8 +2082,7 @@ mlxsw_sp_nexthop_group_refresh(struct mlxsw_sp *mlxsw_sp,
|
||||||
/* Nothing was added or removed, so no need to reallocate. Just
|
/* Nothing was added or removed, so no need to reallocate. Just
|
||||||
* update MAC on existing adjacency indexes.
|
* update MAC on existing adjacency indexes.
|
||||||
*/
|
*/
|
||||||
err = mlxsw_sp_nexthop_group_mac_update(mlxsw_sp, nh_grp,
|
err = mlxsw_sp_nexthop_group_update(mlxsw_sp, nh_grp, false);
|
||||||
false);
|
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_warn(mlxsw_sp->bus_info->dev, "Failed to update neigh MAC in adjacency table.\n");
|
dev_warn(mlxsw_sp->bus_info->dev, "Failed to update neigh MAC in adjacency table.\n");
|
||||||
goto set_trap;
|
goto set_trap;
|
||||||
|
@ -2099,7 +2109,7 @@ mlxsw_sp_nexthop_group_refresh(struct mlxsw_sp *mlxsw_sp,
|
||||||
nh_grp->adj_index_valid = 1;
|
nh_grp->adj_index_valid = 1;
|
||||||
nh_grp->adj_index = adj_index;
|
nh_grp->adj_index = adj_index;
|
||||||
nh_grp->ecmp_size = ecmp_size;
|
nh_grp->ecmp_size = ecmp_size;
|
||||||
err = mlxsw_sp_nexthop_group_mac_update(mlxsw_sp, nh_grp, true);
|
err = mlxsw_sp_nexthop_group_update(mlxsw_sp, nh_grp, true);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_warn(mlxsw_sp->bus_info->dev, "Failed to update neigh MAC in adjacency table.\n");
|
dev_warn(mlxsw_sp->bus_info->dev, "Failed to update neigh MAC in adjacency table.\n");
|
||||||
goto set_trap;
|
goto set_trap;
|
||||||
|
@ -2287,6 +2297,48 @@ static bool mlxsw_sp_netdev_ipip_type(const struct mlxsw_sp *mlxsw_sp,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void mlxsw_sp_nexthop_type_fini(struct mlxsw_sp *mlxsw_sp,
|
||||||
|
struct mlxsw_sp_nexthop *nh)
|
||||||
|
{
|
||||||
|
switch (nh->type) {
|
||||||
|
case MLXSW_SP_NEXTHOP_TYPE_ETH:
|
||||||
|
mlxsw_sp_nexthop_neigh_fini(mlxsw_sp, nh);
|
||||||
|
mlxsw_sp_nexthop_rif_fini(nh);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlxsw_sp_nexthop4_type_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
|
struct mlxsw_sp_nexthop *nh,
|
||||||
|
struct fib_nh *fib_nh)
|
||||||
|
{
|
||||||
|
struct net_device *dev = fib_nh->nh_dev;
|
||||||
|
struct mlxsw_sp_rif *rif;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
nh->type = MLXSW_SP_NEXTHOP_TYPE_ETH;
|
||||||
|
rif = mlxsw_sp_rif_find_by_dev(mlxsw_sp, dev);
|
||||||
|
if (!rif)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
mlxsw_sp_nexthop_rif_init(nh, rif);
|
||||||
|
err = mlxsw_sp_nexthop_neigh_init(mlxsw_sp, nh);
|
||||||
|
if (err)
|
||||||
|
goto err_neigh_init;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err_neigh_init:
|
||||||
|
mlxsw_sp_nexthop_rif_fini(nh);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mlxsw_sp_nexthop4_type_fini(struct mlxsw_sp *mlxsw_sp,
|
||||||
|
struct mlxsw_sp_nexthop *nh)
|
||||||
|
{
|
||||||
|
mlxsw_sp_nexthop_type_fini(mlxsw_sp, nh);
|
||||||
|
}
|
||||||
|
|
||||||
static int mlxsw_sp_nexthop4_init(struct mlxsw_sp *mlxsw_sp,
|
static int mlxsw_sp_nexthop4_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
struct mlxsw_sp_nexthop_group *nh_grp,
|
struct mlxsw_sp_nexthop_group *nh_grp,
|
||||||
struct mlxsw_sp_nexthop *nh,
|
struct mlxsw_sp_nexthop *nh,
|
||||||
|
@ -2294,7 +2346,6 @@ static int mlxsw_sp_nexthop4_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
{
|
{
|
||||||
struct net_device *dev = fib_nh->nh_dev;
|
struct net_device *dev = fib_nh->nh_dev;
|
||||||
struct in_device *in_dev;
|
struct in_device *in_dev;
|
||||||
struct mlxsw_sp_rif *rif;
|
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
nh->nh_grp = nh_grp;
|
nh->nh_grp = nh_grp;
|
||||||
|
@ -2312,19 +2363,13 @@ static int mlxsw_sp_nexthop4_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
fib_nh->nh_flags & RTNH_F_LINKDOWN)
|
fib_nh->nh_flags & RTNH_F_LINKDOWN)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
rif = mlxsw_sp_rif_find_by_dev(mlxsw_sp, dev);
|
err = mlxsw_sp_nexthop4_type_init(mlxsw_sp, nh, fib_nh);
|
||||||
if (!rif)
|
|
||||||
return 0;
|
|
||||||
mlxsw_sp_nexthop_rif_init(nh, rif);
|
|
||||||
|
|
||||||
err = mlxsw_sp_nexthop_neigh_init(mlxsw_sp, nh);
|
|
||||||
if (err)
|
if (err)
|
||||||
goto err_nexthop_neigh_init;
|
goto err_nexthop_neigh_init;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_nexthop_neigh_init:
|
err_nexthop_neigh_init:
|
||||||
mlxsw_sp_nexthop_rif_fini(nh);
|
|
||||||
mlxsw_sp_nexthop_remove(mlxsw_sp, nh);
|
mlxsw_sp_nexthop_remove(mlxsw_sp, nh);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
@ -2332,8 +2377,7 @@ static int mlxsw_sp_nexthop4_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
static void mlxsw_sp_nexthop4_fini(struct mlxsw_sp *mlxsw_sp,
|
static void mlxsw_sp_nexthop4_fini(struct mlxsw_sp *mlxsw_sp,
|
||||||
struct mlxsw_sp_nexthop *nh)
|
struct mlxsw_sp_nexthop *nh)
|
||||||
{
|
{
|
||||||
mlxsw_sp_nexthop_neigh_fini(mlxsw_sp, nh);
|
mlxsw_sp_nexthop4_type_fini(mlxsw_sp, nh);
|
||||||
mlxsw_sp_nexthop_rif_fini(nh);
|
|
||||||
mlxsw_sp_nexthop_remove(mlxsw_sp, nh);
|
mlxsw_sp_nexthop_remove(mlxsw_sp, nh);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2342,7 +2386,6 @@ static void mlxsw_sp_nexthop4_event(struct mlxsw_sp *mlxsw_sp,
|
||||||
{
|
{
|
||||||
struct mlxsw_sp_nexthop_key key;
|
struct mlxsw_sp_nexthop_key key;
|
||||||
struct mlxsw_sp_nexthop *nh;
|
struct mlxsw_sp_nexthop *nh;
|
||||||
struct mlxsw_sp_rif *rif;
|
|
||||||
|
|
||||||
if (mlxsw_sp->router->aborted)
|
if (mlxsw_sp->router->aborted)
|
||||||
return;
|
return;
|
||||||
|
@ -2352,18 +2395,12 @@ static void mlxsw_sp_nexthop4_event(struct mlxsw_sp *mlxsw_sp,
|
||||||
if (WARN_ON_ONCE(!nh))
|
if (WARN_ON_ONCE(!nh))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
rif = mlxsw_sp_rif_find_by_dev(mlxsw_sp, fib_nh->nh_dev);
|
|
||||||
if (!rif)
|
|
||||||
return;
|
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case FIB_EVENT_NH_ADD:
|
case FIB_EVENT_NH_ADD:
|
||||||
mlxsw_sp_nexthop_rif_init(nh, rif);
|
mlxsw_sp_nexthop4_type_init(mlxsw_sp, nh, fib_nh);
|
||||||
mlxsw_sp_nexthop_neigh_init(mlxsw_sp, nh);
|
|
||||||
break;
|
break;
|
||||||
case FIB_EVENT_NH_DEL:
|
case FIB_EVENT_NH_DEL:
|
||||||
mlxsw_sp_nexthop_neigh_fini(mlxsw_sp, nh);
|
mlxsw_sp_nexthop4_type_fini(mlxsw_sp, nh);
|
||||||
mlxsw_sp_nexthop_rif_fini(nh);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2376,8 +2413,7 @@ static void mlxsw_sp_nexthop_rif_gone_sync(struct mlxsw_sp *mlxsw_sp,
|
||||||
struct mlxsw_sp_nexthop *nh, *tmp;
|
struct mlxsw_sp_nexthop *nh, *tmp;
|
||||||
|
|
||||||
list_for_each_entry_safe(nh, tmp, &rif->nexthop_list, rif_list_node) {
|
list_for_each_entry_safe(nh, tmp, &rif->nexthop_list, rif_list_node) {
|
||||||
mlxsw_sp_nexthop_neigh_fini(mlxsw_sp, nh);
|
mlxsw_sp_nexthop_type_fini(mlxsw_sp, nh);
|
||||||
mlxsw_sp_nexthop_rif_fini(nh);
|
|
||||||
mlxsw_sp_nexthop_group_refresh(mlxsw_sp, nh->nh_grp);
|
mlxsw_sp_nexthop_group_refresh(mlxsw_sp, nh->nh_grp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3487,22 +3523,16 @@ mlxsw_sp_fib6_entry_rt_find(const struct mlxsw_sp_fib6_entry *fib6_entry,
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlxsw_sp_nexthop6_init(struct mlxsw_sp *mlxsw_sp,
|
static int mlxsw_sp_nexthop6_type_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
struct mlxsw_sp_nexthop_group *nh_grp,
|
struct mlxsw_sp_nexthop_group *nh_grp,
|
||||||
struct mlxsw_sp_nexthop *nh,
|
struct mlxsw_sp_nexthop *nh,
|
||||||
const struct rt6_info *rt)
|
const struct rt6_info *rt)
|
||||||
{
|
{
|
||||||
struct net_device *dev = rt->dst.dev;
|
struct net_device *dev = rt->dst.dev;
|
||||||
struct mlxsw_sp_rif *rif;
|
struct mlxsw_sp_rif *rif;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
nh->nh_grp = nh_grp;
|
nh->type = MLXSW_SP_NEXTHOP_TYPE_ETH;
|
||||||
memcpy(&nh->gw_addr, &rt->rt6i_gateway, sizeof(nh->gw_addr));
|
|
||||||
|
|
||||||
if (!dev)
|
|
||||||
return 0;
|
|
||||||
nh->ifindex = dev->ifindex;
|
|
||||||
|
|
||||||
rif = mlxsw_sp_rif_find_by_dev(mlxsw_sp, dev);
|
rif = mlxsw_sp_rif_find_by_dev(mlxsw_sp, dev);
|
||||||
if (!rif)
|
if (!rif)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -3519,11 +3549,33 @@ static int mlxsw_sp_nexthop6_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void mlxsw_sp_nexthop6_type_fini(struct mlxsw_sp *mlxsw_sp,
|
||||||
|
struct mlxsw_sp_nexthop *nh)
|
||||||
|
{
|
||||||
|
mlxsw_sp_nexthop_type_fini(mlxsw_sp, nh);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mlxsw_sp_nexthop6_init(struct mlxsw_sp *mlxsw_sp,
|
||||||
|
struct mlxsw_sp_nexthop_group *nh_grp,
|
||||||
|
struct mlxsw_sp_nexthop *nh,
|
||||||
|
const struct rt6_info *rt)
|
||||||
|
{
|
||||||
|
struct net_device *dev = rt->dst.dev;
|
||||||
|
|
||||||
|
nh->nh_grp = nh_grp;
|
||||||
|
memcpy(&nh->gw_addr, &rt->rt6i_gateway, sizeof(nh->gw_addr));
|
||||||
|
|
||||||
|
if (!dev)
|
||||||
|
return 0;
|
||||||
|
nh->ifindex = dev->ifindex;
|
||||||
|
|
||||||
|
return mlxsw_sp_nexthop6_type_init(mlxsw_sp, nh_grp, nh, rt);
|
||||||
|
}
|
||||||
|
|
||||||
static void mlxsw_sp_nexthop6_fini(struct mlxsw_sp *mlxsw_sp,
|
static void mlxsw_sp_nexthop6_fini(struct mlxsw_sp *mlxsw_sp,
|
||||||
struct mlxsw_sp_nexthop *nh)
|
struct mlxsw_sp_nexthop *nh)
|
||||||
{
|
{
|
||||||
mlxsw_sp_nexthop_neigh_fini(mlxsw_sp, nh);
|
mlxsw_sp_nexthop6_type_fini(mlxsw_sp, nh);
|
||||||
mlxsw_sp_nexthop_rif_fini(nh);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool mlxsw_sp_rt6_is_gateway(const struct mlxsw_sp *mlxsw_sp,
|
static bool mlxsw_sp_rt6_is_gateway(const struct mlxsw_sp *mlxsw_sp,
|
||||||
|
|
Loading…
Reference in New Issue