Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs-2.6
[firefly-linux-kernel-4.4.55.git] / net / ieee802154 / nl-phy.c
index 02548b292b53bf416f4e0ced8daeeb9e842666e3..c64a38d57aa36c4b521fd06c7a9c8c46bd0f337b 100644 (file)
@@ -24,6 +24,7 @@
 
 #include <linux/kernel.h>
 #include <linux/slab.h>
+#include <linux/if_arp.h>
 #include <net/netlink.h>
 #include <net/genetlink.h>
 #include <net/wpan-phy.h>
@@ -213,12 +214,37 @@ static int ieee802154_add_iface(struct sk_buff *skb,
                goto nla_put_failure;
        }
 
+       if (info->attrs[IEEE802154_ATTR_HW_ADDR] &&
+           nla_len(info->attrs[IEEE802154_ATTR_HW_ADDR]) !=
+                       IEEE802154_ADDR_LEN) {
+               rc = -EINVAL;
+               goto nla_put_failure;
+       }
+
        dev = phy->add_iface(phy, devname);
        if (IS_ERR(dev)) {
                rc = PTR_ERR(dev);
                goto nla_put_failure;
        }
 
+       if (info->attrs[IEEE802154_ATTR_HW_ADDR]) {
+               struct sockaddr addr;
+
+               addr.sa_family = ARPHRD_IEEE802154;
+               nla_memcpy(&addr.sa_data, info->attrs[IEEE802154_ATTR_HW_ADDR],
+                               IEEE802154_ADDR_LEN);
+
+               /*
+                * strangely enough, some callbacks (inetdev_event) from
+                * dev_set_mac_address require RTNL_LOCK
+                */
+               rtnl_lock();
+               rc = dev_set_mac_address(dev, &addr);
+               rtnl_unlock();
+               if (rc)
+                       goto dev_unregister;
+       }
+
        NLA_PUT_STRING(msg, IEEE802154_ATTR_PHY_NAME, wpan_phy_name(phy));
        NLA_PUT_STRING(msg, IEEE802154_ATTR_DEV_NAME, dev->name);
 
@@ -228,6 +254,11 @@ static int ieee802154_add_iface(struct sk_buff *skb,
 
        return ieee802154_nl_reply(msg, info);
 
+dev_unregister:
+       rtnl_lock(); /* del_iface must be called with RTNL lock */
+       phy->del_iface(phy, dev);
+       dev_put(dev);
+       rtnl_unlock();
 nla_put_failure:
        nlmsg_free(msg);
 out_dev: