2012-05-15 20:50:20 +00:00
/*
* Copyright ( C ) 2007 - 2012 Siemens AG
*
* Written by :
* Alexander Smirnov < alex . bluesman . smirnov @ gmail . com >
*
* 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 .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/netdevice.h>
2012-05-15 20:50:28 +00:00
# include <net/netlink.h>
2014-11-17 08:20:49 +01:00
# include <net/nl802154.h>
2012-05-15 20:50:20 +00:00
# include <net/mac802154.h>
2014-03-14 21:23:59 +01:00
# include <net/ieee802154_netdev.h>
2012-05-15 20:50:20 +00:00
# include <net/route.h>
2014-10-25 09:41:02 +02:00
# include <net/cfg802154.h>
2012-05-15 20:50:20 +00:00
2014-10-25 09:41:00 +02:00
# include "ieee802154_i.h"
2014-11-02 04:18:36 +01:00
# include "cfg.h"
2012-05-15 20:50:20 +00:00
2014-10-27 17:13:30 +01:00
static void ieee802154_tasklet_handler ( unsigned long data )
{
struct ieee802154_local * local = ( struct ieee802154_local * ) data ;
struct sk_buff * skb ;
while ( ( skb = skb_dequeue ( & local - > skb_queue ) ) ) {
switch ( skb - > pkt_type ) {
case IEEE802154_RX_MSG :
/* Clear skb->pkt_type in order to not confuse kernel
* netstack .
*/
skb - > pkt_type = 0 ;
ieee802154_rx ( & local - > hw , skb ) ;
break ;
default :
WARN ( 1 , " mac802154: Packet is of unknown type %d \n " ,
skb - > pkt_type ) ;
kfree_skb ( skb ) ;
break ;
}
}
}
2014-10-25 17:16:34 +02:00
struct ieee802154_hw *
2014-10-28 18:21:18 +01:00
ieee802154_alloc_hw ( size_t priv_data_len , const struct ieee802154_ops * ops )
2012-05-15 20:50:20 +00:00
{
struct wpan_phy * phy ;
2014-10-25 17:16:35 +02:00
struct ieee802154_local * local ;
2012-05-15 20:50:20 +00:00
size_t priv_size ;
2014-10-26 09:37:08 +01:00
if ( ! ops | | ! ( ops - > xmit_async | | ops - > xmit_sync ) | | ! ops - > ed | |
! ops - > start | | ! ops - > stop | | ! ops - > set_channel ) {
2013-10-30 15:28:07 +08:00
pr_err ( " undefined IEEE802.15.4 device operations \n " ) ;
2012-05-15 20:50:20 +00:00
return NULL ;
}
/* Ensure 32-byte alignment of our private data and hw private data.
2014-10-25 17:16:35 +02:00
* We use the wpan_phy priv data for both our ieee802154_local and for
2012-05-15 20:50:20 +00:00
* the driver ' s private data
*
* in memory it ' ll be like this :
*
2014-10-25 17:16:35 +02:00
* + - - - - - - - - - - - - - - - - - - - - - - - - - +
* | struct wpan_phy |
* + - - - - - - - - - - - - - - - - - - - - - - - - - +
* | struct ieee802154_local |
* + - - - - - - - - - - - - - - - - - - - - - - - - - +
* | driver ' s private data |
* + - - - - - - - - - - - - - - - - - - - - - - - - - +
2012-05-15 20:50:20 +00:00
*
* Due to ieee802154 layer isn ' t aware of driver and MAC structures ,
2014-10-25 05:25:08 +02:00
* so lets align them here .
2012-05-15 20:50:20 +00:00
*/
2014-10-25 17:16:35 +02:00
priv_size = ALIGN ( sizeof ( * local ) , NETDEV_ALIGN ) + priv_data_len ;
2012-05-15 20:50:20 +00:00
2014-11-09 08:36:47 +01:00
phy = wpan_phy_new ( & mac802154_config_ops , priv_size ) ;
2012-05-15 20:50:20 +00:00
if ( ! phy ) {
2013-10-30 15:28:07 +08:00
pr_err ( " failure to allocate master IEEE802.15.4 device \n " ) ;
2012-05-15 20:50:20 +00:00
return NULL ;
}
2014-11-12 03:36:51 +01:00
phy - > privid = mac802154_wpan_phy_privid ;
2014-10-25 17:16:35 +02:00
local = wpan_phy_priv ( phy ) ;
local - > phy = phy ;
local - > hw . phy = local - > phy ;
local - > hw . priv = ( char * ) local + ALIGN ( sizeof ( * local ) , NETDEV_ALIGN ) ;
local - > ops = ops ;
2012-05-15 20:50:20 +00:00
2014-10-25 17:16:38 +02:00
INIT_LIST_HEAD ( & local - > interfaces ) ;
mutex_init ( & local - > iflist_mtx ) ;
2012-05-15 20:50:20 +00:00
2014-10-27 17:13:30 +01:00
tasklet_init ( & local - > tasklet ,
ieee802154_tasklet_handler ,
( unsigned long ) local ) ;
skb_queue_head_init ( & local - > skb_queue ) ;
2014-10-25 17:16:35 +02:00
return & local - > hw ;
2012-05-15 20:50:20 +00:00
}
2014-10-25 17:16:34 +02:00
EXPORT_SYMBOL ( ieee802154_alloc_hw ) ;
2012-05-15 20:50:20 +00:00
2014-10-25 17:16:34 +02:00
void ieee802154_free_hw ( struct ieee802154_hw * hw )
2012-05-15 20:50:20 +00:00
{
2014-10-25 17:16:39 +02:00
struct ieee802154_local * local = hw_to_local ( hw ) ;
2012-05-15 20:50:20 +00:00
2014-10-25 17:16:38 +02:00
BUG_ON ( ! list_empty ( & local - > interfaces ) ) ;
2012-05-15 20:50:28 +00:00
2014-10-25 17:16:38 +02:00
mutex_destroy ( & local - > iflist_mtx ) ;
2012-12-14 01:03:03 +00:00
2014-10-25 17:16:35 +02:00
wpan_phy_free ( local - > phy ) ;
2012-05-15 20:50:20 +00:00
}
2014-10-25 17:16:34 +02:00
EXPORT_SYMBOL ( ieee802154_free_hw ) ;
2012-05-15 20:50:20 +00:00
2014-11-12 19:51:56 +01:00
static void ieee802154_setup_wpan_phy_pib ( struct wpan_phy * wpan_phy )
{
/* TODO warn on empty symbol_duration
* Should be done when all drivers sets this value .
*/
wpan_phy - > lifs_period = IEEE802154_LIFS_PERIOD *
wpan_phy - > symbol_duration ;
wpan_phy - > sifs_period = IEEE802154_SIFS_PERIOD *
wpan_phy - > symbol_duration ;
}
2014-10-25 17:16:34 +02:00
int ieee802154_register_hw ( struct ieee802154_hw * hw )
2012-05-15 20:50:20 +00:00
{
2014-10-25 17:16:39 +02:00
struct ieee802154_local * local = hw_to_local ( hw ) ;
2014-11-05 20:51:19 +01:00
struct net_device * dev ;
2014-07-03 00:20:43 +02:00
int rc = - ENOSYS ;
2014-10-25 17:16:41 +02:00
local - > workqueue =
2014-10-25 17:16:35 +02:00
create_singlethread_workqueue ( wpan_phy_name ( local - > phy ) ) ;
2014-10-25 17:16:41 +02:00
if ( ! local - > workqueue ) {
2014-07-03 00:20:43 +02:00
rc = - ENOMEM ;
2012-05-15 20:50:20 +00:00
goto out ;
2014-07-03 00:20:43 +02:00
}
2012-05-15 20:50:20 +00:00
2014-11-12 19:51:56 +01:00
hrtimer_init ( & local - > ifs_timer , CLOCK_MONOTONIC , HRTIMER_MODE_REL ) ;
local - > ifs_timer . function = ieee802154_xmit_ifs_timer ;
2014-10-25 17:16:35 +02:00
wpan_phy_set_dev ( local - > phy , local - > hw . parent ) ;
2012-05-15 20:50:20 +00:00
2014-11-12 19:51:56 +01:00
ieee802154_setup_wpan_phy_pib ( local - > phy ) ;
2014-10-25 17:16:35 +02:00
rc = wpan_phy_register ( local - > phy ) ;
2012-05-15 20:50:20 +00:00
if ( rc < 0 )
goto out_wq ;
2014-11-05 20:51:19 +01:00
rtnl_lock ( ) ;
2014-11-17 08:20:52 +01:00
dev = ieee802154_if_add ( local , " wpan%d " , NL802154_IFTYPE_NODE ,
cpu_to_le64 ( 0x0000000000000000ULL ) ) ;
2014-11-05 20:51:19 +01:00
if ( IS_ERR ( dev ) ) {
rtnl_unlock ( ) ;
rc = PTR_ERR ( dev ) ;
goto out_wq ;
}
rtnl_unlock ( ) ;
2012-05-15 20:50:20 +00:00
return 0 ;
out_wq :
2014-10-25 17:16:41 +02:00
destroy_workqueue ( local - > workqueue ) ;
2012-05-15 20:50:20 +00:00
out :
return rc ;
}
2014-10-25 17:16:34 +02:00
EXPORT_SYMBOL ( ieee802154_register_hw ) ;
2012-05-15 20:50:20 +00:00
2014-10-25 17:16:34 +02:00
void ieee802154_unregister_hw ( struct ieee802154_hw * hw )
2012-05-15 20:50:20 +00:00
{
2014-10-25 17:16:39 +02:00
struct ieee802154_local * local = hw_to_local ( hw ) ;
2012-05-15 20:50:20 +00:00
2014-10-27 17:13:30 +01:00
tasklet_kill ( & local - > tasklet ) ;
2014-10-25 17:16:41 +02:00
flush_workqueue ( local - > workqueue ) ;
destroy_workqueue ( local - > workqueue ) ;
2012-05-15 20:50:20 +00:00
rtnl_lock ( ) ;
2014-11-12 03:36:48 +01:00
ieee802154_remove_interfaces ( local ) ;
2012-05-15 20:50:28 +00:00
2012-05-15 20:50:20 +00:00
rtnl_unlock ( ) ;
2014-10-25 17:16:35 +02:00
wpan_phy_unregister ( local - > phy ) ;
2012-05-15 20:50:20 +00:00
}
2014-10-25 17:16:34 +02:00
EXPORT_SYMBOL ( ieee802154_unregister_hw ) ;
2012-05-15 20:50:20 +00:00
2014-11-12 03:36:53 +01:00
static int __init ieee802154_init ( void )
{
return ieee802154_iface_init ( ) ;
}
static void __exit ieee802154_exit ( void )
{
ieee802154_iface_exit ( ) ;
rcu_barrier ( ) ;
}
subsys_initcall ( ieee802154_init ) ;
module_exit ( ieee802154_exit ) ;
2014-11-12 03:36:52 +01:00
MODULE_DESCRIPTION ( " IEEE 802.15.4 subsystem " ) ;
2012-05-15 20:50:20 +00:00
MODULE_LICENSE ( " GPL v2 " ) ;