rcu_read_unlock();
 }
 
+static void hsr_del_ports(struct hsr_priv *hsr)
+{
+       struct hsr_port *port;
+
+       port = hsr_port_get_hsr(hsr, HSR_PT_SLAVE_A);
+       if (port)
+               hsr_del_port(port);
+
+       port = hsr_port_get_hsr(hsr, HSR_PT_SLAVE_B);
+       if (port)
+               hsr_del_port(port);
+
+       port = hsr_port_get_hsr(hsr, HSR_PT_MASTER);
+       if (port)
+               hsr_del_port(port);
+}
+
 /* This has to be called after all the readers are gone.
  * Otherwise we would have to check the return value of
  * hsr_port_get_hsr().
  */
 static void hsr_dev_destroy(struct net_device *hsr_dev)
 {
-       struct hsr_priv *hsr;
-       struct hsr_port *port;
-       struct hsr_port *tmp;
-
-       hsr = netdev_priv(hsr_dev);
+       struct hsr_priv *hsr = netdev_priv(hsr_dev);
 
        hsr_debugfs_term(hsr);
-
-       list_for_each_entry_safe(port, tmp, &hsr->ports, port_list)
-               hsr_del_port(port);
+       hsr_del_ports(hsr);
 
        del_timer_sync(&hsr->prune_timer);
        del_timer_sync(&hsr->announce_timer);
                     struct netlink_ext_ack *extack)
 {
        struct hsr_priv *hsr;
-       struct hsr_port *port;
-       struct hsr_port *tmp;
        int res;
 
        hsr = netdev_priv(hsr_dev);
 err_add_slaves:
        unregister_netdevice(hsr_dev);
 err_unregister:
-       list_for_each_entry_safe(port, tmp, &hsr->ports, port_list)
-               hsr_del_port(port);
+       hsr_del_ports(hsr);
 err_add_master:
        hsr_del_self_node(hsr);
 
 
 }
 
 /* Setup device to be added to the HSR bridge. */
-static int hsr_portdev_setup(struct net_device *dev, struct hsr_port *port)
+static int hsr_portdev_setup(struct hsr_priv *hsr, struct net_device *dev,
+                            struct hsr_port *port,
+                            struct netlink_ext_ack *extack)
+
 {
+       struct net_device *hsr_dev;
+       struct hsr_port *master;
        int res;
 
-       dev_hold(dev);
        res = dev_set_promiscuity(dev, 1);
        if (res)
                goto fail_promiscuity;
 
-       /* FIXME:
-        * What does net device "adjacency" mean? Should we do
-        * res = netdev_master_upper_dev_link(port->dev, port->hsr->dev); ?
-        */
+       master = hsr_port_get_hsr(hsr, HSR_PT_MASTER);
+       hsr_dev = master->dev;
+
+       res = netdev_upper_dev_link(dev, hsr_dev, extack);
+       if (res)
+               goto fail_upper_dev_link;
 
        res = netdev_rx_handler_register(dev, hsr_handle_frame, port);
        if (res)
        return 0;
 
 fail_rx_handler:
+       netdev_upper_dev_unlink(dev, hsr_dev);
+fail_upper_dev_link:
        dev_set_promiscuity(dev, -1);
 fail_promiscuity:
        dev_put(dev);
                return -ENOMEM;
 
        if (type != HSR_PT_MASTER) {
-               res = hsr_portdev_setup(dev, port);
+               res = hsr_portdev_setup(hsr, dev, port, extack);
                if (res)
                        goto fail_dev_setup;
        }
        list_del_rcu(&port->port_list);
 
        if (port != master) {
-               if (master) {
-                       netdev_update_features(master->dev);
-                       dev_set_mtu(master->dev, hsr_get_max_mtu(hsr));
-               }
+               netdev_update_features(master->dev);
+               dev_set_mtu(master->dev, hsr_get_max_mtu(hsr));
                netdev_rx_handler_unregister(port->dev);
                dev_set_promiscuity(port->dev, -1);
+               netdev_upper_dev_unlink(port->dev, master->dev);
        }
 
-       /* FIXME?
-        * netdev_upper_dev_unlink(port->dev, port->hsr->dev);
-        */
-
        synchronize_rcu();
 
-       if (port != master)
-               dev_put(port->dev);
        kfree(port);
 }