Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Horatiu Vultur | 2852 | 99.82% | 17 | 94.44% |
Vladimir Oltean | 5 | 0.18% | 1 | 5.56% |
Total | 2857 | 18 |
// SPDX-License-Identifier: GPL-2.0+ #include <linux/if_bridge.h> #include <net/switchdev.h> #include "lan966x_main.h" static struct notifier_block lan966x_netdevice_nb __read_mostly; static void lan966x_port_set_mcast_ip_flood(struct lan966x_port *port, u32 pgid_ip) { struct lan966x *lan966x = port->lan966x; u32 flood_mask_ip; flood_mask_ip = lan_rd(lan966x, ANA_PGID(pgid_ip)); flood_mask_ip = ANA_PGID_PGID_GET(flood_mask_ip); /* If mcast snooping is not enabled then use mcast flood mask * to decide to enable multicast flooding or not. */ if (!port->mcast_ena) { u32 flood_mask; flood_mask = lan_rd(lan966x, ANA_PGID(PGID_MC)); flood_mask = ANA_PGID_PGID_GET(flood_mask); if (flood_mask & BIT(port->chip_port)) flood_mask_ip |= BIT(port->chip_port); else flood_mask_ip &= ~BIT(port->chip_port); } else { flood_mask_ip &= ~BIT(port->chip_port); } lan_rmw(ANA_PGID_PGID_SET(flood_mask_ip), ANA_PGID_PGID, lan966x, ANA_PGID(pgid_ip)); } static void lan966x_port_set_mcast_flood(struct lan966x_port *port, bool enabled) { u32 val = lan_rd(port->lan966x, ANA_PGID(PGID_MC)); val = ANA_PGID_PGID_GET(val); if (enabled) val |= BIT(port->chip_port); else val &= ~BIT(port->chip_port); lan_rmw(ANA_PGID_PGID_SET(val), ANA_PGID_PGID, port->lan966x, ANA_PGID(PGID_MC)); if (!port->mcast_ena) { lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV4); lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV6); } } static void lan966x_port_set_ucast_flood(struct lan966x_port *port, bool enabled) { u32 val = lan_rd(port->lan966x, ANA_PGID(PGID_UC)); val = ANA_PGID_PGID_GET(val); if (enabled) val |= BIT(port->chip_port); else val &= ~BIT(port->chip_port); lan_rmw(ANA_PGID_PGID_SET(val), ANA_PGID_PGID, port->lan966x, ANA_PGID(PGID_UC)); } static void lan966x_port_set_bcast_flood(struct lan966x_port *port, bool enabled) { u32 val = lan_rd(port->lan966x, ANA_PGID(PGID_BC)); val = ANA_PGID_PGID_GET(val); if (enabled) val |= BIT(port->chip_port); else val &= ~BIT(port->chip_port); lan_rmw(ANA_PGID_PGID_SET(val), ANA_PGID_PGID, port->lan966x, ANA_PGID(PGID_BC)); } static void lan966x_port_set_learning(struct lan966x_port *port, bool enabled) { lan_rmw(ANA_PORT_CFG_LEARN_ENA_SET(enabled), ANA_PORT_CFG_LEARN_ENA, port->lan966x, ANA_PORT_CFG(port->chip_port)); port->learn_ena = enabled; } static void lan966x_port_bridge_flags(struct lan966x_port *port, struct switchdev_brport_flags flags) { if (flags.mask & BR_MCAST_FLOOD) lan966x_port_set_mcast_flood(port, !!(flags.val & BR_MCAST_FLOOD)); if (flags.mask & BR_FLOOD) lan966x_port_set_ucast_flood(port, !!(flags.val & BR_FLOOD)); if (flags.mask & BR_BCAST_FLOOD) lan966x_port_set_bcast_flood(port, !!(flags.val & BR_BCAST_FLOOD)); if (flags.mask & BR_LEARNING) lan966x_port_set_learning(port, !!(flags.val & BR_LEARNING)); } static int lan966x_port_pre_bridge_flags(struct lan966x_port *port, struct switchdev_brport_flags flags) { if (flags.mask & ~(BR_MCAST_FLOOD | BR_FLOOD | BR_BCAST_FLOOD | BR_LEARNING)) return -EINVAL; return 0; } void lan966x_update_fwd_mask(struct lan966x *lan966x) { int i; for (i = 0; i < lan966x->num_phys_ports; i++) { struct lan966x_port *port = lan966x->ports[i]; unsigned long mask = 0; if (port && lan966x->bridge_fwd_mask & BIT(i)) { mask = lan966x->bridge_fwd_mask & ~BIT(i); if (port->bond) mask &= ~lan966x_lag_get_mask(lan966x, port->bond); } mask |= BIT(CPU_PORT); lan_wr(ANA_PGID_PGID_SET(mask), lan966x, ANA_PGID(PGID_SRC + i)); } } void lan966x_port_stp_state_set(struct lan966x_port *port, u8 state) { struct lan966x *lan966x = port->lan966x; bool learn_ena = false; if ((state == BR_STATE_FORWARDING || state == BR_STATE_LEARNING) && port->learn_ena) learn_ena = true; if (state == BR_STATE_FORWARDING) lan966x->bridge_fwd_mask |= BIT(port->chip_port); else lan966x->bridge_fwd_mask &= ~BIT(port->chip_port); lan_rmw(ANA_PORT_CFG_LEARN_ENA_SET(learn_ena), ANA_PORT_CFG_LEARN_ENA, lan966x, ANA_PORT_CFG(port->chip_port)); lan966x_update_fwd_mask(lan966x); } void lan966x_port_ageing_set(struct lan966x_port *port, unsigned long ageing_clock_t) { unsigned long ageing_jiffies = clock_t_to_jiffies(ageing_clock_t); u32 ageing_time = jiffies_to_msecs(ageing_jiffies) / 1000; lan966x_mac_set_ageing(port->lan966x, ageing_time); } static void lan966x_port_mc_set(struct lan966x_port *port, bool mcast_ena) { struct lan966x *lan966x = port->lan966x; port->mcast_ena = mcast_ena; if (mcast_ena) lan966x_mdb_restore_entries(lan966x); else lan966x_mdb_clear_entries(lan966x); lan_rmw(ANA_CPU_FWD_CFG_IGMP_REDIR_ENA_SET(mcast_ena) | ANA_CPU_FWD_CFG_MLD_REDIR_ENA_SET(mcast_ena) | ANA_CPU_FWD_CFG_IPMC_CTRL_COPY_ENA_SET(mcast_ena), ANA_CPU_FWD_CFG_IGMP_REDIR_ENA | ANA_CPU_FWD_CFG_MLD_REDIR_ENA | ANA_CPU_FWD_CFG_IPMC_CTRL_COPY_ENA, lan966x, ANA_CPU_FWD_CFG(port->chip_port)); lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV4); lan966x_port_set_mcast_ip_flood(port, PGID_MCIPV6); } static int lan966x_port_attr_set(struct net_device *dev, const void *ctx, const struct switchdev_attr *attr, struct netlink_ext_ack *extack) { struct lan966x_port *port = netdev_priv(dev); int err = 0; if (ctx && ctx != port) return 0; switch (attr->id) { case SWITCHDEV_ATTR_ID_PORT_BRIDGE_FLAGS: lan966x_port_bridge_flags(port, attr->u.brport_flags); break; case SWITCHDEV_ATTR_ID_PORT_PRE_BRIDGE_FLAGS: err = lan966x_port_pre_bridge_flags(port, attr->u.brport_flags); break; case SWITCHDEV_ATTR_ID_PORT_STP_STATE: lan966x_port_stp_state_set(port, attr->u.stp_state); break; case SWITCHDEV_ATTR_ID_BRIDGE_AGEING_TIME: lan966x_port_ageing_set(port, attr->u.ageing_time); break; case SWITCHDEV_ATTR_ID_BRIDGE_VLAN_FILTERING: lan966x_vlan_port_set_vlan_aware(port, attr->u.vlan_filtering); lan966x_vlan_port_apply(port); break; case SWITCHDEV_ATTR_ID_BRIDGE_MC_DISABLED: lan966x_port_mc_set(port, !attr->u.mc_disabled); break; default: err = -EOPNOTSUPP; break; } return err; } static int lan966x_port_bridge_join(struct lan966x_port *port, struct net_device *brport_dev, struct net_device *bridge, struct netlink_ext_ack *extack) { struct switchdev_brport_flags flags = {0}; struct lan966x *lan966x = port->lan966x; struct net_device *dev = port->dev; int err; if (!lan966x->bridge_mask) { lan966x->bridge = bridge; } else { if (lan966x->bridge != bridge) { NL_SET_ERR_MSG_MOD(extack, "Not allow to add port to different bridge"); return -ENODEV; } } err = switchdev_bridge_port_offload(brport_dev, dev, port, &lan966x_switchdev_nb, &lan966x_switchdev_blocking_nb, false, extack); if (err) return err; lan966x->bridge_mask |= BIT(port->chip_port); flags.mask = BR_LEARNING | BR_FLOOD | BR_MCAST_FLOOD | BR_BCAST_FLOOD; flags.val = flags.mask; lan966x_port_bridge_flags(port, flags); return 0; } static void lan966x_port_bridge_leave(struct lan966x_port *port, struct net_device *bridge) { struct switchdev_brport_flags flags = {0}; struct lan966x *lan966x = port->lan966x; flags.mask = BR_LEARNING | BR_FLOOD | BR_MCAST_FLOOD | BR_BCAST_FLOOD; flags.val = flags.mask & ~BR_LEARNING; lan966x_port_bridge_flags(port, flags); lan966x->bridge_mask &= ~BIT(port->chip_port); if (!lan966x->bridge_mask) lan966x->bridge = NULL; /* Set the port back to host mode */ lan966x_vlan_port_set_vlan_aware(port, false); lan966x_vlan_port_set_vid(port, HOST_PVID, false, false); lan966x_vlan_port_apply(port); } int lan966x_port_changeupper(struct net_device *dev, struct net_device *brport_dev, struct netdev_notifier_changeupper_info *info) { struct lan966x_port *port = netdev_priv(dev); struct netlink_ext_ack *extack; int err = 0; extack = netdev_notifier_info_to_extack(&info->info); if (netif_is_bridge_master(info->upper_dev)) { if (info->linking) err = lan966x_port_bridge_join(port, brport_dev, info->upper_dev, extack); else lan966x_port_bridge_leave(port, info->upper_dev); } if (netif_is_lag_master(info->upper_dev)) { if (info->linking) err = lan966x_lag_port_join(port, info->upper_dev, info->upper_dev, extack); else lan966x_lag_port_leave(port, info->upper_dev); } return err; } int lan966x_port_prechangeupper(struct net_device *dev, struct net_device *brport_dev, struct netdev_notifier_changeupper_info *info) { struct lan966x_port *port = netdev_priv(dev); int err = NOTIFY_DONE; if (netif_is_bridge_master(info->upper_dev) && !info->linking) { switchdev_bridge_port_unoffload(port->dev, port, NULL, NULL); lan966x_fdb_flush_workqueue(port->lan966x); } if (netif_is_lag_master(info->upper_dev)) { err = lan966x_lag_port_prechangeupper(dev, info); if (err || info->linking) return err; switchdev_bridge_port_unoffload(brport_dev, port, NULL, NULL); lan966x_fdb_flush_workqueue(port->lan966x); } return err; } static int lan966x_foreign_bridging_check(struct net_device *upper, bool *has_foreign, bool *seen_lan966x, struct netlink_ext_ack *extack) { struct lan966x *lan966x = NULL; struct net_device *dev; struct list_head *iter; if (!netif_is_bridge_master(upper) && !netif_is_lag_master(upper)) return 0; netdev_for_each_lower_dev(upper, dev, iter) { if (lan966x_netdevice_check(dev)) { struct lan966x_port *port = netdev_priv(dev); if (lan966x) { /* Upper already has at least one port of a * lan966x switch inside it, check that it's * the same instance of the driver. */ if (port->lan966x != lan966x) { NL_SET_ERR_MSG_MOD(extack, "Bridging between multiple lan966x switches disallowed"); return -EINVAL; } } else { /* This is the first lan966x port inside this * upper device */ lan966x = port->lan966x; *seen_lan966x = true; } } else if (netif_is_lag_master(dev)) { /* Allow to have bond interfaces that have only lan966x * devices */ if (lan966x_foreign_bridging_check(dev, has_foreign, seen_lan966x, extack)) return -EINVAL; } else { *has_foreign = true; } if (*seen_lan966x && *has_foreign) { NL_SET_ERR_MSG_MOD(extack, "Bridging lan966x ports with foreign interfaces disallowed"); return -EINVAL; } } return 0; } static int lan966x_bridge_check(struct net_device *dev, struct netdev_notifier_changeupper_info *info) { bool has_foreign = false; bool seen_lan966x = false; return lan966x_foreign_bridging_check(info->upper_dev, &has_foreign, &seen_lan966x, info->info.extack); } static int lan966x_netdevice_port_event(struct net_device *dev, struct notifier_block *nb, unsigned long event, void *ptr) { int err = 0; if (!lan966x_netdevice_check(dev)) { switch (event) { case NETDEV_CHANGEUPPER: case NETDEV_PRECHANGEUPPER: err = lan966x_bridge_check(dev, ptr); if (err) return err; if (netif_is_lag_master(dev)) { if (event == NETDEV_CHANGEUPPER) err = lan966x_lag_netdev_changeupper(dev, ptr); else err = lan966x_lag_netdev_prechangeupper(dev, ptr); return err; } break; default: return 0; } return 0; } switch (event) { case NETDEV_PRECHANGEUPPER: err = lan966x_port_prechangeupper(dev, dev, ptr); break; case NETDEV_CHANGEUPPER: err = lan966x_bridge_check(dev, ptr); if (err) return err; err = lan966x_port_changeupper(dev, dev, ptr); break; case NETDEV_CHANGELOWERSTATE: err = lan966x_lag_port_changelowerstate(dev, ptr); break; } return err; } static int lan966x_netdevice_event(struct notifier_block *nb, unsigned long event, void *ptr) { struct net_device *dev = netdev_notifier_info_to_dev(ptr); int ret; ret = lan966x_netdevice_port_event(dev, nb, event, ptr); return notifier_from_errno(ret); } static bool lan966x_foreign_dev_check(const struct net_device *dev, const struct net_device *foreign_dev) { struct lan966x_port *port = netdev_priv(dev); struct lan966x *lan966x = port->lan966x; int i; if (netif_is_bridge_master(foreign_dev)) if (lan966x->bridge == foreign_dev) return false; if (netif_is_lag_master(foreign_dev)) for (i = 0; i < lan966x->num_phys_ports; ++i) if (lan966x->ports[i] && lan966x->ports[i]->bond == foreign_dev) return false; return true; } static int lan966x_switchdev_event(struct notifier_block *nb, unsigned long event, void *ptr) { struct net_device *dev = switchdev_notifier_info_to_dev(ptr); int err; switch (event) { case SWITCHDEV_PORT_ATTR_SET: err = switchdev_handle_port_attr_set(dev, ptr, lan966x_netdevice_check, lan966x_port_attr_set); return notifier_from_errno(err); case SWITCHDEV_FDB_ADD_TO_DEVICE: case SWITCHDEV_FDB_DEL_TO_DEVICE: err = switchdev_handle_fdb_event_to_device(dev, event, ptr, lan966x_netdevice_check, lan966x_foreign_dev_check, lan966x_handle_fdb); return notifier_from_errno(err); } return NOTIFY_DONE; } static int lan966x_handle_port_vlan_add(struct lan966x_port *port, const struct switchdev_obj *obj) { const struct switchdev_obj_port_vlan *v = SWITCHDEV_OBJ_PORT_VLAN(obj); struct lan966x *lan966x = port->lan966x; if (!netif_is_bridge_master(obj->orig_dev)) lan966x_vlan_port_add_vlan(port, v->vid, v->flags & BRIDGE_VLAN_INFO_PVID, v->flags & BRIDGE_VLAN_INFO_UNTAGGED); else lan966x_vlan_cpu_add_vlan(lan966x, v->vid); return 0; } static int lan966x_handle_port_obj_add(struct net_device *dev, const void *ctx, const struct switchdev_obj *obj, struct netlink_ext_ack *extack) { struct lan966x_port *port = netdev_priv(dev); int err; if (ctx && ctx != port) return 0; switch (obj->id) { case SWITCHDEV_OBJ_ID_PORT_VLAN: err = lan966x_handle_port_vlan_add(port, obj); break; case SWITCHDEV_OBJ_ID_PORT_MDB: case SWITCHDEV_OBJ_ID_HOST_MDB: err = lan966x_handle_port_mdb_add(port, obj); break; default: err = -EOPNOTSUPP; break; } return err; } static int lan966x_handle_port_vlan_del(struct lan966x_port *port, const struct switchdev_obj *obj) { const struct switchdev_obj_port_vlan *v = SWITCHDEV_OBJ_PORT_VLAN(obj); struct lan966x *lan966x = port->lan966x; if (!netif_is_bridge_master(obj->orig_dev)) lan966x_vlan_port_del_vlan(port, v->vid); else lan966x_vlan_cpu_del_vlan(lan966x, v->vid); return 0; } static int lan966x_handle_port_obj_del(struct net_device *dev, const void *ctx, const struct switchdev_obj *obj) { struct lan966x_port *port = netdev_priv(dev); int err; if (ctx && ctx != port) return 0; switch (obj->id) { case SWITCHDEV_OBJ_ID_PORT_VLAN: err = lan966x_handle_port_vlan_del(port, obj); break; case SWITCHDEV_OBJ_ID_PORT_MDB: case SWITCHDEV_OBJ_ID_HOST_MDB: err = lan966x_handle_port_mdb_del(port, obj); break; default: err = -EOPNOTSUPP; break; } return err; } static int lan966x_switchdev_blocking_event(struct notifier_block *nb, unsigned long event, void *ptr) { struct net_device *dev = switchdev_notifier_info_to_dev(ptr); int err; switch (event) { case SWITCHDEV_PORT_OBJ_ADD: err = switchdev_handle_port_obj_add(dev, ptr, lan966x_netdevice_check, lan966x_handle_port_obj_add); return notifier_from_errno(err); case SWITCHDEV_PORT_OBJ_DEL: err = switchdev_handle_port_obj_del(dev, ptr, lan966x_netdevice_check, lan966x_handle_port_obj_del); return notifier_from_errno(err); case SWITCHDEV_PORT_ATTR_SET: err = switchdev_handle_port_attr_set(dev, ptr, lan966x_netdevice_check, lan966x_port_attr_set); return notifier_from_errno(err); } return NOTIFY_DONE; } static struct notifier_block lan966x_netdevice_nb __read_mostly = { .notifier_call = lan966x_netdevice_event, }; struct notifier_block lan966x_switchdev_nb __read_mostly = { .notifier_call = lan966x_switchdev_event, }; struct notifier_block lan966x_switchdev_blocking_nb __read_mostly = { .notifier_call = lan966x_switchdev_blocking_event, }; void lan966x_register_notifier_blocks(void) { register_netdevice_notifier(&lan966x_netdevice_nb); register_switchdev_notifier(&lan966x_switchdev_nb); register_switchdev_blocking_notifier(&lan966x_switchdev_blocking_nb); } void lan966x_unregister_notifier_blocks(void) { unregister_switchdev_blocking_notifier(&lan966x_switchdev_blocking_nb); unregister_switchdev_notifier(&lan966x_switchdev_nb); unregister_netdevice_notifier(&lan966x_netdevice_nb); }
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1