}
 
 int b53_vlan_add(struct dsa_switch *ds, int port,
-                const struct switchdev_obj_port_vlan *vlan)
+                const struct switchdev_obj_port_vlan *vlan,
+                struct netlink_ext_ack *extack)
 {
        struct b53_device *dev = ds->priv;
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
 
                             bool tx_pause, bool rx_pause);
 int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering);
 int b53_vlan_add(struct dsa_switch *ds, int port,
-                const struct switchdev_obj_port_vlan *vlan);
+                const struct switchdev_obj_port_vlan *vlan,
+                struct netlink_ext_ack *extack);
 int b53_vlan_del(struct dsa_switch *ds, int port,
                 const struct switchdev_obj_port_vlan *vlan);
 int b53_fdb_add(struct dsa_switch *ds, int port,
 
                else
                        vlan.flags = 0;
 
-               ret = ds->ops->port_vlan_add(ds, port_num, &vlan);
+               ret = ds->ops->port_vlan_add(ds, port_num, &vlan, NULL);
                if (ret)
                        return ret;
        }
 
 }
 
 static int dsa_loop_port_vlan_add(struct dsa_switch *ds, int port,
-                                 const struct switchdev_obj_port_vlan *vlan)
+                                 const struct switchdev_obj_port_vlan *vlan,
+                                 struct netlink_ext_ack *extack)
 {
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
        bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
 
 }
 
 static int hellcreek_vlan_prepare(struct dsa_switch *ds, int port,
-                                 const struct switchdev_obj_port_vlan *vlan)
+                                 const struct switchdev_obj_port_vlan *vlan,
+                                 struct netlink_ext_ack *extack)
 {
        struct hellcreek *hellcreek = ds->priv;
        int i;
                if (!dsa_is_user_port(ds, i))
                        continue;
 
-               if (vlan->vid == restricted_vid)
+               if (vlan->vid == restricted_vid) {
+                       NL_SET_ERR_MSG_MOD(extack, "VID restricted by driver");
                        return -EBUSY;
+               }
        }
 
        return 0;
 }
 
 static int hellcreek_vlan_add(struct dsa_switch *ds, int port,
-                             const struct switchdev_obj_port_vlan *vlan)
+                             const struct switchdev_obj_port_vlan *vlan,
+                             struct netlink_ext_ack *extack)
 {
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
        bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
        struct hellcreek *hellcreek = ds->priv;
        int err;
 
-       err = hellcreek_vlan_prepare(ds, port, vlan);
+       err = hellcreek_vlan_prepare(ds, port, vlan, extack);
        if (err)
                return err;
 
 
 }
 
 static int gswip_port_vlan_prepare(struct dsa_switch *ds, int port,
-                                  const struct switchdev_obj_port_vlan *vlan)
+                                  const struct switchdev_obj_port_vlan *vlan,
+                                  struct netlink_ext_ack *extack)
 {
        struct gswip_priv *priv = ds->priv;
        struct net_device *bridge = dsa_to_port(ds, port)->bridge_dev;
                        }
                }
 
-               if (idx == -1)
+               if (idx == -1) {
+                       NL_SET_ERR_MSG_MOD(extack, "No slot in VLAN table");
                        return -ENOSPC;
+               }
        }
 
        return 0;
 }
 
 static int gswip_port_vlan_add(struct dsa_switch *ds, int port,
-                              const struct switchdev_obj_port_vlan *vlan)
+                              const struct switchdev_obj_port_vlan *vlan,
+                              struct netlink_ext_ack *extack)
 {
        struct gswip_priv *priv = ds->priv;
        struct net_device *bridge = dsa_to_port(ds, port)->bridge_dev;
        bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
        int err;
 
-       err = gswip_port_vlan_prepare(ds, port, vlan);
+       err = gswip_port_vlan_prepare(ds, port, vlan, extack);
        if (err)
                return err;
 
 
 }
 
 static int ksz8795_port_vlan_add(struct dsa_switch *ds, int port,
-                                const struct switchdev_obj_port_vlan *vlan)
+                                const struct switchdev_obj_port_vlan *vlan,
+                                struct netlink_ext_ack *extack)
 {
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
        struct ksz_device *dev = ds->priv;
 
 }
 
 static int ksz9477_port_vlan_add(struct dsa_switch *ds, int port,
-                                const struct switchdev_obj_port_vlan *vlan)
+                                const struct switchdev_obj_port_vlan *vlan,
+                                struct netlink_ext_ack *extack)
 {
        struct ksz_device *dev = ds->priv;
        u32 vlan_table[3];
 
        err = ksz9477_get_vlan_table(dev, vlan->vid, vlan_table);
        if (err) {
-               dev_dbg(dev->dev, "Failed to get vlan table\n");
+               NL_SET_ERR_MSG_MOD(extack, "Failed to get vlan table");
                return err;
        }
 
 
        err = ksz9477_set_vlan_table(dev, vlan->vid, vlan_table);
        if (err) {
-               dev_dbg(dev->dev, "Failed to set vlan table\n");
+               NL_SET_ERR_MSG_MOD(extack, "Failed to set vlan table");
                return err;
        }
 
 
 
 static int
 mt7530_port_vlan_add(struct dsa_switch *ds, int port,
-                    const struct switchdev_obj_port_vlan *vlan)
+                    const struct switchdev_obj_port_vlan *vlan,
+                    struct netlink_ext_ack *extack)
 {
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
        bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
 
 }
 
 static int mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port,
-                                  const struct switchdev_obj_port_vlan *vlan)
+                                  const struct switchdev_obj_port_vlan *vlan,
+                                  struct netlink_ext_ack *extack)
 {
        struct mv88e6xxx_chip *chip = ds->priv;
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
 
 }
 
 static int felix_vlan_add(struct dsa_switch *ds, int port,
-                         const struct switchdev_obj_port_vlan *vlan)
+                         const struct switchdev_obj_port_vlan *vlan,
+                         struct netlink_ext_ack *extack)
 {
        struct ocelot *ocelot = ds->priv;
        u16 flags = vlan->flags;
 
 
 static int
 qca8k_port_vlan_add(struct dsa_switch *ds, int port,
-                   const struct switchdev_obj_port_vlan *vlan)
+                   const struct switchdev_obj_port_vlan *vlan,
+                   struct netlink_ext_ack *extack)
 {
        bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
        bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
 
 int rtl8366_vlan_filtering(struct dsa_switch *ds, int port,
                           bool vlan_filtering);
 int rtl8366_vlan_add(struct dsa_switch *ds, int port,
-                    const struct switchdev_obj_port_vlan *vlan);
+                    const struct switchdev_obj_port_vlan *vlan,
+                    struct netlink_ext_ack *extack);
 int rtl8366_vlan_del(struct dsa_switch *ds, int port,
                     const struct switchdev_obj_port_vlan *vlan);
 void rtl8366_get_strings(struct dsa_switch *ds, int port, u32 stringset,
 
 EXPORT_SYMBOL_GPL(rtl8366_vlan_filtering);
 
 int rtl8366_vlan_add(struct dsa_switch *ds, int port,
-                    const struct switchdev_obj_port_vlan *vlan)
+                    const struct switchdev_obj_port_vlan *vlan,
+                    struct netlink_ext_ack *extack)
 {
        bool untagged = !!(vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED);
        bool pvid = !!(vlan->flags & BRIDGE_VLAN_INFO_PVID);
        u32 untag = 0;
        int ret;
 
-       if (!smi->ops->is_vlan_valid(smi, vlan->vid))
+       if (!smi->ops->is_vlan_valid(smi, vlan->vid)) {
+               NL_SET_ERR_MSG_MOD(extack, "VLAN ID not valid");
                return -EINVAL;
+       }
 
        /* Enable VLAN in the hardware
         * FIXME: what's with this 4k business?
         * Just rtl8366_enable_vlan() seems inconclusive.
         */
        ret = rtl8366_enable_vlan4k(smi, true);
-       if (ret)
+       if (ret) {
+               NL_SET_ERR_MSG_MOD(extack, "Failed to enable VLAN 4K");
                return ret;
+       }
 
        dev_info(smi->dev, "add VLAN %d on port %d, %s, %s\n",
                 vlan->vid, port, untagged ? "untagged" : "tagged",
 
 }
 
 static int sja1105_vlan_add(struct dsa_switch *ds, int port,
-                           const struct switchdev_obj_port_vlan *vlan)
+                           const struct switchdev_obj_port_vlan *vlan,
+                           struct netlink_ext_ack *extack)
 {
        struct sja1105_private *priv = ds->priv;
        bool vlan_table_changed = false;
         */
        if (priv->vlan_state != SJA1105_VLAN_FILTERING_FULL &&
            vid_is_dsa_8021q(vlan->vid)) {
-               dev_err(ds->dev, "Range 1024-3071 reserved for dsa_8021q operation\n");
+               NL_SET_ERR_MSG_MOD(extack,
+                                  "Range 1024-3071 reserved for dsa_8021q operation");
                return -EBUSY;
        }
 
 
        int     (*port_vlan_filtering)(struct dsa_switch *ds, int port,
                                       bool vlan_filtering);
        int     (*port_vlan_add)(struct dsa_switch *ds, int port,
-                                const struct switchdev_obj_port_vlan *vlan);
+                                const struct switchdev_obj_port_vlan *vlan,
+                                struct netlink_ext_ack *extack);
        int     (*port_vlan_del)(struct dsa_switch *ds, int port,
                                 const struct switchdev_obj_port_vlan *vlan);
        /*
 
        const struct switchdev_obj_port_vlan *vlan;
        int sw_index;
        int port;
+       struct netlink_ext_ack *extack;
 };
 
 /* DSA_NOTIFIER_MTU */
 int dsa_port_mrouter(struct dsa_port *dp, bool mrouter,
                     struct netlink_ext_ack *extack);
 int dsa_port_vlan_add(struct dsa_port *dp,
-                     const struct switchdev_obj_port_vlan *vlan);
+                     const struct switchdev_obj_port_vlan *vlan,
+                     struct netlink_ext_ack *extack);
 int dsa_port_vlan_del(struct dsa_port *dp,
                      const struct switchdev_obj_port_vlan *vlan);
 int dsa_port_link_register_of(struct dsa_port *dp);
 
 }
 
 int dsa_port_vlan_add(struct dsa_port *dp,
-                     const struct switchdev_obj_port_vlan *vlan)
+                     const struct switchdev_obj_port_vlan *vlan,
+                     struct netlink_ext_ack *extack)
 {
        struct dsa_notifier_vlan_info info = {
                .sw_index = dp->ds->index,
                .port = dp->index,
                .vlan = vlan,
+               .extack = extack,
        };
 
        return dsa_port_notify(dp, DSA_NOTIFIER_VLAN_ADD, &info);
 
                rcu_read_lock();
                err = dsa_slave_vlan_check_for_8021q_uppers(dev, &vlan);
                rcu_read_unlock();
-               if (err)
+               if (err) {
+                       NL_SET_ERR_MSG_MOD(extack,
+                                          "Port already has a VLAN upper with this VID");
                        return err;
+               }
        }
 
-       err = dsa_port_vlan_add(dp, &vlan);
+       err = dsa_port_vlan_add(dp, &vlan, extack);
        if (err)
                return err;
 
         */
        vlan.flags &= ~BRIDGE_VLAN_INFO_PVID;
 
-       err = dsa_port_vlan_add(dp->cpu_dp, &vlan);
+       err = dsa_port_vlan_add(dp->cpu_dp, &vlan, extack);
        if (err)
                return err;
 
                /* This API only allows programming tagged, non-PVID VIDs */
                .flags = 0,
        };
+       struct netlink_ext_ack extack = {0};
        int ret;
 
        /* User port... */
-       ret = dsa_port_vlan_add(dp, &vlan);
-       if (ret)
+       ret = dsa_port_vlan_add(dp, &vlan, &extack);
+       if (ret) {
+               if (extack._msg)
+                       netdev_err(dev, "%s\n", extack._msg);
                return ret;
+       }
 
        /* And CPU port... */
-       ret = dsa_port_vlan_add(dp->cpu_dp, &vlan);
-       if (ret)
+       ret = dsa_port_vlan_add(dp->cpu_dp, &vlan, &extack);
+       if (ret) {
+               if (extack._msg)
+                       netdev_err(dev, "CPU port %d: %s\n", dp->cpu_dp->index,
+                                  extack._msg);
                return ret;
+       }
 
        return vlan_vid_add(master, proto, vid);
 }
 
 
        for (port = 0; port < ds->num_ports; port++) {
                if (dsa_switch_vlan_match(ds, port, info)) {
-                       err = ds->ops->port_vlan_add(ds, port, info->vlan);
+                       err = ds->ops->port_vlan_add(ds, port, info->vlan,
+                                                    info->extack);
                        if (err)
                                return err;
                }