Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

team: Move team device type change at the end of team_port_add

Attempting to add a port device that is already up will expectedly fail,
but not before modifying the team device header_ops.

In the case of the syzbot reproducer the gre0 device is
already in state UP when it attempts to add it as a
port device of team0, this fails but before that
header_ops->create of team0 is changed from eth_header to ipgre_header
in the call to team_dev_type_check_change.

Later when we end up in ipgre_header() struct ip_tunnel* points to nonsense
as the private data of the device still holds a struct team.

Example sequence of iproute2 commands to reproduce the hang/BUG():
ip link add dev team0 type team
ip link add dev gre0 type gre
ip link set dev gre0 up
ip link set dev gre0 master team0
ip link set dev team0 up
ping -I team0 1.1.1.1

Move team_dev_type_check_change down where all other checks have passed
as it changes the dev type with no way to restore it in case
one of the checks that follow it fail.

Also make sure to preserve the origial mtu assignment:
- If port_dev is not the same type as dev, dev takes mtu from port_dev
- If port_dev is the same type as dev, port_dev takes mtu from dev

This is done by adding a conditional before the call to dev_set_mtu
to prevent it from assigning port_dev->mtu = dev->mtu and instead
letting team_dev_type_check_change assign dev->mtu = port_dev->mtu.
The conditional is needed because the patch moves the call to
team_dev_type_check_change past dev_set_mtu.

Testing:
- team device driver in-tree selftests
- Add/remove various devices as slaves of team device
- syzbot

Reported-by: syzbot+a2a3b519de727b0f7903@syzkaller.appspotmail.com
Closes: https://syzkaller.appspot.com/bug?extid=a2a3b519de727b0f7903
Fixes: 1d76efe1577b ("team: add support for non-ethernet devices")
Signed-off-by: Nikola Z. Ivanov <zlatistiv@gmail.com>
Reviewed-by: Jiri Pirko <jiri@nvidia.com>
Link: https://patch.msgid.link/20251122002027.695151-1-zlatistiv@gmail.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>

authored by

Nikola Z. Ivanov and committed by
Jakub Kicinski
0ae9cfc4 d2099d9f

+15 -8
+15 -8
drivers/net/team/team_core.c
··· 1191 1191 return -EPERM; 1192 1192 } 1193 1193 1194 - err = team_dev_type_check_change(dev, port_dev); 1195 - if (err) 1196 - return err; 1197 - 1198 1194 if (port_dev->flags & IFF_UP) { 1199 1195 NL_SET_ERR_MSG(extack, "Device is up. Set it down before adding it as a team port"); 1200 1196 netdev_err(dev, "Device %s is up. Set it down before adding it as a team port\n", ··· 1208 1212 INIT_LIST_HEAD(&port->qom_list); 1209 1213 1210 1214 port->orig.mtu = port_dev->mtu; 1211 - err = dev_set_mtu(port_dev, dev->mtu); 1212 - if (err) { 1213 - netdev_dbg(dev, "Error %d calling dev_set_mtu\n", err); 1214 - goto err_set_mtu; 1215 + /* 1216 + * MTU assignment will be handled in team_dev_type_check_change 1217 + * if dev and port_dev are of different types 1218 + */ 1219 + if (dev->type == port_dev->type) { 1220 + err = dev_set_mtu(port_dev, dev->mtu); 1221 + if (err) { 1222 + netdev_dbg(dev, "Error %d calling dev_set_mtu\n", err); 1223 + goto err_set_mtu; 1224 + } 1215 1225 } 1216 1226 1217 1227 memcpy(port->orig.dev_addr, port_dev->dev_addr, port_dev->addr_len); ··· 1292 1290 } 1293 1291 } 1294 1292 1293 + err = team_dev_type_check_change(dev, port_dev); 1294 + if (err) 1295 + goto err_set_dev_type; 1296 + 1295 1297 if (dev->flags & IFF_UP) { 1296 1298 netif_addr_lock_bh(dev); 1297 1299 dev_uc_sync_multiple(port_dev, dev); ··· 1314 1308 1315 1309 return 0; 1316 1310 1311 + err_set_dev_type: 1317 1312 err_set_slave_promisc: 1318 1313 __team_option_inst_del_port(team, port); 1319 1314