if (!port)
                ret = 0;
        else {
+               const struct net_bridge_port *dst = NULL;
+
                fdb = br_fdb_find_rcu(port->br, addr, 0);
-               ret = fdb && fdb->dst && fdb->dst->dev != dev &&
-                       fdb->dst->state == BR_STATE_FORWARDING;
+               if (fdb)
+                       dst = READ_ONCE(fdb->dst);
+
+               ret = dst && dst->dev != dev &&
+                     dst->state == BR_STATE_FORWARDING;
        }
        rcu_read_unlock();
 
        fdb = kmem_cache_alloc(br_fdb_cache, GFP_ATOMIC);
        if (fdb) {
                memcpy(fdb->key.addr.addr, addr, ETH_ALEN);
-               fdb->dst = source;
+               WRITE_ONCE(fdb->dst, source);
                fdb->key.vlan_id = vid;
                fdb->flags = flags;
                fdb->updated = fdb->used = jiffies;
                        }
 
                        /* fastpath: update of existing entry */
-                       if (unlikely(source != fdb->dst &&
+                       if (unlikely(source != READ_ONCE(fdb->dst) &&
                                     !test_bit(BR_FDB_STICKY, &fdb->flags))) {
                                br_switchdev_fdb_notify(fdb, RTM_DELNEIGH);
-                               fdb->dst = source;
+                               WRITE_ONCE(fdb->dst, source);
                                fdb_modified = true;
                                /* Take over HW learned entry */
                                if (unlikely(test_bit(BR_FDB_ADDED_BY_EXT_LEARN,
                         const struct net_bridge_fdb_entry *fdb,
                         u32 portid, u32 seq, int type, unsigned int flags)
 {
+       const struct net_bridge_port *dst = READ_ONCE(fdb->dst);
        unsigned long now = jiffies;
        struct nda_cacheinfo ci;
        struct nlmsghdr *nlh;
        ndm->ndm_pad2    = 0;
        ndm->ndm_flags   = 0;
        ndm->ndm_type    = 0;
-       ndm->ndm_ifindex = fdb->dst ? fdb->dst->dev->ifindex : br->dev->ifindex;
+       ndm->ndm_ifindex = dst ? dst->dev->ifindex : br->dev->ifindex;
        ndm->ndm_state   = fdb_to_nud(br, fdb);
 
        if (test_bit(BR_FDB_OFFLOADED, &fdb->flags))
                if (flags & NLM_F_EXCL)
                        return -EEXIST;
 
-               if (fdb->dst != source) {
-                       fdb->dst = source;
+               if (READ_ONCE(fdb->dst) != source) {
+                       WRITE_ONCE(fdb->dst, source);
                        modified = true;
                }
        }
        struct net_bridge_fdb_entry *fdb;
 
        fdb = br_fdb_find(br, addr, vlan);
-       if (!fdb || fdb->dst != p)
+       if (!fdb || READ_ONCE(fdb->dst) != p)
                return -ENOENT;
 
        fdb_delete(br, fdb, true);
        } else {
                fdb->updated = jiffies;
 
-               if (fdb->dst != p) {
-                       fdb->dst = p;
+               if (READ_ONCE(fdb->dst) != p) {
+                       WRITE_ONCE(fdb->dst, p);
                        modified = true;
                }
 
 
 void
 br_switchdev_fdb_notify(const struct net_bridge_fdb_entry *fdb, int type)
 {
+       const struct net_bridge_port *dst = READ_ONCE(fdb->dst);
        struct switchdev_notifier_fdb_info info = {
                .addr = fdb->key.addr.addr,
                .vid = fdb->key.vlan_id,
                .offloaded = test_bit(BR_FDB_OFFLOADED, &fdb->flags),
        };
 
-       if (!fdb->dst)
+       if (!dst)
                return;
 
        switch (type) {
        case RTM_DELNEIGH:
                call_switchdev_notifiers(SWITCHDEV_FDB_DEL_TO_DEVICE,
-                                        fdb->dst->dev, &info.info, NULL);
+                                        dst->dev, &info.info, NULL);
                break;
        case RTM_NEWNEIGH:
                call_switchdev_notifiers(SWITCHDEV_FDB_ADD_TO_DEVICE,
-                                        fdb->dst->dev, &info.info, NULL);
+                                        dst->dev, &info.info, NULL);
                break;
        }
 }