u8                      devlink_port_setup:1;
 
+       /* Master state bits, valid only on CPU ports */
+       u8                      master_admin_up:1;
+       u8                      master_oper_up:1;
+
        u8                      setup:1;
 
        struct device_node      *dn;
        return dp->type == DSA_PORT_TYPE_UNUSED;
 }
 
+static inline bool dsa_port_master_is_operational(struct dsa_port *dp)
+{
+       return dsa_port_is_cpu(dp) && dp->master_admin_up &&
+              dp->master_oper_up;
+}
+
 static inline bool dsa_is_unused_port(struct dsa_switch *ds, int p)
 {
        return dsa_to_port(ds, p)->type == DSA_PORT_TYPE_UNUSED;
        int     (*tag_8021q_vlan_add)(struct dsa_switch *ds, int port, u16 vid,
                                      u16 flags);
        int     (*tag_8021q_vlan_del)(struct dsa_switch *ds, int port, u16 vid);
+
+       /*
+        * DSA master tracking operations
+        */
+       void    (*master_state_change)(struct dsa_switch *ds,
+                                      const struct net_device *master,
+                                      bool operational);
 };
 
 #define DSA_DEVLINK_PARAM_DRIVER(_id, _name, _type, _cmodes)           \
 
        return err;
 }
 
+static void dsa_tree_master_state_change(struct dsa_switch_tree *dst,
+                                        struct net_device *master)
+{
+       struct dsa_notifier_master_state_info info;
+       struct dsa_port *cpu_dp = master->dsa_ptr;
+
+       info.master = master;
+       info.operational = dsa_port_master_is_operational(cpu_dp);
+
+       dsa_tree_notify(dst, DSA_NOTIFIER_MASTER_STATE_CHANGE, &info);
+}
+
+void dsa_tree_master_admin_state_change(struct dsa_switch_tree *dst,
+                                       struct net_device *master,
+                                       bool up)
+{
+       struct dsa_port *cpu_dp = master->dsa_ptr;
+       bool notify = false;
+
+       if ((dsa_port_master_is_operational(cpu_dp)) !=
+           (up && cpu_dp->master_oper_up))
+               notify = true;
+
+       cpu_dp->master_admin_up = up;
+
+       if (notify)
+               dsa_tree_master_state_change(dst, master);
+}
+
+void dsa_tree_master_oper_state_change(struct dsa_switch_tree *dst,
+                                      struct net_device *master,
+                                      bool up)
+{
+       struct dsa_port *cpu_dp = master->dsa_ptr;
+       bool notify = false;
+
+       if ((dsa_port_master_is_operational(cpu_dp)) !=
+           (cpu_dp->master_admin_up && up))
+               notify = true;
+
+       cpu_dp->master_oper_up = up;
+
+       if (notify)
+               dsa_tree_master_state_change(dst, master);
+}
+
 static struct dsa_port *dsa_port_touch(struct dsa_switch *ds, int index)
 {
        struct dsa_switch_tree *dst = ds->dst;
 
        DSA_NOTIFIER_TAG_PROTO_DISCONNECT,
        DSA_NOTIFIER_TAG_8021Q_VLAN_ADD,
        DSA_NOTIFIER_TAG_8021Q_VLAN_DEL,
+       DSA_NOTIFIER_MASTER_STATE_CHANGE,
 };
 
 /* DSA_NOTIFIER_AGEING_TIME */
        u16 vid;
 };
 
+/* DSA_NOTIFIER_MASTER_STATE_CHANGE */
+struct dsa_notifier_master_state_info {
+       const struct net_device *master;
+       bool operational;
+};
+
 struct dsa_switchdev_event_work {
        struct dsa_switch *ds;
        int port;
                              struct net_device *master,
                              const struct dsa_device_ops *tag_ops,
                              const struct dsa_device_ops *old_tag_ops);
+void dsa_tree_master_admin_state_change(struct dsa_switch_tree *dst,
+                                       struct net_device *master,
+                                       bool up);
+void dsa_tree_master_oper_state_change(struct dsa_switch_tree *dst,
+                                      struct net_device *master,
+                                      bool up);
 unsigned int dsa_bridge_num_get(const struct net_device *bridge_dev, int max);
 void dsa_bridge_num_put(const struct net_device *bridge_dev,
                        unsigned int bridge_num);
 
                err = dsa_port_lag_change(dp, info->lower_state_info);
                return notifier_from_errno(err);
        }
+       case NETDEV_CHANGE:
+       case NETDEV_UP: {
+               /* Track state of master port.
+                * DSA driver may require the master port (and indirectly
+                * the tagger) to be available for some special operation.
+                */
+               if (netdev_uses_dsa(dev)) {
+                       struct dsa_port *cpu_dp = dev->dsa_ptr;
+                       struct dsa_switch_tree *dst = cpu_dp->ds->dst;
+
+                       /* Track when the master port is UP */
+                       dsa_tree_master_oper_state_change(dst, dev,
+                                                         netif_oper_up(dev));
+
+                       /* Track when the master port is ready and can accept
+                        * packet.
+                        * NETDEV_UP event is not enough to flag a port as ready.
+                        * We also have to wait for linkwatch_do_dev to dev_activate
+                        * and emit a NETDEV_CHANGE event.
+                        * We check if a master port is ready by checking if the dev
+                        * have a qdisc assigned and is not noop.
+                        */
+                       dsa_tree_master_admin_state_change(dst, dev,
+                                                          !qdisc_tx_is_noop(dev));
+
+                       return NOTIFY_OK;
+               }
+
+               return NOTIFY_DONE;
+       }
        case NETDEV_GOING_DOWN: {
                struct dsa_port *dp, *cpu_dp;
                struct dsa_switch_tree *dst;
                cpu_dp = dev->dsa_ptr;
                dst = cpu_dp->ds->dst;
 
+               dsa_tree_master_admin_state_change(dst, dev, false);
+
                list_for_each_entry(dp, &dst->ports, list) {
                        if (!dsa_port_is_user(dp))
                                continue;
 
        return 0;
 }
 
+static int
+dsa_switch_master_state_change(struct dsa_switch *ds,
+                              struct dsa_notifier_master_state_info *info)
+{
+       if (!ds->ops->master_state_change)
+               return 0;
+
+       ds->ops->master_state_change(ds, info->master, info->operational);
+
+       return 0;
+}
+
 static int dsa_switch_event(struct notifier_block *nb,
                            unsigned long event, void *info)
 {
        case DSA_NOTIFIER_TAG_8021Q_VLAN_DEL:
                err = dsa_switch_tag_8021q_vlan_del(ds, info);
                break;
+       case DSA_NOTIFIER_MASTER_STATE_CHANGE:
+               err = dsa_switch_master_state_change(ds, info);
+               break;
        default:
                err = -EOPNOTSUPP;
                break;