diff options
Diffstat (limited to 'drivers/net/dsa/b53/b53_common.c')
| -rw-r--r-- | drivers/net/dsa/b53/b53_common.c | 38 | 
1 files changed, 33 insertions, 5 deletions
| diff --git a/drivers/net/dsa/b53/b53_common.c b/drivers/net/dsa/b53/b53_common.c index 68e2381694b9..c283593bef17 100644 --- a/drivers/net/dsa/b53/b53_common.c +++ b/drivers/net/dsa/b53/b53_common.c @@ -1474,6 +1474,10 @@ static int b53_arl_rw_op(struct b53_device *dev, unsigned int op)  		reg |= ARLTBL_RW;  	else  		reg &= ~ARLTBL_RW; +	if (dev->vlan_enabled) +		reg &= ~ARLTBL_IVL_SVL_SELECT; +	else +		reg |= ARLTBL_IVL_SVL_SELECT;  	b53_write8(dev, B53_ARLIO_PAGE, B53_ARLTBL_RW_CTRL, reg);  	return b53_arl_op_wait(dev); @@ -1483,6 +1487,7 @@ static int b53_arl_read(struct b53_device *dev, u64 mac,  			u16 vid, struct b53_arl_entry *ent, u8 *idx,  			bool is_valid)  { +	DECLARE_BITMAP(free_bins, B53_ARLTBL_MAX_BIN_ENTRIES);  	unsigned int i;  	int ret; @@ -1490,6 +1495,8 @@ static int b53_arl_read(struct b53_device *dev, u64 mac,  	if (ret)  		return ret; +	bitmap_zero(free_bins, dev->num_arl_entries); +  	/* Read the bins */  	for (i = 0; i < dev->num_arl_entries; i++) {  		u64 mac_vid; @@ -1501,13 +1508,24 @@ static int b53_arl_read(struct b53_device *dev, u64 mac,  			   B53_ARLTBL_DATA_ENTRY(i), &fwd_entry);  		b53_arl_to_entry(ent, mac_vid, fwd_entry); -		if (!(fwd_entry & ARLTBL_VALID)) +		if (!(fwd_entry & ARLTBL_VALID)) { +			set_bit(i, free_bins);  			continue; +		}  		if ((mac_vid & ARLTBL_MAC_MASK) != mac)  			continue; +		if (dev->vlan_enabled && +		    ((mac_vid >> ARLTBL_VID_S) & ARLTBL_VID_MASK) != vid) +			continue;  		*idx = i; +		return 0;  	} +	if (bitmap_weight(free_bins, dev->num_arl_entries) == 0) +		return -ENOSPC; + +	*idx = find_first_bit(free_bins, dev->num_arl_entries); +  	return -ENOENT;  } @@ -1537,10 +1555,21 @@ static int b53_arl_op(struct b53_device *dev, int op, int port,  	if (op)  		return ret; -	/* We could not find a matching MAC, so reset to a new entry */ -	if (ret) { +	switch (ret) { +	case -ENOSPC: +		dev_dbg(dev->dev, "{%pM,%.4d} no space left in ARL\n", +			addr, vid); +		return is_valid ? ret : 0; +	case -ENOENT: +		/* We could not find a matching MAC, so reset to a new entry */ +		dev_dbg(dev->dev, "{%pM,%.4d} not found, using idx: %d\n", +			addr, vid, idx);  		fwd_entry = 0; -		idx = 1; +		break; +	default: +		dev_dbg(dev->dev, "{%pM,%.4d} found, using idx: %d\n", +			addr, vid, idx); +		break;  	}  	/* For multicast address, the port is a bitmask and the validity @@ -1558,7 +1587,6 @@ static int b53_arl_op(struct b53_device *dev, int op, int port,  		ent.is_valid = !!(ent.port);  	} -	ent.is_valid = is_valid;  	ent.vid = vid;  	ent.is_static = true;  	ent.is_age = false; | 
