Support for IEEE 802.15.4 management information base,
routine like setting of HW address, PAN id, channel etc...

Signed-off-by: Alexander Smirnov <alex.bluesman.smir...@gmail.com>
---
 include/linux/if.h        |    3 +
 net/mac802154/Makefile    |    2 +-
 net/mac802154/mac802154.h |   22 ++++
 net/mac802154/mib.c       |  247 +++++++++++++++++++++++++++++++++++++++++++++
 net/mac802154/mib.h       |   35 +++++++
 5 files changed, 308 insertions(+), 1 deletions(-)
 create mode 100644 net/mac802154/mib.c
 create mode 100644 net/mac802154/mib.h

diff --git a/include/linux/if.h b/include/linux/if.h
index 06b6ef6..deaf14c 100644
--- a/include/linux/if.h
+++ b/include/linux/if.h
@@ -76,6 +76,9 @@
 #define IFF_BRIDGE_PORT        0x4000          /* device used as bridge port */
 #define IFF_OVS_DATAPATH       0x8000  /* device used as Open vSwitch
                                         * datapath port */
+
+#define IFF_IEEE802154_COORD   0x400   /* IEEE802.15.4 PAN coordinator */
+
 #define IFF_TX_SKB_SHARING     0x10000 /* The interface supports sharing
                                         * skbs on transmit */
 #define IFF_UNICAST_FLT        0x20000         /* Supports unicast filtering   
*/
diff --git a/net/mac802154/Makefile b/net/mac802154/Makefile
index 490beef..875feb2 100644
--- a/net/mac802154/Makefile
+++ b/net/mac802154/Makefile
@@ -1,2 +1,2 @@
 obj-$(CONFIG_MAC802154)        += mac802154.o
-mac802154-objs         := ieee802154_dev.o rx.o tx.o
+mac802154-objs         := ieee802154_dev.o rx.o tx.o mib.o
diff --git a/net/mac802154/mac802154.h b/net/mac802154/mac802154.h
index 99c16ac..752ae6a 100644
--- a/net/mac802154/mac802154.h
+++ b/net/mac802154/mac802154.h
@@ -53,4 +53,26 @@ struct mac802154_priv {
 
 #define mac802154_to_priv(_hw) container_of(_hw, struct mac802154_priv, hw)
 
+struct mac802154_sub_if_data {
+       struct list_head list; /* the ieee802154_priv->slaves list */
+
+       struct mac802154_priv *hw;
+       struct net_device *dev;
+
+       int type;
+
+       spinlock_t mib_lock;
+
+       u16 pan_id;
+       u16 short_addr;
+
+       u8 chan;
+       u8 page;
+
+       /* MAC BSN field */
+       u8 bsn;
+       /* MAC DSN field */
+       u8 dsn;
+};
+
 #endif /* MAC802154_H */
diff --git a/net/mac802154/mib.c b/net/mac802154/mib.c
new file mode 100644
index 0000000..8574bf6
--- /dev/null
+++ b/net/mac802154/mib.c
@@ -0,0 +1,247 @@
+/*
+ * Copyright 2007, 2008, 2009 Siemens AG
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2
+ * as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Written by:
+ * Dmitry Eremin-Solenikov <dbarysh...@gmail.com>
+ * Sergey Lapin <sla...@ossfans.org>
+ * Maxim Gorbachyov <maxim.gorbac...@siemens.com>
+ */
+
+#include <linux/if_arp.h>
+
+#include <net/mac802154.h>
+#include <net/wpan-phy.h>
+
+#include "mac802154.h"
+#include "mib.h"
+
+struct phy_chan_notify_work {
+       struct work_struct work;
+       struct net_device *dev;
+};
+
+struct hw_addr_filt_notify_work {
+       struct work_struct work;
+       struct net_device *dev;
+       unsigned long changed;
+};
+
+static void hw_addr_notify(struct work_struct *work)
+{
+       struct hw_addr_filt_notify_work *nw =
+               container_of(work, struct hw_addr_filt_notify_work, work);
+       struct mac802154_priv *hw = mac802154_slave_get_priv(nw->dev);
+       int res;
+
+       res = hw->ops->set_hw_addr_filt(&hw->hw, &hw->hw.hw_filt, nw->changed);
+       if (res)
+               pr_debug("(%s): failed changed mask %lx\n",
+                       __func__, nw->changed);
+
+       kfree(nw);
+
+       return;
+}
+
+static void set_hw_addr_filt(struct net_device *dev, unsigned long changed)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       struct hw_addr_filt_notify_work *work;
+
+       work = kzalloc(sizeof(*work), GFP_ATOMIC);
+       if (!work)
+               return;
+
+       INIT_WORK(&work->work, hw_addr_notify);
+       work->dev = dev;
+       work->changed = changed;
+       queue_work(priv->hw->dev_workqueue, &work->work);
+
+       return;
+}
+
+static void phy_chan_notify(struct work_struct *work)
+{
+       struct phy_chan_notify_work *nw =
+               container_of(work, struct phy_chan_notify_work, work);
+       struct mac802154_priv *hw = mac802154_slave_get_priv(nw->dev);
+       struct mac802154_sub_if_data *priv = netdev_priv(nw->dev);
+       int res;
+
+       res = hw->ops->set_channel(&hw->hw, priv->page, priv->chan);
+       if (res)
+               pr_debug("(%s): set_channel failed\n", __func__);
+
+       kfree(nw);
+
+       return;
+}
+
+u16 mac802154_dev_get_pan_id(const struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       u16 ret;
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       ret = priv->pan_id;
+       spin_unlock_bh(&priv->mib_lock);
+
+       return ret;
+}
+
+u16 mac802154_dev_get_short_addr(const struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       u16 ret;
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       ret = priv->short_addr;
+       spin_unlock_bh(&priv->mib_lock);
+
+       return ret;
+}
+
+void mac802154_dev_set_pan_id(struct net_device *dev, u16 val)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       priv->pan_id = val;
+       spin_unlock_bh(&priv->mib_lock);
+
+       if (priv->hw->ops->set_hw_addr_filt &&
+               (priv->hw->hw.hw_filt.pan_id != priv->pan_id)) {
+               priv->hw->hw.hw_filt.pan_id = priv->pan_id;
+               set_hw_addr_filt(dev, IEEE802515_PANID_CHANGED);
+       }
+}
+
+void mac802154_dev_set_pan_coord(struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       int pan_coord = !!(dev->priv_flags & IFF_IEEE802154_COORD);
+
+       if (priv->hw->ops->set_hw_addr_filt &&
+               (priv->hw->hw.hw_filt.pan_coord != pan_coord)) {
+               priv->hw->hw.hw_filt.pan_coord = pan_coord;
+               set_hw_addr_filt(dev, IEEE802515_PANC_CHANGED);
+       }
+}
+
+void mac802154_dev_set_short_addr(struct net_device *dev, u16 val)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       priv->short_addr = val;
+       spin_unlock_bh(&priv->mib_lock);
+
+       if (priv->hw->ops->set_hw_addr_filt &&
+               (priv->hw->hw.hw_filt.short_addr != priv->short_addr)) {
+               priv->hw->hw.hw_filt.short_addr = priv->short_addr;
+               set_hw_addr_filt(dev, IEEE802515_SADDR_CHANGED);
+       }
+}
+
+void mac802154_dev_set_ieee_addr(struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+
+       if (priv->hw->ops->set_hw_addr_filt &&
+               memcmp(priv->hw->hw.hw_filt.ieee_addr,
+                       dev->dev_addr, IEEE802154_ALEN)) {
+               memcpy(priv->hw->hw.hw_filt.ieee_addr,
+                       dev->dev_addr, IEEE802154_ALEN);
+               set_hw_addr_filt(dev, IEEE802515_IEEEADDR_CHANGED);
+       }
+}
+
+void mac802154_dev_set_page_channel(struct net_device *dev, u8 page, u8 chan)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       struct phy_chan_notify_work *work;
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       priv->page = page;
+       priv->chan = chan;
+       spin_unlock_bh(&priv->mib_lock);
+
+       if (priv->hw->phy->current_channel != priv->chan ||
+           priv->hw->phy->current_page != priv->page) {
+               work = kzalloc(sizeof(*work), GFP_ATOMIC);
+               if (!work)
+                       return;
+
+               INIT_WORK(&work->work, phy_chan_notify);
+               work->dev = dev;
+               queue_work(priv->hw->dev_workqueue, &work->work);
+       }
+}
+
+u8 mac802154_dev_get_dsn(const struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       u16 ret;
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       ret = priv->dsn++;
+       spin_unlock_bh(&priv->mib_lock);
+
+       return ret;
+}
+
+u8 mac802154_dev_get_bsn(const struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       u16 ret;
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       spin_lock_bh(&priv->mib_lock);
+       ret = priv->bsn++;
+       spin_unlock_bh(&priv->mib_lock);
+
+       return ret;
+}
+
+struct mac802154_priv *mac802154_slave_get_priv(struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       return priv->hw;
+}
+
+struct wpan_phy *mac802154_get_phy(const struct net_device *dev)
+{
+       struct mac802154_sub_if_data *priv = netdev_priv(dev);
+
+       BUG_ON(dev->type != ARPHRD_IEEE802154);
+
+       return to_phy(get_device(&priv->hw->phy->dev));
+}
diff --git a/net/mac802154/mib.h b/net/mac802154/mib.h
new file mode 100644
index 0000000..a243339
--- /dev/null
+++ b/net/mac802154/mib.h
@@ -0,0 +1,35 @@
+/*
+ * Copyright 2008 Siemens AG
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2
+ * as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ */
+
+#ifndef MIB802154_H
+#define MIB802154_H
+
+/* MIB API */
+u8 mac802154_dev_get_dsn(const struct net_device *dev);
+u8 mac802154_dev_get_bsn(const struct net_device *dev);
+u16 mac802154_dev_get_pan_id(const struct net_device *dev);
+u16 mac802154_dev_get_short_addr(const struct net_device *dev);
+void mac802154_dev_set_pan_id(struct net_device *dev, u16 val);
+void mac802154_dev_set_pan_coord(struct net_device *dev);
+void mac802154_dev_set_short_addr(struct net_device *dev, u16 val);
+void mac802154_dev_set_ieee_addr(struct net_device *dev);
+void mac802154_dev_set_page_channel(struct net_device *dev, u8 page, u8 chan);
+struct wpan_phy *mac802154_get_phy(const struct net_device *dev);
+struct mac802154_priv *mac802154_slave_get_priv(struct net_device *dev);
+
+#endif
-- 
1.7.2.3


------------------------------------------------------------------------------
All the data continuously generated in your IT infrastructure 
contains a definitive record of customers, application performance, 
security threats, fraudulent activity, and more. Splunk takes this 
data and makes sense of it. IT sense. And common sense.
http://p.sf.net/sfu/splunk-novd2d
_______________________________________________
Linux-zigbee-devel mailing list
Linux-zigbee-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/linux-zigbee-devel

Reply via email to