};
 
 static inline u32 ocelot_mact_read_macaccess(struct ocelot *ocelot)
+       __must_hold(&ocelot->mact_lock)
 {
        return ocelot_read(ocelot, ANA_TABLES_MACACCESS);
 }
 
 static inline int ocelot_mact_wait_for_completion(struct ocelot *ocelot)
+       __must_hold(&ocelot->mact_lock)
 {
        u32 val;
 
 static void ocelot_mact_select(struct ocelot *ocelot,
                               const unsigned char mac[ETH_ALEN],
                               unsigned int vid)
+       __must_hold(&ocelot->mact_lock)
 {
        u32 macl = 0, mach = 0;
 
                ANA_TABLES_MACACCESS_ENTRYTYPE(type) |
                ANA_TABLES_MACACCESS_MAC_TABLE_CMD(MACACCESS_CMD_LEARN);
        unsigned int mc_ports;
+       int err;
 
        /* Set MAC_CPU_COPY if the CPU port is used by a multicast entry */
        if (type == ENTRYTYPE_MACv4)
        if (mc_ports & BIT(ocelot->num_phys_ports))
                cmd |= ANA_TABLES_MACACCESS_MAC_CPU_COPY;
 
+       mutex_lock(&ocelot->mact_lock);
+
        ocelot_mact_select(ocelot, mac, vid);
 
        /* Issue a write command */
        ocelot_write(ocelot, cmd, ANA_TABLES_MACACCESS);
 
-       return ocelot_mact_wait_for_completion(ocelot);
+       err = ocelot_mact_wait_for_completion(ocelot);
+
+       mutex_unlock(&ocelot->mact_lock);
+
+       return err;
 }
 EXPORT_SYMBOL(ocelot_mact_learn);
 
 int ocelot_mact_forget(struct ocelot *ocelot,
                       const unsigned char mac[ETH_ALEN], unsigned int vid)
 {
+       int err;
+
+       mutex_lock(&ocelot->mact_lock);
+
        ocelot_mact_select(ocelot, mac, vid);
 
        /* Issue a forget command */
                     ANA_TABLES_MACACCESS_MAC_TABLE_CMD(MACACCESS_CMD_FORGET),
                     ANA_TABLES_MACACCESS);
 
-       return ocelot_mact_wait_for_completion(ocelot);
+       err = ocelot_mact_wait_for_completion(ocelot);
+
+       mutex_unlock(&ocelot->mact_lock);
+
+       return err;
 }
 EXPORT_SYMBOL(ocelot_mact_forget);
 
                   | ANA_AGENCTRL_LEARN_IGNORE_VLAN,
                   ANA_AGENCTRL);
 
-       /* Clear the MAC table */
+       /* Clear the MAC table. We are not concurrent with anyone, so
+        * holding &ocelot->mact_lock is pointless.
+        */
        ocelot_write(ocelot, MACACCESS_CMD_INIT, ANA_TABLES_MACACCESS);
 }
 
 
 static int ocelot_mact_read(struct ocelot *ocelot, int port, int row, int col,
                            struct ocelot_mact_entry *entry)
+       __must_hold(&ocelot->mact_lock)
 {
        u32 val, dst, macl, mach;
        char mac[ETH_ALEN];
 int ocelot_fdb_dump(struct ocelot *ocelot, int port,
                    dsa_fdb_dump_cb_t *cb, void *data)
 {
+       int err = 0;
        int i, j;
 
+       /* We could take the lock just around ocelot_mact_read, but doing so
+        * thousands of times in a row seems rather pointless and inefficient.
+        */
+       mutex_lock(&ocelot->mact_lock);
+
        /* Loop through all the mac tables entries. */
        for (i = 0; i < ocelot->num_mact_rows; i++) {
                for (j = 0; j < 4; j++) {
                        struct ocelot_mact_entry entry;
                        bool is_static;
-                       int ret;
 
-                       ret = ocelot_mact_read(ocelot, port, i, j, &entry);
+                       err = ocelot_mact_read(ocelot, port, i, j, &entry);
                        /* If the entry is invalid (wrong port, invalid...),
                         * skip it.
                         */
-                       if (ret == -EINVAL)
+                       if (err == -EINVAL)
                                continue;
-                       else if (ret)
-                               return ret;
+                       else if (err)
+                               break;
 
                        is_static = (entry.type == ENTRYTYPE_LOCKED);
 
-                       ret = cb(entry.mac, entry.vid, is_static, data);
-                       if (ret)
-                               return ret;
+                       err = cb(entry.mac, entry.vid, is_static, data);
+                       if (err)
+                               break;
                }
        }
 
-       return 0;
+       mutex_unlock(&ocelot->mact_lock);
+
+       return err;
 }
 EXPORT_SYMBOL(ocelot_fdb_dump);
 
 
        mutex_init(&ocelot->stats_lock);
        mutex_init(&ocelot->ptp_lock);
+       mutex_init(&ocelot->mact_lock);
        spin_lock_init(&ocelot->ptp_clock_lock);
        spin_lock_init(&ocelot->ts_id_lock);
        snprintf(queue_name, sizeof(queue_name), "%s-stats",