return ret;
 }
 
+/* Must be called under rcu_read_lock() */
+static int
+dsa_slave_vlan_check_for_8021q_uppers(struct net_device *slave,
+                                     const struct switchdev_obj_port_vlan *vlan)
+{
+       struct net_device *upper_dev;
+       struct list_head *iter;
+
+       netdev_for_each_upper_dev_rcu(slave, upper_dev, iter) {
+               u16 vid;
+
+               if (!is_vlan_dev(upper_dev))
+                       continue;
+
+               vid = vlan_dev_vlan_id(upper_dev);
+               if (vid >= vlan->vid_begin && vid <= vlan->vid_end)
+                       return -EBUSY;
+       }
+
+       return 0;
+}
+
 static int dsa_slave_vlan_add(struct net_device *dev,
                              const struct switchdev_obj *obj,
                              struct switchdev_trans *trans)
 
        vlan = *SWITCHDEV_OBJ_PORT_VLAN(obj);
 
+       /* Deny adding a bridge VLAN when there is already an 802.1Q upper with
+        * the same VID.
+        */
+       if (trans->ph_prepare) {
+               rcu_read_lock();
+               err = dsa_slave_vlan_check_for_8021q_uppers(dev, &vlan);
+               rcu_read_unlock();
+               if (err)
+                       return err;
+       }
+
        err = dsa_port_vlan_add(dp, &vlan, trans);
        if (err)
                return err;
 
        return 0;
 }
 
-static int dsa_port_vlan_device_check(struct net_device *vlan_dev,
-                                     int vlan_dev_vid,
-                                     void *arg)
-{
-       struct switchdev_obj_port_vlan *vlan = arg;
-       u16 vid;
-
-       for (vid = vlan->vid_begin; vid <= vlan->vid_end; ++vid) {
-               if (vid == vlan_dev_vid)
-                       return -EBUSY;
-       }
-
-       return 0;
-}
-
-static int dsa_port_vlan_check(struct dsa_switch *ds, int port,
-                              const struct switchdev_obj_port_vlan *vlan)
-{
-       const struct dsa_port *dp = dsa_to_port(ds, port);
-       int err = 0;
-
-       /* Device is not bridged, let it proceed with the VLAN device
-        * creation.
-        */
-       if (!dp->bridge_dev)
-               return err;
-
-       /* dsa_slave_vlan_rx_{add,kill}_vid() cannot use the prepare phase and
-        * already checks whether there is an overlapping bridge VLAN entry
-        * with the same VID, so here we only need to check that if we are
-        * adding a bridge VLAN entry there is not an overlapping VLAN device
-        * claiming that VID.
-        */
-       return vlan_for_each(dp->slave, dsa_port_vlan_device_check,
-                            (void *)vlan);
-}
-
 static bool dsa_switch_vlan_match(struct dsa_switch *ds, int port,
                                  struct dsa_notifier_vlan_info *info)
 {
 
        for (port = 0; port < ds->num_ports; port++) {
                if (dsa_switch_vlan_match(ds, port, info)) {
-                       err = dsa_port_vlan_check(ds, port, info->vlan);
-                       if (err)
-                               return err;
-
                        err = ds->ops->port_vlan_prepare(ds, port, info->vlan);
                        if (err)
                                return err;