Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 09dbf629 authored by Petr Machata's avatar Petr Machata Committed by David S. Miller
Browse files

mlxsw: spectrum_router: Update nexthop RIF on update



The function mlxsw_sp_nexthop_rif_update() walks the list of nexthops
associated with a RIF, and updates the corresponding entries in the
switch. It is used in particular when a tunnel underlay netdevice moves
to a different VRF, and all the nexthops are migrated over to a new RIF.
The problem is that each nexthop holds a reference to its RIF, and that
is not updated. So after the old RIF is gone, further activity on these
nexthops (such as downing the underlay netdevice) dereferences a
dangling pointer.

Fix the issue by updating rif of impacted nexthops before calling
mlxsw_sp_nexthop_rif_update().

Fixes: 0c5f1cd5 ("mlxsw: spectrum_router: Generalize __mlxsw_sp_ipip_entry_update_tunnel()")
Signed-off-by: default avatarPetr Machata <petrm@mellanox.com>
Reviewed-by: default avatarIdo Schimmel <idosch@mellanox.com>
Signed-off-by: default avatarJiri Pirko <jiri@mellanox.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent d97cda5f
Loading
Loading
Loading
Loading
+21 −7
Original line number Original line Diff line number Diff line
@@ -1370,8 +1370,9 @@ static void mlxsw_sp_netdevice_ipip_ol_down_event(struct mlxsw_sp *mlxsw_sp,
		mlxsw_sp_ipip_entry_ol_down_event(mlxsw_sp, ipip_entry);
		mlxsw_sp_ipip_entry_ol_down_event(mlxsw_sp, ipip_entry);
}
}


static void mlxsw_sp_nexthop_rif_update(struct mlxsw_sp *mlxsw_sp,
static void mlxsw_sp_nexthop_rif_migrate(struct mlxsw_sp *mlxsw_sp,
					struct mlxsw_sp_rif *rif);
					 struct mlxsw_sp_rif *old_rif,
					 struct mlxsw_sp_rif *new_rif);
static int
static int
mlxsw_sp_ipip_entry_ol_lb_update(struct mlxsw_sp *mlxsw_sp,
mlxsw_sp_ipip_entry_ol_lb_update(struct mlxsw_sp *mlxsw_sp,
				 struct mlxsw_sp_ipip_entry *ipip_entry,
				 struct mlxsw_sp_ipip_entry *ipip_entry,
@@ -1389,17 +1390,18 @@ mlxsw_sp_ipip_entry_ol_lb_update(struct mlxsw_sp *mlxsw_sp,
		return PTR_ERR(new_lb_rif);
		return PTR_ERR(new_lb_rif);
	ipip_entry->ol_lb = new_lb_rif;
	ipip_entry->ol_lb = new_lb_rif;


	if (keep_encap) {
	if (keep_encap)
		list_splice_init(&old_lb_rif->common.nexthop_list,
		mlxsw_sp_nexthop_rif_migrate(mlxsw_sp, &old_lb_rif->common,
				 &new_lb_rif->common.nexthop_list);
					     &new_lb_rif->common);
		mlxsw_sp_nexthop_rif_update(mlxsw_sp, &new_lb_rif->common);
	}


	mlxsw_sp_rif_destroy(&old_lb_rif->common);
	mlxsw_sp_rif_destroy(&old_lb_rif->common);


	return 0;
	return 0;
}
}


static void mlxsw_sp_nexthop_rif_update(struct mlxsw_sp *mlxsw_sp,
					struct mlxsw_sp_rif *rif);

/**
/**
 * Update the offload related to an IPIP entry. This always updates decap, and
 * Update the offload related to an IPIP entry. This always updates decap, and
 * in addition to that it also:
 * in addition to that it also:
@@ -3560,6 +3562,18 @@ static void mlxsw_sp_nexthop_rif_update(struct mlxsw_sp *mlxsw_sp,
	}
	}
}
}


static void mlxsw_sp_nexthop_rif_migrate(struct mlxsw_sp *mlxsw_sp,
					 struct mlxsw_sp_rif *old_rif,
					 struct mlxsw_sp_rif *new_rif)
{
	struct mlxsw_sp_nexthop *nh;

	list_splice_init(&old_rif->nexthop_list, &new_rif->nexthop_list);
	list_for_each_entry(nh, &new_rif->nexthop_list, rif_list_node)
		nh->rif = new_rif;
	mlxsw_sp_nexthop_rif_update(mlxsw_sp, new_rif);
}

static void mlxsw_sp_nexthop_rif_gone_sync(struct mlxsw_sp *mlxsw_sp,
static void mlxsw_sp_nexthop_rif_gone_sync(struct mlxsw_sp *mlxsw_sp,
					   struct mlxsw_sp_rif *rif)
					   struct mlxsw_sp_rif *rif)
{
{