#include "br_private.h"
 
 static struct kmem_cache *br_fdb_cache __read_mostly;
-static struct net_bridge_fdb_entry *fdb_find(struct hlist_head *head,
-                                            const unsigned char *addr,
-                                            __u16 vid);
 static int fdb_insert(struct net_bridge *br, struct net_bridge_port *source,
                      const unsigned char *addr, u16 vid);
 static void fdb_notify(struct net_bridge *br,
        kmem_cache_free(br_fdb_cache, ent);
 }
 
+static struct net_bridge_fdb_entry *fdb_find_rcu(struct hlist_head *head,
+                                                const unsigned char *addr,
+                                                __u16 vid)
+{
+       struct net_bridge_fdb_entry *f;
+
+       hlist_for_each_entry_rcu(f, head, hlist)
+               if (ether_addr_equal(f->addr.addr, addr) && f->vlan_id == vid)
+                       break;
+
+       return f;
+}
+
+/* requires bridge hash_lock */
+static struct net_bridge_fdb_entry *br_fdb_find(struct net_bridge *br,
+                                               const unsigned char *addr,
+                                               __u16 vid)
+{
+       struct hlist_head *head = &br->hash[br_mac_hash(addr, vid)];
+       struct net_bridge_fdb_entry *fdb;
+
+       rcu_read_lock();
+       fdb = fdb_find_rcu(head, addr, vid);
+       rcu_read_unlock();
+
+       return fdb;
+}
+
+struct net_bridge_fdb_entry *br_fdb_find_rcu(struct net_bridge *br,
+                                            const unsigned char *addr,
+                                            __u16 vid)
+{
+       struct hlist_head *head = &br->hash[br_mac_hash(addr, vid)];
+
+       return fdb_find_rcu(head, addr, vid);
+}
+
 /* When a static FDB entry is added, the mac address from the entry is
  * added to the bridge private HW address list and all required ports
  * are then updated with the new information.
                              const struct net_bridge_port *p,
                              const unsigned char *addr, u16 vid)
 {
-       struct hlist_head *head = &br->hash[br_mac_hash(addr, vid)];
        struct net_bridge_fdb_entry *f;
 
        spin_lock_bh(&br->hash_lock);
-       f = fdb_find(head, addr, vid);
+       f = br_fdb_find(br, addr, vid);
        if (f && f->is_local && !f->added_by_user && f->dst == p)
                fdb_delete_local(br, p, f);
        spin_unlock_bh(&br->hash_lock);
        spin_lock_bh(&br->hash_lock);
 
        /* If old entry was unassociated with any port, then delete it. */
-       f = __br_fdb_get(br, br->dev->dev_addr, 0);
+       f = br_fdb_find(br, br->dev->dev_addr, 0);
        if (f && f->is_local && !f->dst && !f->added_by_user)
                fdb_delete_local(br, NULL, f);
 
        list_for_each_entry(v, &vg->vlan_list, vlist) {
                if (!br_vlan_should_use(v))
                        continue;
-               f = __br_fdb_get(br, br->dev->dev_addr, v->vid);
+               f = br_fdb_find(br, br->dev->dev_addr, v->vid);
                if (f && f->is_local && !f->dst && !f->added_by_user)
                        fdb_delete_local(br, NULL, f);
                fdb_insert(br, NULL, newaddr, v->vid);
        spin_unlock_bh(&br->hash_lock);
 }
 
-/* No locking or refcounting, assumes caller has rcu_read_lock */
-struct net_bridge_fdb_entry *__br_fdb_get(struct net_bridge *br,
-                                         const unsigned char *addr,
-                                         __u16 vid)
-{
-       struct net_bridge_fdb_entry *fdb;
-
-       hlist_for_each_entry_rcu(fdb,
-                               &br->hash[br_mac_hash(addr, vid)], hlist) {
-               if (ether_addr_equal(fdb->addr.addr, addr) &&
-                   fdb->vlan_id == vid) {
-                       return fdb;
-               }
-       }
-
-       return NULL;
-}
-
 #if IS_ENABLED(CONFIG_ATM_LANE)
 /* Interface used by ATM LANE hook to test
  * if an addr is on some other bridge port */
        if (!port)
                ret = 0;
        else {
-               fdb = __br_fdb_get(port->br, addr, 0);
+               fdb = br_fdb_find_rcu(port->br, addr, 0);
                ret = fdb && fdb->dst && fdb->dst->dev != dev &&
                        fdb->dst->state == BR_STATE_FORWARDING;
        }
        return num;
 }
 
-static struct net_bridge_fdb_entry *fdb_find(struct hlist_head *head,
-                                            const unsigned char *addr,
-                                            __u16 vid)
-{
-       struct net_bridge_fdb_entry *fdb;
-
-       hlist_for_each_entry(fdb, head, hlist) {
-               if (ether_addr_equal(fdb->addr.addr, addr) &&
-                   fdb->vlan_id == vid)
-                       return fdb;
-       }
-       return NULL;
-}
-
-static struct net_bridge_fdb_entry *fdb_find_rcu(struct hlist_head *head,
-                                                const unsigned char *addr,
-                                                __u16 vid)
-{
-       struct net_bridge_fdb_entry *fdb;
-
-       hlist_for_each_entry_rcu(fdb, head, hlist) {
-               if (ether_addr_equal(fdb->addr.addr, addr) &&
-                   fdb->vlan_id == vid)
-                       return fdb;
-       }
-       return NULL;
-}
-
 static struct net_bridge_fdb_entry *fdb_create(struct hlist_head *head,
                                               struct net_bridge_port *source,
                                               const unsigned char *addr,
        if (!is_valid_ether_addr(addr))
                return -EINVAL;
 
-       fdb = fdb_find(head, addr, vid);
+       fdb = br_fdb_find(br, addr, vid);
        if (fdb) {
                /* it is okay to have multiple ports with same
                 * address, just use the first one.
                }
        } else {
                spin_lock(&br->hash_lock);
-               if (likely(!fdb_find(head, addr, vid))) {
+               if (likely(!fdb_find_rcu(head, addr, vid))) {
                        fdb = fdb_create(head, source, addr, vid, 0, 0);
                        if (fdb) {
                                if (unlikely(added_by_user))
                return -EINVAL;
        }
 
-       fdb = fdb_find(head, addr, vid);
+       fdb = br_fdb_find(br, addr, vid);
        if (fdb == NULL) {
                if (!(flags & NLM_F_CREATE))
                        return -ENOENT;
 static int fdb_delete_by_addr(struct net_bridge *br, const u8 *addr,
                              u16 vid)
 {
-       struct hlist_head *head = &br->hash[br_mac_hash(addr, vid)];
        struct net_bridge_fdb_entry *fdb;
 
-       fdb = fdb_find(head, addr, vid);
+       fdb = br_fdb_find(br, addr, vid);
        if (!fdb)
                return -ENOENT;
 
                                       const u8 *addr, u16 vlan)
 {
        struct net_bridge *br = p->br;
-       struct hlist_head *head = &br->hash[br_mac_hash(addr, vlan)];
        struct net_bridge_fdb_entry *fdb;
 
-       fdb = fdb_find(head, addr, vlan);
+       fdb = br_fdb_find(br, addr, vlan);
        if (!fdb || fdb->dst != p)
                return -ENOENT;
 
        spin_lock_bh(&br->hash_lock);
 
        head = &br->hash[br_mac_hash(addr, vid)];
-       fdb = fdb_find(head, addr, vid);
+       fdb = br_fdb_find(br, addr, vid);
        if (!fdb) {
                fdb = fdb_create(head, p, addr, vid, 0, 0);
                if (!fdb) {
 int br_fdb_external_learn_del(struct net_bridge *br, struct net_bridge_port *p,
                              const unsigned char *addr, u16 vid)
 {
-       struct hlist_head *head;
        struct net_bridge_fdb_entry *fdb;
        int err = 0;
 
        ASSERT_RTNL();
        spin_lock_bh(&br->hash_lock);
 
-       head = &br->hash[br_mac_hash(addr, vid)];
-       fdb = fdb_find(head, addr, vid);
+       fdb = br_fdb_find(br, addr, vid);
        if (fdb && fdb->added_by_external_learn)
                fdb_delete(br, fdb);
        else