962 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			962 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
 | 
						|
/* Copyright (c) 2017-2018 Mellanox Technologies. All rights reserved */
 | 
						|
 | 
						|
#include <linux/kernel.h>
 | 
						|
#include <linux/bitops.h>
 | 
						|
#include <linux/if_vlan.h>
 | 
						|
#include <linux/if_bridge.h>
 | 
						|
#include <linux/netdevice.h>
 | 
						|
#include <linux/rtnetlink.h>
 | 
						|
 | 
						|
#include "spectrum.h"
 | 
						|
#include "reg.h"
 | 
						|
 | 
						|
struct mlxsw_sp_fid_family;
 | 
						|
 | 
						|
struct mlxsw_sp_fid_core {
 | 
						|
	struct mlxsw_sp_fid_family *fid_family_arr[MLXSW_SP_FID_TYPE_MAX];
 | 
						|
	unsigned int *port_fid_mappings;
 | 
						|
};
 | 
						|
 | 
						|
struct mlxsw_sp_fid {
 | 
						|
	struct list_head list;
 | 
						|
	struct mlxsw_sp_rif *rif;
 | 
						|
	unsigned int ref_count;
 | 
						|
	u16 fid_index;
 | 
						|
	struct mlxsw_sp_fid_family *fid_family;
 | 
						|
};
 | 
						|
 | 
						|
struct mlxsw_sp_fid_8021q {
 | 
						|
	struct mlxsw_sp_fid common;
 | 
						|
	u16 vid;
 | 
						|
};
 | 
						|
 | 
						|
struct mlxsw_sp_fid_8021d {
 | 
						|
	struct mlxsw_sp_fid common;
 | 
						|
	int br_ifindex;
 | 
						|
};
 | 
						|
 | 
						|
struct mlxsw_sp_flood_table {
 | 
						|
	enum mlxsw_sp_flood_type packet_type;
 | 
						|
	enum mlxsw_reg_sfgc_bridge_type bridge_type;
 | 
						|
	enum mlxsw_flood_table_type table_type;
 | 
						|
	int table_index;
 | 
						|
};
 | 
						|
 | 
						|
struct mlxsw_sp_fid_ops {
 | 
						|
	void (*setup)(struct mlxsw_sp_fid *fid, const void *arg);
 | 
						|
	int (*configure)(struct mlxsw_sp_fid *fid);
 | 
						|
	void (*deconfigure)(struct mlxsw_sp_fid *fid);
 | 
						|
	int (*index_alloc)(struct mlxsw_sp_fid *fid, const void *arg,
 | 
						|
			   u16 *p_fid_index);
 | 
						|
	bool (*compare)(const struct mlxsw_sp_fid *fid,
 | 
						|
			const void *arg);
 | 
						|
	u16 (*flood_index)(const struct mlxsw_sp_fid *fid);
 | 
						|
	int (*port_vid_map)(struct mlxsw_sp_fid *fid,
 | 
						|
			    struct mlxsw_sp_port *port, u16 vid);
 | 
						|
	void (*port_vid_unmap)(struct mlxsw_sp_fid *fid,
 | 
						|
			       struct mlxsw_sp_port *port, u16 vid);
 | 
						|
};
 | 
						|
 | 
						|
struct mlxsw_sp_fid_family {
 | 
						|
	enum mlxsw_sp_fid_type type;
 | 
						|
	size_t fid_size;
 | 
						|
	u16 start_index;
 | 
						|
	u16 end_index;
 | 
						|
	struct list_head fids_list;
 | 
						|
	unsigned long *fids_bitmap;
 | 
						|
	const struct mlxsw_sp_flood_table *flood_tables;
 | 
						|
	int nr_flood_tables;
 | 
						|
	enum mlxsw_sp_rif_type rif_type;
 | 
						|
	const struct mlxsw_sp_fid_ops *ops;
 | 
						|
	struct mlxsw_sp *mlxsw_sp;
 | 
						|
};
 | 
						|
 | 
						|
static const int mlxsw_sp_sfgc_uc_packet_types[MLXSW_REG_SFGC_TYPE_MAX] = {
 | 
						|
	[MLXSW_REG_SFGC_TYPE_UNKNOWN_UNICAST]			= 1,
 | 
						|
};
 | 
						|
 | 
						|
static const int mlxsw_sp_sfgc_bc_packet_types[MLXSW_REG_SFGC_TYPE_MAX] = {
 | 
						|
	[MLXSW_REG_SFGC_TYPE_BROADCAST]				= 1,
 | 
						|
	[MLXSW_REG_SFGC_TYPE_UNREGISTERED_MULTICAST_NON_IP]	= 1,
 | 
						|
	[MLXSW_REG_SFGC_TYPE_IPV4_LINK_LOCAL]			= 1,
 | 
						|
	[MLXSW_REG_SFGC_TYPE_IPV6_ALL_HOST]			= 1,
 | 
						|
	[MLXSW_REG_SFGC_TYPE_UNREGISTERED_MULTICAST_IPV6]	= 1,
 | 
						|
};
 | 
						|
 | 
						|
static const int mlxsw_sp_sfgc_mc_packet_types[MLXSW_REG_SFGC_TYPE_MAX] = {
 | 
						|
	[MLXSW_REG_SFGC_TYPE_UNREGISTERED_MULTICAST_IPV4]	= 1,
 | 
						|
};
 | 
						|
 | 
						|
static const int *mlxsw_sp_packet_type_sfgc_types[] = {
 | 
						|
	[MLXSW_SP_FLOOD_TYPE_UC]	= mlxsw_sp_sfgc_uc_packet_types,
 | 
						|
	[MLXSW_SP_FLOOD_TYPE_BC]	= mlxsw_sp_sfgc_bc_packet_types,
 | 
						|
	[MLXSW_SP_FLOOD_TYPE_MC]	= mlxsw_sp_sfgc_mc_packet_types,
 | 
						|
};
 | 
						|
 | 
						|
static const struct mlxsw_sp_flood_table *
 | 
						|
mlxsw_sp_fid_flood_table_lookup(const struct mlxsw_sp_fid *fid,
 | 
						|
				enum mlxsw_sp_flood_type packet_type)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family = fid->fid_family;
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < fid_family->nr_flood_tables; i++) {
 | 
						|
		if (fid_family->flood_tables[i].packet_type != packet_type)
 | 
						|
			continue;
 | 
						|
		return &fid_family->flood_tables[i];
 | 
						|
	}
 | 
						|
 | 
						|
	return NULL;
 | 
						|
}
 | 
						|
 | 
						|
int mlxsw_sp_fid_flood_set(struct mlxsw_sp_fid *fid,
 | 
						|
			   enum mlxsw_sp_flood_type packet_type, u8 local_port,
 | 
						|
			   bool member)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family = fid->fid_family;
 | 
						|
	const struct mlxsw_sp_fid_ops *ops = fid_family->ops;
 | 
						|
	const struct mlxsw_sp_flood_table *flood_table;
 | 
						|
	char *sftr_pl;
 | 
						|
	int err;
 | 
						|
 | 
						|
	if (WARN_ON(!fid_family->flood_tables || !ops->flood_index))
 | 
						|
		return -EINVAL;
 | 
						|
 | 
						|
	flood_table = mlxsw_sp_fid_flood_table_lookup(fid, packet_type);
 | 
						|
	if (!flood_table)
 | 
						|
		return -ESRCH;
 | 
						|
 | 
						|
	sftr_pl = kmalloc(MLXSW_REG_SFTR_LEN, GFP_KERNEL);
 | 
						|
	if (!sftr_pl)
 | 
						|
		return -ENOMEM;
 | 
						|
 | 
						|
	mlxsw_reg_sftr_pack(sftr_pl, flood_table->table_index,
 | 
						|
			    ops->flood_index(fid), flood_table->table_type, 1,
 | 
						|
			    local_port, member);
 | 
						|
	err = mlxsw_reg_write(fid_family->mlxsw_sp->core, MLXSW_REG(sftr),
 | 
						|
			      sftr_pl);
 | 
						|
	kfree(sftr_pl);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
int mlxsw_sp_fid_port_vid_map(struct mlxsw_sp_fid *fid,
 | 
						|
			      struct mlxsw_sp_port *mlxsw_sp_port, u16 vid)
 | 
						|
{
 | 
						|
	if (WARN_ON(!fid->fid_family->ops->port_vid_map))
 | 
						|
		return -EINVAL;
 | 
						|
	return fid->fid_family->ops->port_vid_map(fid, mlxsw_sp_port, vid);
 | 
						|
}
 | 
						|
 | 
						|
void mlxsw_sp_fid_port_vid_unmap(struct mlxsw_sp_fid *fid,
 | 
						|
				 struct mlxsw_sp_port *mlxsw_sp_port, u16 vid)
 | 
						|
{
 | 
						|
	fid->fid_family->ops->port_vid_unmap(fid, mlxsw_sp_port, vid);
 | 
						|
}
 | 
						|
 | 
						|
enum mlxsw_sp_rif_type mlxsw_sp_fid_rif_type(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return fid->fid_family->rif_type;
 | 
						|
}
 | 
						|
 | 
						|
u16 mlxsw_sp_fid_index(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return fid->fid_index;
 | 
						|
}
 | 
						|
 | 
						|
enum mlxsw_sp_fid_type mlxsw_sp_fid_type(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return fid->fid_family->type;
 | 
						|
}
 | 
						|
 | 
						|
void mlxsw_sp_fid_rif_set(struct mlxsw_sp_fid *fid, struct mlxsw_sp_rif *rif)
 | 
						|
{
 | 
						|
	fid->rif = rif;
 | 
						|
}
 | 
						|
 | 
						|
enum mlxsw_sp_rif_type
 | 
						|
mlxsw_sp_fid_type_rif_type(const struct mlxsw_sp *mlxsw_sp,
 | 
						|
			   enum mlxsw_sp_fid_type type)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_core *fid_core = mlxsw_sp->fid_core;
 | 
						|
 | 
						|
	return fid_core->fid_family_arr[type]->rif_type;
 | 
						|
}
 | 
						|
 | 
						|
static struct mlxsw_sp_fid_8021q *
 | 
						|
mlxsw_sp_fid_8021q_fid(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return container_of(fid, struct mlxsw_sp_fid_8021q, common);
 | 
						|
}
 | 
						|
 | 
						|
u16 mlxsw_sp_fid_8021q_vid(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return mlxsw_sp_fid_8021q_fid(fid)->vid;
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_fid_8021q_setup(struct mlxsw_sp_fid *fid, const void *arg)
 | 
						|
{
 | 
						|
	u16 vid = *(u16 *) arg;
 | 
						|
 | 
						|
	mlxsw_sp_fid_8021q_fid(fid)->vid = vid;
 | 
						|
}
 | 
						|
 | 
						|
static enum mlxsw_reg_sfmr_op mlxsw_sp_sfmr_op(bool valid)
 | 
						|
{
 | 
						|
	return valid ? MLXSW_REG_SFMR_OP_CREATE_FID :
 | 
						|
		       MLXSW_REG_SFMR_OP_DESTROY_FID;
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_op(struct mlxsw_sp *mlxsw_sp, u16 fid_index,
 | 
						|
			   u16 fid_offset, bool valid)
 | 
						|
{
 | 
						|
	char sfmr_pl[MLXSW_REG_SFMR_LEN];
 | 
						|
 | 
						|
	mlxsw_reg_sfmr_pack(sfmr_pl, mlxsw_sp_sfmr_op(valid), fid_index,
 | 
						|
			    fid_offset);
 | 
						|
	return mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(sfmr), sfmr_pl);
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_vid_map(struct mlxsw_sp *mlxsw_sp, u16 fid_index,
 | 
						|
				u16 vid, bool valid)
 | 
						|
{
 | 
						|
	enum mlxsw_reg_svfa_mt mt = MLXSW_REG_SVFA_MT_VID_TO_FID;
 | 
						|
	char svfa_pl[MLXSW_REG_SVFA_LEN];
 | 
						|
 | 
						|
	mlxsw_reg_svfa_pack(svfa_pl, 0, mt, valid, fid_index, vid);
 | 
						|
	return mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(svfa), svfa_pl);
 | 
						|
}
 | 
						|
 | 
						|
static int __mlxsw_sp_fid_port_vid_map(struct mlxsw_sp *mlxsw_sp, u16 fid_index,
 | 
						|
				       u8 local_port, u16 vid, bool valid)
 | 
						|
{
 | 
						|
	enum mlxsw_reg_svfa_mt mt = MLXSW_REG_SVFA_MT_PORT_VID_TO_FID;
 | 
						|
	char svfa_pl[MLXSW_REG_SVFA_LEN];
 | 
						|
 | 
						|
	mlxsw_reg_svfa_pack(svfa_pl, local_port, mt, valid, fid_index, vid);
 | 
						|
	return mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(svfa), svfa_pl);
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_8021q_configure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = fid->fid_family->mlxsw_sp;
 | 
						|
	struct mlxsw_sp_fid_8021q *fid_8021q;
 | 
						|
	int err;
 | 
						|
 | 
						|
	err = mlxsw_sp_fid_op(mlxsw_sp, fid->fid_index, fid->fid_index, true);
 | 
						|
	if (err)
 | 
						|
		return err;
 | 
						|
 | 
						|
	fid_8021q = mlxsw_sp_fid_8021q_fid(fid);
 | 
						|
	err = mlxsw_sp_fid_vid_map(mlxsw_sp, fid->fid_index, fid_8021q->vid,
 | 
						|
				   true);
 | 
						|
	if (err)
 | 
						|
		goto err_fid_map;
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_fid_map:
 | 
						|
	mlxsw_sp_fid_op(mlxsw_sp, fid->fid_index, 0, false);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_fid_8021q_deconfigure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = fid->fid_family->mlxsw_sp;
 | 
						|
	struct mlxsw_sp_fid_8021q *fid_8021q;
 | 
						|
 | 
						|
	fid_8021q = mlxsw_sp_fid_8021q_fid(fid);
 | 
						|
	mlxsw_sp_fid_vid_map(mlxsw_sp, fid->fid_index, fid_8021q->vid, false);
 | 
						|
	mlxsw_sp_fid_op(mlxsw_sp, fid->fid_index, 0, false);
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_8021q_index_alloc(struct mlxsw_sp_fid *fid,
 | 
						|
					  const void *arg, u16 *p_fid_index)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family = fid->fid_family;
 | 
						|
	u16 vid = *(u16 *) arg;
 | 
						|
 | 
						|
	/* Use 1:1 mapping for simplicity although not a must */
 | 
						|
	if (vid < fid_family->start_index || vid > fid_family->end_index)
 | 
						|
		return -EINVAL;
 | 
						|
	*p_fid_index = vid;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static bool
 | 
						|
mlxsw_sp_fid_8021q_compare(const struct mlxsw_sp_fid *fid, const void *arg)
 | 
						|
{
 | 
						|
	u16 vid = *(u16 *) arg;
 | 
						|
 | 
						|
	return mlxsw_sp_fid_8021q_fid(fid)->vid == vid;
 | 
						|
}
 | 
						|
 | 
						|
static u16 mlxsw_sp_fid_8021q_flood_index(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return fid->fid_index;
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_8021q_port_vid_map(struct mlxsw_sp_fid *fid,
 | 
						|
					   struct mlxsw_sp_port *mlxsw_sp_port,
 | 
						|
					   u16 vid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	u8 local_port = mlxsw_sp_port->local_port;
 | 
						|
 | 
						|
	/* In case there are no {Port, VID} => FID mappings on the port,
 | 
						|
	 * we can use the global VID => FID mapping we created when the
 | 
						|
	 * FID was configured.
 | 
						|
	 */
 | 
						|
	if (mlxsw_sp->fid_core->port_fid_mappings[local_port] == 0)
 | 
						|
		return 0;
 | 
						|
	return __mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index, local_port,
 | 
						|
					   vid, true);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
mlxsw_sp_fid_8021q_port_vid_unmap(struct mlxsw_sp_fid *fid,
 | 
						|
				  struct mlxsw_sp_port *mlxsw_sp_port, u16 vid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	u8 local_port = mlxsw_sp_port->local_port;
 | 
						|
 | 
						|
	if (mlxsw_sp->fid_core->port_fid_mappings[local_port] == 0)
 | 
						|
		return;
 | 
						|
	__mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index, local_port, vid,
 | 
						|
				    false);
 | 
						|
}
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_ops mlxsw_sp_fid_8021q_ops = {
 | 
						|
	.setup			= mlxsw_sp_fid_8021q_setup,
 | 
						|
	.configure		= mlxsw_sp_fid_8021q_configure,
 | 
						|
	.deconfigure		= mlxsw_sp_fid_8021q_deconfigure,
 | 
						|
	.index_alloc		= mlxsw_sp_fid_8021q_index_alloc,
 | 
						|
	.compare		= mlxsw_sp_fid_8021q_compare,
 | 
						|
	.flood_index		= mlxsw_sp_fid_8021q_flood_index,
 | 
						|
	.port_vid_map		= mlxsw_sp_fid_8021q_port_vid_map,
 | 
						|
	.port_vid_unmap		= mlxsw_sp_fid_8021q_port_vid_unmap,
 | 
						|
};
 | 
						|
 | 
						|
static const struct mlxsw_sp_flood_table mlxsw_sp_fid_8021q_flood_tables[] = {
 | 
						|
	{
 | 
						|
		.packet_type	= MLXSW_SP_FLOOD_TYPE_UC,
 | 
						|
		.bridge_type	= MLXSW_REG_SFGC_BRIDGE_TYPE_1Q_FID,
 | 
						|
		.table_type	= MLXSW_REG_SFGC_TABLE_TYPE_FID_OFFSET,
 | 
						|
		.table_index	= 0,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.packet_type	= MLXSW_SP_FLOOD_TYPE_MC,
 | 
						|
		.bridge_type	= MLXSW_REG_SFGC_BRIDGE_TYPE_1Q_FID,
 | 
						|
		.table_type	= MLXSW_REG_SFGC_TABLE_TYPE_FID_OFFSET,
 | 
						|
		.table_index	= 1,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.packet_type	= MLXSW_SP_FLOOD_TYPE_BC,
 | 
						|
		.bridge_type	= MLXSW_REG_SFGC_BRIDGE_TYPE_1Q_FID,
 | 
						|
		.table_type	= MLXSW_REG_SFGC_TABLE_TYPE_FID_OFFSET,
 | 
						|
		.table_index	= 2,
 | 
						|
	},
 | 
						|
};
 | 
						|
 | 
						|
/* Range and flood configuration must match mlxsw_config_profile */
 | 
						|
static const struct mlxsw_sp_fid_family mlxsw_sp_fid_8021q_family = {
 | 
						|
	.type			= MLXSW_SP_FID_TYPE_8021Q,
 | 
						|
	.fid_size		= sizeof(struct mlxsw_sp_fid_8021q),
 | 
						|
	.start_index		= 1,
 | 
						|
	.end_index		= VLAN_VID_MASK,
 | 
						|
	.flood_tables		= mlxsw_sp_fid_8021q_flood_tables,
 | 
						|
	.nr_flood_tables	= ARRAY_SIZE(mlxsw_sp_fid_8021q_flood_tables),
 | 
						|
	.rif_type		= MLXSW_SP_RIF_TYPE_VLAN,
 | 
						|
	.ops			= &mlxsw_sp_fid_8021q_ops,
 | 
						|
};
 | 
						|
 | 
						|
static struct mlxsw_sp_fid_8021d *
 | 
						|
mlxsw_sp_fid_8021d_fid(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return container_of(fid, struct mlxsw_sp_fid_8021d, common);
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_fid_8021d_setup(struct mlxsw_sp_fid *fid, const void *arg)
 | 
						|
{
 | 
						|
	int br_ifindex = *(int *) arg;
 | 
						|
 | 
						|
	mlxsw_sp_fid_8021d_fid(fid)->br_ifindex = br_ifindex;
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_8021d_configure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family = fid->fid_family;
 | 
						|
 | 
						|
	return mlxsw_sp_fid_op(fid_family->mlxsw_sp, fid->fid_index, 0, true);
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_fid_8021d_deconfigure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	mlxsw_sp_fid_op(fid->fid_family->mlxsw_sp, fid->fid_index, 0, false);
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_8021d_index_alloc(struct mlxsw_sp_fid *fid,
 | 
						|
					  const void *arg, u16 *p_fid_index)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family = fid->fid_family;
 | 
						|
	u16 nr_fids, fid_index;
 | 
						|
 | 
						|
	nr_fids = fid_family->end_index - fid_family->start_index + 1;
 | 
						|
	fid_index = find_first_zero_bit(fid_family->fids_bitmap, nr_fids);
 | 
						|
	if (fid_index == nr_fids)
 | 
						|
		return -ENOBUFS;
 | 
						|
	*p_fid_index = fid_family->start_index + fid_index;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static bool
 | 
						|
mlxsw_sp_fid_8021d_compare(const struct mlxsw_sp_fid *fid, const void *arg)
 | 
						|
{
 | 
						|
	int br_ifindex = *(int *) arg;
 | 
						|
 | 
						|
	return mlxsw_sp_fid_8021d_fid(fid)->br_ifindex == br_ifindex;
 | 
						|
}
 | 
						|
 | 
						|
static u16 mlxsw_sp_fid_8021d_flood_index(const struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	return fid->fid_index - fid->fid_family->start_index;
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_port_vp_mode_trans(struct mlxsw_sp_port *mlxsw_sp_port)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	struct mlxsw_sp_port_vlan *mlxsw_sp_port_vlan;
 | 
						|
	int err;
 | 
						|
 | 
						|
	list_for_each_entry(mlxsw_sp_port_vlan, &mlxsw_sp_port->vlans_list,
 | 
						|
			    list) {
 | 
						|
		struct mlxsw_sp_fid *fid = mlxsw_sp_port_vlan->fid;
 | 
						|
		u16 vid = mlxsw_sp_port_vlan->vid;
 | 
						|
 | 
						|
		if (!fid)
 | 
						|
			continue;
 | 
						|
 | 
						|
		err = __mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index,
 | 
						|
						  mlxsw_sp_port->local_port,
 | 
						|
						  vid, true);
 | 
						|
		if (err)
 | 
						|
			goto err_fid_port_vid_map;
 | 
						|
	}
 | 
						|
 | 
						|
	err = mlxsw_sp_port_vp_mode_set(mlxsw_sp_port, true);
 | 
						|
	if (err)
 | 
						|
		goto err_port_vp_mode_set;
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_port_vp_mode_set:
 | 
						|
err_fid_port_vid_map:
 | 
						|
	list_for_each_entry_continue_reverse(mlxsw_sp_port_vlan,
 | 
						|
					     &mlxsw_sp_port->vlans_list, list) {
 | 
						|
		struct mlxsw_sp_fid *fid = mlxsw_sp_port_vlan->fid;
 | 
						|
		u16 vid = mlxsw_sp_port_vlan->vid;
 | 
						|
 | 
						|
		if (!fid)
 | 
						|
			continue;
 | 
						|
 | 
						|
		__mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index,
 | 
						|
					    mlxsw_sp_port->local_port, vid,
 | 
						|
					    false);
 | 
						|
	}
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_port_vlan_mode_trans(struct mlxsw_sp_port *mlxsw_sp_port)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	struct mlxsw_sp_port_vlan *mlxsw_sp_port_vlan;
 | 
						|
 | 
						|
	mlxsw_sp_port_vp_mode_set(mlxsw_sp_port, false);
 | 
						|
 | 
						|
	list_for_each_entry_reverse(mlxsw_sp_port_vlan,
 | 
						|
				    &mlxsw_sp_port->vlans_list, list) {
 | 
						|
		struct mlxsw_sp_fid *fid = mlxsw_sp_port_vlan->fid;
 | 
						|
		u16 vid = mlxsw_sp_port_vlan->vid;
 | 
						|
 | 
						|
		if (!fid)
 | 
						|
			continue;
 | 
						|
 | 
						|
		__mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index,
 | 
						|
					    mlxsw_sp_port->local_port, vid,
 | 
						|
					    false);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_8021d_port_vid_map(struct mlxsw_sp_fid *fid,
 | 
						|
					   struct mlxsw_sp_port *mlxsw_sp_port,
 | 
						|
					   u16 vid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	u8 local_port = mlxsw_sp_port->local_port;
 | 
						|
	int err;
 | 
						|
 | 
						|
	err = __mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index,
 | 
						|
					  mlxsw_sp_port->local_port, vid, true);
 | 
						|
	if (err)
 | 
						|
		return err;
 | 
						|
 | 
						|
	if (mlxsw_sp->fid_core->port_fid_mappings[local_port]++ == 0) {
 | 
						|
		err = mlxsw_sp_port_vp_mode_trans(mlxsw_sp_port);
 | 
						|
		if (err)
 | 
						|
			goto err_port_vp_mode_trans;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_port_vp_mode_trans:
 | 
						|
	mlxsw_sp->fid_core->port_fid_mappings[local_port]--;
 | 
						|
	__mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index,
 | 
						|
				    mlxsw_sp_port->local_port, vid, false);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
mlxsw_sp_fid_8021d_port_vid_unmap(struct mlxsw_sp_fid *fid,
 | 
						|
				  struct mlxsw_sp_port *mlxsw_sp_port, u16 vid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	u8 local_port = mlxsw_sp_port->local_port;
 | 
						|
 | 
						|
	if (mlxsw_sp->fid_core->port_fid_mappings[local_port] == 1)
 | 
						|
		mlxsw_sp_port_vlan_mode_trans(mlxsw_sp_port);
 | 
						|
	mlxsw_sp->fid_core->port_fid_mappings[local_port]--;
 | 
						|
	__mlxsw_sp_fid_port_vid_map(mlxsw_sp, fid->fid_index,
 | 
						|
				    mlxsw_sp_port->local_port, vid, false);
 | 
						|
}
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_ops mlxsw_sp_fid_8021d_ops = {
 | 
						|
	.setup			= mlxsw_sp_fid_8021d_setup,
 | 
						|
	.configure		= mlxsw_sp_fid_8021d_configure,
 | 
						|
	.deconfigure		= mlxsw_sp_fid_8021d_deconfigure,
 | 
						|
	.index_alloc		= mlxsw_sp_fid_8021d_index_alloc,
 | 
						|
	.compare		= mlxsw_sp_fid_8021d_compare,
 | 
						|
	.flood_index		= mlxsw_sp_fid_8021d_flood_index,
 | 
						|
	.port_vid_map		= mlxsw_sp_fid_8021d_port_vid_map,
 | 
						|
	.port_vid_unmap		= mlxsw_sp_fid_8021d_port_vid_unmap,
 | 
						|
};
 | 
						|
 | 
						|
static const struct mlxsw_sp_flood_table mlxsw_sp_fid_8021d_flood_tables[] = {
 | 
						|
	{
 | 
						|
		.packet_type	= MLXSW_SP_FLOOD_TYPE_UC,
 | 
						|
		.bridge_type	= MLXSW_REG_SFGC_BRIDGE_TYPE_VFID,
 | 
						|
		.table_type	= MLXSW_REG_SFGC_TABLE_TYPE_FID,
 | 
						|
		.table_index	= 0,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.packet_type	= MLXSW_SP_FLOOD_TYPE_MC,
 | 
						|
		.bridge_type	= MLXSW_REG_SFGC_BRIDGE_TYPE_VFID,
 | 
						|
		.table_type	= MLXSW_REG_SFGC_TABLE_TYPE_FID,
 | 
						|
		.table_index	= 1,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.packet_type	= MLXSW_SP_FLOOD_TYPE_BC,
 | 
						|
		.bridge_type	= MLXSW_REG_SFGC_BRIDGE_TYPE_VFID,
 | 
						|
		.table_type	= MLXSW_REG_SFGC_TABLE_TYPE_FID,
 | 
						|
		.table_index	= 2,
 | 
						|
	},
 | 
						|
};
 | 
						|
 | 
						|
/* Range and flood configuration must match mlxsw_config_profile */
 | 
						|
static const struct mlxsw_sp_fid_family mlxsw_sp_fid_8021d_family = {
 | 
						|
	.type			= MLXSW_SP_FID_TYPE_8021D,
 | 
						|
	.fid_size		= sizeof(struct mlxsw_sp_fid_8021d),
 | 
						|
	.start_index		= VLAN_N_VID,
 | 
						|
	.end_index		= VLAN_N_VID + MLXSW_SP_FID_8021D_MAX - 1,
 | 
						|
	.flood_tables		= mlxsw_sp_fid_8021d_flood_tables,
 | 
						|
	.nr_flood_tables	= ARRAY_SIZE(mlxsw_sp_fid_8021d_flood_tables),
 | 
						|
	.rif_type		= MLXSW_SP_RIF_TYPE_FID,
 | 
						|
	.ops			= &mlxsw_sp_fid_8021d_ops,
 | 
						|
};
 | 
						|
 | 
						|
static int mlxsw_sp_fid_rfid_configure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	/* rFIDs are allocated by the device during init */
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_fid_rfid_deconfigure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_rfid_index_alloc(struct mlxsw_sp_fid *fid,
 | 
						|
					 const void *arg, u16 *p_fid_index)
 | 
						|
{
 | 
						|
	u16 rif_index = *(u16 *) arg;
 | 
						|
 | 
						|
	*p_fid_index = fid->fid_family->start_index + rif_index;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static bool mlxsw_sp_fid_rfid_compare(const struct mlxsw_sp_fid *fid,
 | 
						|
				      const void *arg)
 | 
						|
{
 | 
						|
	u16 rif_index = *(u16 *) arg;
 | 
						|
 | 
						|
	return fid->fid_index == rif_index + fid->fid_family->start_index;
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_rfid_port_vid_map(struct mlxsw_sp_fid *fid,
 | 
						|
					  struct mlxsw_sp_port *mlxsw_sp_port,
 | 
						|
					  u16 vid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	u8 local_port = mlxsw_sp_port->local_port;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* We only need to transition the port to virtual mode since
 | 
						|
	 * {Port, VID} => FID is done by the firmware upon RIF creation.
 | 
						|
	 */
 | 
						|
	if (mlxsw_sp->fid_core->port_fid_mappings[local_port]++ == 0) {
 | 
						|
		err = mlxsw_sp_port_vp_mode_trans(mlxsw_sp_port);
 | 
						|
		if (err)
 | 
						|
			goto err_port_vp_mode_trans;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_port_vp_mode_trans:
 | 
						|
	mlxsw_sp->fid_core->port_fid_mappings[local_port]--;
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
mlxsw_sp_fid_rfid_port_vid_unmap(struct mlxsw_sp_fid *fid,
 | 
						|
				 struct mlxsw_sp_port *mlxsw_sp_port, u16 vid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
	u8 local_port = mlxsw_sp_port->local_port;
 | 
						|
 | 
						|
	if (mlxsw_sp->fid_core->port_fid_mappings[local_port] == 1)
 | 
						|
		mlxsw_sp_port_vlan_mode_trans(mlxsw_sp_port);
 | 
						|
	mlxsw_sp->fid_core->port_fid_mappings[local_port]--;
 | 
						|
}
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_ops mlxsw_sp_fid_rfid_ops = {
 | 
						|
	.configure		= mlxsw_sp_fid_rfid_configure,
 | 
						|
	.deconfigure		= mlxsw_sp_fid_rfid_deconfigure,
 | 
						|
	.index_alloc		= mlxsw_sp_fid_rfid_index_alloc,
 | 
						|
	.compare		= mlxsw_sp_fid_rfid_compare,
 | 
						|
	.port_vid_map		= mlxsw_sp_fid_rfid_port_vid_map,
 | 
						|
	.port_vid_unmap		= mlxsw_sp_fid_rfid_port_vid_unmap,
 | 
						|
};
 | 
						|
 | 
						|
#define MLXSW_SP_RFID_BASE	(15 * 1024)
 | 
						|
#define MLXSW_SP_RFID_MAX	1024
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_family mlxsw_sp_fid_rfid_family = {
 | 
						|
	.type			= MLXSW_SP_FID_TYPE_RFID,
 | 
						|
	.fid_size		= sizeof(struct mlxsw_sp_fid),
 | 
						|
	.start_index		= MLXSW_SP_RFID_BASE,
 | 
						|
	.end_index		= MLXSW_SP_RFID_BASE + MLXSW_SP_RFID_MAX - 1,
 | 
						|
	.rif_type		= MLXSW_SP_RIF_TYPE_SUBPORT,
 | 
						|
	.ops			= &mlxsw_sp_fid_rfid_ops,
 | 
						|
};
 | 
						|
 | 
						|
static int mlxsw_sp_fid_dummy_configure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = fid->fid_family->mlxsw_sp;
 | 
						|
 | 
						|
	return mlxsw_sp_fid_op(mlxsw_sp, fid->fid_index, 0, true);
 | 
						|
}
 | 
						|
 | 
						|
static void mlxsw_sp_fid_dummy_deconfigure(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	mlxsw_sp_fid_op(fid->fid_family->mlxsw_sp, fid->fid_index, 0, false);
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_dummy_index_alloc(struct mlxsw_sp_fid *fid,
 | 
						|
					  const void *arg, u16 *p_fid_index)
 | 
						|
{
 | 
						|
	*p_fid_index = fid->fid_family->start_index;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static bool mlxsw_sp_fid_dummy_compare(const struct mlxsw_sp_fid *fid,
 | 
						|
				       const void *arg)
 | 
						|
{
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_ops mlxsw_sp_fid_dummy_ops = {
 | 
						|
	.configure		= mlxsw_sp_fid_dummy_configure,
 | 
						|
	.deconfigure		= mlxsw_sp_fid_dummy_deconfigure,
 | 
						|
	.index_alloc		= mlxsw_sp_fid_dummy_index_alloc,
 | 
						|
	.compare		= mlxsw_sp_fid_dummy_compare,
 | 
						|
};
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_family mlxsw_sp_fid_dummy_family = {
 | 
						|
	.type			= MLXSW_SP_FID_TYPE_DUMMY,
 | 
						|
	.fid_size		= sizeof(struct mlxsw_sp_fid),
 | 
						|
	.start_index		= VLAN_N_VID - 1,
 | 
						|
	.end_index		= VLAN_N_VID - 1,
 | 
						|
	.ops			= &mlxsw_sp_fid_dummy_ops,
 | 
						|
};
 | 
						|
 | 
						|
static const struct mlxsw_sp_fid_family *mlxsw_sp_fid_family_arr[] = {
 | 
						|
	[MLXSW_SP_FID_TYPE_8021Q]	= &mlxsw_sp_fid_8021q_family,
 | 
						|
	[MLXSW_SP_FID_TYPE_8021D]	= &mlxsw_sp_fid_8021d_family,
 | 
						|
	[MLXSW_SP_FID_TYPE_RFID]	= &mlxsw_sp_fid_rfid_family,
 | 
						|
	[MLXSW_SP_FID_TYPE_DUMMY]	= &mlxsw_sp_fid_dummy_family,
 | 
						|
};
 | 
						|
 | 
						|
static struct mlxsw_sp_fid *mlxsw_sp_fid_get(struct mlxsw_sp *mlxsw_sp,
 | 
						|
					     enum mlxsw_sp_fid_type type,
 | 
						|
					     const void *arg)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family;
 | 
						|
	struct mlxsw_sp_fid *fid;
 | 
						|
	u16 fid_index;
 | 
						|
	int err;
 | 
						|
 | 
						|
	fid_family = mlxsw_sp->fid_core->fid_family_arr[type];
 | 
						|
	list_for_each_entry(fid, &fid_family->fids_list, list) {
 | 
						|
		if (!fid->fid_family->ops->compare(fid, arg))
 | 
						|
			continue;
 | 
						|
		fid->ref_count++;
 | 
						|
		return fid;
 | 
						|
	}
 | 
						|
 | 
						|
	fid = kzalloc(fid_family->fid_size, GFP_KERNEL);
 | 
						|
	if (!fid)
 | 
						|
		return ERR_PTR(-ENOMEM);
 | 
						|
	fid->fid_family = fid_family;
 | 
						|
 | 
						|
	err = fid->fid_family->ops->index_alloc(fid, arg, &fid_index);
 | 
						|
	if (err)
 | 
						|
		goto err_index_alloc;
 | 
						|
	fid->fid_index = fid_index;
 | 
						|
	__set_bit(fid_index - fid_family->start_index, fid_family->fids_bitmap);
 | 
						|
 | 
						|
	if (fid->fid_family->ops->setup)
 | 
						|
		fid->fid_family->ops->setup(fid, arg);
 | 
						|
 | 
						|
	err = fid->fid_family->ops->configure(fid);
 | 
						|
	if (err)
 | 
						|
		goto err_configure;
 | 
						|
 | 
						|
	list_add(&fid->list, &fid_family->fids_list);
 | 
						|
	fid->ref_count++;
 | 
						|
	return fid;
 | 
						|
 | 
						|
err_configure:
 | 
						|
	__clear_bit(fid_index - fid_family->start_index,
 | 
						|
		    fid_family->fids_bitmap);
 | 
						|
err_index_alloc:
 | 
						|
	kfree(fid);
 | 
						|
	return ERR_PTR(err);
 | 
						|
}
 | 
						|
 | 
						|
void mlxsw_sp_fid_put(struct mlxsw_sp_fid *fid)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_family *fid_family = fid->fid_family;
 | 
						|
 | 
						|
	if (--fid->ref_count == 1 && fid->rif) {
 | 
						|
		/* Destroy the associated RIF and let it drop the last
 | 
						|
		 * reference on the FID.
 | 
						|
		 */
 | 
						|
		return mlxsw_sp_rif_destroy(fid->rif);
 | 
						|
	} else if (fid->ref_count == 0) {
 | 
						|
		list_del(&fid->list);
 | 
						|
		fid->fid_family->ops->deconfigure(fid);
 | 
						|
		__clear_bit(fid->fid_index - fid_family->start_index,
 | 
						|
			    fid_family->fids_bitmap);
 | 
						|
		kfree(fid);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
struct mlxsw_sp_fid *mlxsw_sp_fid_8021q_get(struct mlxsw_sp *mlxsw_sp, u16 vid)
 | 
						|
{
 | 
						|
	return mlxsw_sp_fid_get(mlxsw_sp, MLXSW_SP_FID_TYPE_8021Q, &vid);
 | 
						|
}
 | 
						|
 | 
						|
struct mlxsw_sp_fid *mlxsw_sp_fid_8021d_get(struct mlxsw_sp *mlxsw_sp,
 | 
						|
					    int br_ifindex)
 | 
						|
{
 | 
						|
	return mlxsw_sp_fid_get(mlxsw_sp, MLXSW_SP_FID_TYPE_8021D, &br_ifindex);
 | 
						|
}
 | 
						|
 | 
						|
struct mlxsw_sp_fid *mlxsw_sp_fid_rfid_get(struct mlxsw_sp *mlxsw_sp,
 | 
						|
					   u16 rif_index)
 | 
						|
{
 | 
						|
	return mlxsw_sp_fid_get(mlxsw_sp, MLXSW_SP_FID_TYPE_RFID, &rif_index);
 | 
						|
}
 | 
						|
 | 
						|
struct mlxsw_sp_fid *mlxsw_sp_fid_dummy_get(struct mlxsw_sp *mlxsw_sp)
 | 
						|
{
 | 
						|
	return mlxsw_sp_fid_get(mlxsw_sp, MLXSW_SP_FID_TYPE_DUMMY, NULL);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
mlxsw_sp_fid_flood_table_init(struct mlxsw_sp_fid_family *fid_family,
 | 
						|
			      const struct mlxsw_sp_flood_table *flood_table)
 | 
						|
{
 | 
						|
	enum mlxsw_sp_flood_type packet_type = flood_table->packet_type;
 | 
						|
	const int *sfgc_packet_types;
 | 
						|
	int i;
 | 
						|
 | 
						|
	sfgc_packet_types = mlxsw_sp_packet_type_sfgc_types[packet_type];
 | 
						|
	for (i = 0; i < MLXSW_REG_SFGC_TYPE_MAX; i++) {
 | 
						|
		struct mlxsw_sp *mlxsw_sp = fid_family->mlxsw_sp;
 | 
						|
		char sfgc_pl[MLXSW_REG_SFGC_LEN];
 | 
						|
		int err;
 | 
						|
 | 
						|
		if (!sfgc_packet_types[i])
 | 
						|
			continue;
 | 
						|
		mlxsw_reg_sfgc_pack(sfgc_pl, i, flood_table->bridge_type,
 | 
						|
				    flood_table->table_type,
 | 
						|
				    flood_table->table_index);
 | 
						|
		err = mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(sfgc), sfgc_pl);
 | 
						|
		if (err)
 | 
						|
			return err;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
mlxsw_sp_fid_flood_tables_init(struct mlxsw_sp_fid_family *fid_family)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < fid_family->nr_flood_tables; i++) {
 | 
						|
		const struct mlxsw_sp_flood_table *flood_table;
 | 
						|
		int err;
 | 
						|
 | 
						|
		flood_table = &fid_family->flood_tables[i];
 | 
						|
		err = mlxsw_sp_fid_flood_table_init(fid_family, flood_table);
 | 
						|
		if (err)
 | 
						|
			return err;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int mlxsw_sp_fid_family_register(struct mlxsw_sp *mlxsw_sp,
 | 
						|
					const struct mlxsw_sp_fid_family *tmpl)
 | 
						|
{
 | 
						|
	u16 nr_fids = tmpl->end_index - tmpl->start_index + 1;
 | 
						|
	struct mlxsw_sp_fid_family *fid_family;
 | 
						|
	int err;
 | 
						|
 | 
						|
	fid_family = kmemdup(tmpl, sizeof(*fid_family), GFP_KERNEL);
 | 
						|
	if (!fid_family)
 | 
						|
		return -ENOMEM;
 | 
						|
 | 
						|
	fid_family->mlxsw_sp = mlxsw_sp;
 | 
						|
	INIT_LIST_HEAD(&fid_family->fids_list);
 | 
						|
	fid_family->fids_bitmap = kcalloc(BITS_TO_LONGS(nr_fids),
 | 
						|
					  sizeof(unsigned long), GFP_KERNEL);
 | 
						|
	if (!fid_family->fids_bitmap) {
 | 
						|
		err = -ENOMEM;
 | 
						|
		goto err_alloc_fids_bitmap;
 | 
						|
	}
 | 
						|
 | 
						|
	if (fid_family->flood_tables) {
 | 
						|
		err = mlxsw_sp_fid_flood_tables_init(fid_family);
 | 
						|
		if (err)
 | 
						|
			goto err_fid_flood_tables_init;
 | 
						|
	}
 | 
						|
 | 
						|
	mlxsw_sp->fid_core->fid_family_arr[tmpl->type] = fid_family;
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_fid_flood_tables_init:
 | 
						|
	kfree(fid_family->fids_bitmap);
 | 
						|
err_alloc_fids_bitmap:
 | 
						|
	kfree(fid_family);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
mlxsw_sp_fid_family_unregister(struct mlxsw_sp *mlxsw_sp,
 | 
						|
			       struct mlxsw_sp_fid_family *fid_family)
 | 
						|
{
 | 
						|
	mlxsw_sp->fid_core->fid_family_arr[fid_family->type] = NULL;
 | 
						|
	kfree(fid_family->fids_bitmap);
 | 
						|
	WARN_ON_ONCE(!list_empty(&fid_family->fids_list));
 | 
						|
	kfree(fid_family);
 | 
						|
}
 | 
						|
 | 
						|
int mlxsw_sp_port_fids_init(struct mlxsw_sp_port *mlxsw_sp_port)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
 | 
						|
	/* Track number of FIDs configured on the port with mapping type
 | 
						|
	 * PORT_VID_TO_FID, so that we know when to transition the port
 | 
						|
	 * back to non-virtual (VLAN) mode.
 | 
						|
	 */
 | 
						|
	mlxsw_sp->fid_core->port_fid_mappings[mlxsw_sp_port->local_port] = 0;
 | 
						|
 | 
						|
	return mlxsw_sp_port_vp_mode_set(mlxsw_sp_port, false);
 | 
						|
}
 | 
						|
 | 
						|
void mlxsw_sp_port_fids_fini(struct mlxsw_sp_port *mlxsw_sp_port)
 | 
						|
{
 | 
						|
	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | 
						|
 | 
						|
	mlxsw_sp->fid_core->port_fid_mappings[mlxsw_sp_port->local_port] = 0;
 | 
						|
}
 | 
						|
 | 
						|
int mlxsw_sp_fids_init(struct mlxsw_sp *mlxsw_sp)
 | 
						|
{
 | 
						|
	unsigned int max_ports = mlxsw_core_max_ports(mlxsw_sp->core);
 | 
						|
	struct mlxsw_sp_fid_core *fid_core;
 | 
						|
	int err, i;
 | 
						|
 | 
						|
	fid_core = kzalloc(sizeof(*mlxsw_sp->fid_core), GFP_KERNEL);
 | 
						|
	if (!fid_core)
 | 
						|
		return -ENOMEM;
 | 
						|
	mlxsw_sp->fid_core = fid_core;
 | 
						|
 | 
						|
	fid_core->port_fid_mappings = kcalloc(max_ports, sizeof(unsigned int),
 | 
						|
					      GFP_KERNEL);
 | 
						|
	if (!fid_core->port_fid_mappings) {
 | 
						|
		err = -ENOMEM;
 | 
						|
		goto err_alloc_port_fid_mappings;
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < MLXSW_SP_FID_TYPE_MAX; i++) {
 | 
						|
		err = mlxsw_sp_fid_family_register(mlxsw_sp,
 | 
						|
						   mlxsw_sp_fid_family_arr[i]);
 | 
						|
 | 
						|
		if (err)
 | 
						|
			goto err_fid_ops_register;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
 | 
						|
err_fid_ops_register:
 | 
						|
	for (i--; i >= 0; i--) {
 | 
						|
		struct mlxsw_sp_fid_family *fid_family;
 | 
						|
 | 
						|
		fid_family = fid_core->fid_family_arr[i];
 | 
						|
		mlxsw_sp_fid_family_unregister(mlxsw_sp, fid_family);
 | 
						|
	}
 | 
						|
	kfree(fid_core->port_fid_mappings);
 | 
						|
err_alloc_port_fid_mappings:
 | 
						|
	kfree(fid_core);
 | 
						|
	return err;
 | 
						|
}
 | 
						|
 | 
						|
void mlxsw_sp_fids_fini(struct mlxsw_sp *mlxsw_sp)
 | 
						|
{
 | 
						|
	struct mlxsw_sp_fid_core *fid_core = mlxsw_sp->fid_core;
 | 
						|
	int i;
 | 
						|
 | 
						|
	for (i = 0; i < MLXSW_SP_FID_TYPE_MAX; i++)
 | 
						|
		mlxsw_sp_fid_family_unregister(mlxsw_sp,
 | 
						|
					       fid_core->fid_family_arr[i]);
 | 
						|
	kfree(fid_core->port_fid_mappings);
 | 
						|
	kfree(fid_core);
 | 
						|
}
 |