mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-20 21:04:40 +08:00
Merge branch 'mlxsw-various-fixes'
Ido Schimmel says: ==================== mlxsw: Various fixes This patch set contains various fixes for mlxsw. Patch #1 ensures that only link modes that are supported by both the device and the driver are advertised. When a link mode that is not supported by the driver is negotiated by the device, it will be presented as an unknown speed by ethtool, causing the bond driver to wrongly assume that the link is down. Patch #2 fixes a trivial memory leak upon module removal. Patch #3 fixes a use-after-free that syzkaller was able to trigger once on a slow emulator after a few months of fuzzing. ==================== Link: https://lore.kernel.org/r/20201024133733.2107509-1-idosch@idosch.org Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
19c176eb07
@ -620,6 +620,9 @@ static void mlxsw_emad_transmit_retry(struct mlxsw_core *mlxsw_core,
|
||||
err = mlxsw_emad_transmit(trans->core, trans);
|
||||
if (err == 0)
|
||||
return;
|
||||
|
||||
if (!atomic_dec_and_test(&trans->active))
|
||||
return;
|
||||
} else {
|
||||
err = -EIO;
|
||||
}
|
||||
@ -2064,6 +2067,8 @@ void mlxsw_core_bus_device_unregister(struct mlxsw_core *mlxsw_core,
|
||||
if (!reload)
|
||||
devlink_resources_unregister(devlink, NULL);
|
||||
mlxsw_core->bus->fini(mlxsw_core->bus_priv);
|
||||
if (!reload)
|
||||
devlink_free(devlink);
|
||||
|
||||
return;
|
||||
|
||||
|
@ -1174,11 +1174,14 @@ mlxsw_sp_port_speed_by_width_set(struct mlxsw_sp_port *mlxsw_sp_port)
|
||||
u32 eth_proto_cap, eth_proto_admin, eth_proto_oper;
|
||||
const struct mlxsw_sp_port_type_speed_ops *ops;
|
||||
char ptys_pl[MLXSW_REG_PTYS_LEN];
|
||||
u32 eth_proto_cap_masked;
|
||||
int err;
|
||||
|
||||
ops = mlxsw_sp->port_type_speed_ops;
|
||||
|
||||
/* Set advertised speeds to supported speeds. */
|
||||
/* Set advertised speeds to speeds supported by both the driver
|
||||
* and the device.
|
||||
*/
|
||||
ops->reg_ptys_eth_pack(mlxsw_sp, ptys_pl, mlxsw_sp_port->local_port,
|
||||
0, false);
|
||||
err = mlxsw_reg_query(mlxsw_sp->core, MLXSW_REG(ptys), ptys_pl);
|
||||
@ -1187,8 +1190,10 @@ mlxsw_sp_port_speed_by_width_set(struct mlxsw_sp_port *mlxsw_sp_port)
|
||||
|
||||
ops->reg_ptys_eth_unpack(mlxsw_sp, ptys_pl, ð_proto_cap,
|
||||
ð_proto_admin, ð_proto_oper);
|
||||
eth_proto_cap_masked = ops->ptys_proto_cap_masked_get(eth_proto_cap);
|
||||
ops->reg_ptys_eth_pack(mlxsw_sp, ptys_pl, mlxsw_sp_port->local_port,
|
||||
eth_proto_cap, mlxsw_sp_port->link.autoneg);
|
||||
eth_proto_cap_masked,
|
||||
mlxsw_sp_port->link.autoneg);
|
||||
return mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(ptys), ptys_pl);
|
||||
}
|
||||
|
||||
|
@ -342,6 +342,7 @@ struct mlxsw_sp_port_type_speed_ops {
|
||||
u32 *p_eth_proto_cap,
|
||||
u32 *p_eth_proto_admin,
|
||||
u32 *p_eth_proto_oper);
|
||||
u32 (*ptys_proto_cap_masked_get)(u32 eth_proto_cap);
|
||||
};
|
||||
|
||||
static inline struct net_device *
|
||||
|
@ -1303,6 +1303,20 @@ mlxsw_sp1_reg_ptys_eth_unpack(struct mlxsw_sp *mlxsw_sp, char *payload,
|
||||
p_eth_proto_oper);
|
||||
}
|
||||
|
||||
static u32 mlxsw_sp1_ptys_proto_cap_masked_get(u32 eth_proto_cap)
|
||||
{
|
||||
u32 ptys_proto_cap_masked = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < MLXSW_SP1_PORT_LINK_MODE_LEN; i++) {
|
||||
if (mlxsw_sp1_port_link_mode[i].mask & eth_proto_cap)
|
||||
ptys_proto_cap_masked |=
|
||||
mlxsw_sp1_port_link_mode[i].mask;
|
||||
}
|
||||
|
||||
return ptys_proto_cap_masked;
|
||||
}
|
||||
|
||||
const struct mlxsw_sp_port_type_speed_ops mlxsw_sp1_port_type_speed_ops = {
|
||||
.from_ptys_supported_port = mlxsw_sp1_from_ptys_supported_port,
|
||||
.from_ptys_link = mlxsw_sp1_from_ptys_link,
|
||||
@ -1313,6 +1327,7 @@ const struct mlxsw_sp_port_type_speed_ops mlxsw_sp1_port_type_speed_ops = {
|
||||
.to_ptys_speed = mlxsw_sp1_to_ptys_speed,
|
||||
.reg_ptys_eth_pack = mlxsw_sp1_reg_ptys_eth_pack,
|
||||
.reg_ptys_eth_unpack = mlxsw_sp1_reg_ptys_eth_unpack,
|
||||
.ptys_proto_cap_masked_get = mlxsw_sp1_ptys_proto_cap_masked_get,
|
||||
};
|
||||
|
||||
static const enum ethtool_link_mode_bit_indices
|
||||
@ -1731,6 +1746,20 @@ mlxsw_sp2_reg_ptys_eth_unpack(struct mlxsw_sp *mlxsw_sp, char *payload,
|
||||
p_eth_proto_admin, p_eth_proto_oper);
|
||||
}
|
||||
|
||||
static u32 mlxsw_sp2_ptys_proto_cap_masked_get(u32 eth_proto_cap)
|
||||
{
|
||||
u32 ptys_proto_cap_masked = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; i < MLXSW_SP2_PORT_LINK_MODE_LEN; i++) {
|
||||
if (mlxsw_sp2_port_link_mode[i].mask & eth_proto_cap)
|
||||
ptys_proto_cap_masked |=
|
||||
mlxsw_sp2_port_link_mode[i].mask;
|
||||
}
|
||||
|
||||
return ptys_proto_cap_masked;
|
||||
}
|
||||
|
||||
const struct mlxsw_sp_port_type_speed_ops mlxsw_sp2_port_type_speed_ops = {
|
||||
.from_ptys_supported_port = mlxsw_sp2_from_ptys_supported_port,
|
||||
.from_ptys_link = mlxsw_sp2_from_ptys_link,
|
||||
@ -1741,4 +1770,5 @@ const struct mlxsw_sp_port_type_speed_ops mlxsw_sp2_port_type_speed_ops = {
|
||||
.to_ptys_speed = mlxsw_sp2_to_ptys_speed,
|
||||
.reg_ptys_eth_pack = mlxsw_sp2_reg_ptys_eth_pack,
|
||||
.reg_ptys_eth_unpack = mlxsw_sp2_reg_ptys_eth_unpack,
|
||||
.ptys_proto_cap_masked_get = mlxsw_sp2_ptys_proto_cap_masked_get,
|
||||
};
|
||||
|
Loading…
Reference in New Issue
Block a user