2012-05-15 20:50:30 +00:00
/*
* Loopback IEEE 802.15 .4 interface
*
* Copyright 2007 - 2012 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 .
*
* Written by :
* Sergey Lapin < slapin @ ossfans . org >
* Dmitry Eremin - Solenikov < dbaryshkov @ gmail . com >
* Alexander Smirnov < alex . bluesman . smirnov @ gmail . com >
*/
# include <linux/module.h>
# include <linux/timer.h>
# include <linux/platform_device.h>
# include <linux/netdevice.h>
ieee802154: Introduce the use of the managed version of kzalloc
This patch moves data allocated using kzalloc to managed data allocated
using devm_kzalloc and cleans now unnecessary kfrees in probe and remove
functions. An explicit linux/device.h include is added to make sure
the devm_*() routine declarations are unambiguously available.
The following Coccinelle semantic patch was used for making the change:
@platform@
identifier p, probefn, removefn;
@@
struct platform_driver p = {
.probe = probefn,
.remove = removefn,
};
@prb@
identifier platform.probefn, pdev;
expression e, e1, e2;
@@
probefn(struct platform_device *pdev, ...) {
<+...
- e = kzalloc(e1, e2)
+ e = devm_kzalloc(&pdev->dev, e1, e2)
...
?-kfree(e);
...+>
}
@rem depends on prb@
identifier platform.removefn;
expression e;
@@
removefn(...) {
<...
- kfree(e);
...>
}
Signed-off-by: Himangi Saraogi <himangi774@gmail.com>
Acked-by: Julia Lawall <julia.lawall@lip6.fr>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-19 22:25:11 +05:30
# include <linux/device.h>
2012-05-15 20:50:30 +00:00
# include <linux/spinlock.h>
# include <net/mac802154.h>
2014-10-25 09:41:02 +02:00
# include <net/cfg802154.h>
2012-05-15 20:50:30 +00:00
2015-05-17 21:44:58 +02:00
static int numlbs = 2 ;
2015-05-17 21:45:05 +02:00
2015-05-17 21:45:03 +02:00
static LIST_HEAD ( fakelb_phys ) ;
2015-05-17 21:45:05 +02:00
static DEFINE_SPINLOCK ( fakelb_phys_lock ) ;
static LIST_HEAD ( fakelb_ifup_phys ) ;
static DEFINE_RWLOCK ( fakelb_ifup_phys_lock ) ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:00 +02:00
struct fakelb_phy {
2014-10-25 17:16:34 +02:00
struct ieee802154_hw * hw ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:06 +02:00
u8 page ;
u8 channel ;
2012-05-15 20:50:30 +00:00
struct list_head list ;
2015-05-17 21:45:05 +02:00
struct list_head list_ifup ;
2012-05-15 20:50:30 +00:00
} ;
static int
2014-10-25 17:16:34 +02:00
fakelb_hw_ed ( struct ieee802154_hw * hw , u8 * level )
2012-05-15 20:50:30 +00:00
{
BUG_ON ( ! level ) ;
* level = 0xbe ;
return 0 ;
}
static int
2014-10-28 18:21:19 +01:00
fakelb_hw_channel ( struct ieee802154_hw * hw , u8 page , u8 channel )
2012-05-15 20:50:30 +00:00
{
2015-05-17 21:45:06 +02:00
struct fakelb_phy * phy = hw - > priv ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:06 +02:00
write_lock_bh ( & fakelb_ifup_phys_lock ) ;
phy - > page = page ;
phy - > channel = channel ;
write_unlock_bh ( & fakelb_ifup_phys_lock ) ;
2012-05-15 20:50:30 +00:00
return 0 ;
}
static void
2015-05-17 21:45:00 +02:00
fakelb_hw_deliver ( struct fakelb_phy * phy , struct sk_buff * skb )
2012-05-15 20:50:30 +00:00
{
struct sk_buff * newskb ;
2015-05-17 21:45:05 +02:00
newskb = pskb_copy ( skb , GFP_ATOMIC ) ;
if ( newskb )
ieee802154_rx_irqsafe ( phy - > hw , newskb , 0xcc ) ;
2012-05-15 20:50:30 +00:00
}
static int
2014-10-25 17:16:34 +02:00
fakelb_hw_xmit ( struct ieee802154_hw * hw , struct sk_buff * skb )
2012-05-15 20:50:30 +00:00
{
2015-05-17 21:45:00 +02:00
struct fakelb_phy * current_phy = hw - > priv ;
2015-05-17 21:45:01 +02:00
struct fakelb_phy * phy ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:05 +02:00
read_lock_bh ( & fakelb_ifup_phys_lock ) ;
list_for_each_entry ( phy , & fakelb_ifup_phys , list_ifup ) {
2015-05-17 21:45:01 +02:00
if ( current_phy = = phy )
continue ;
2015-05-17 21:45:06 +02:00
if ( current_phy - > page = = phy - > page & &
current_phy - > channel = = phy - > channel )
2015-05-17 21:45:01 +02:00
fakelb_hw_deliver ( phy , skb ) ;
2012-05-15 20:50:30 +00:00
}
2015-05-17 21:45:05 +02:00
read_unlock_bh ( & fakelb_ifup_phys_lock ) ;
2012-05-15 20:50:30 +00:00
return 0 ;
}
static int
2014-10-25 17:16:34 +02:00
fakelb_hw_start ( struct ieee802154_hw * hw ) {
2015-05-17 21:45:00 +02:00
struct fakelb_phy * phy = hw - > priv ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:05 +02:00
write_lock_bh ( & fakelb_ifup_phys_lock ) ;
list_add ( & phy - > list_ifup , & fakelb_ifup_phys ) ;
write_unlock_bh ( & fakelb_ifup_phys_lock ) ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:05 +02:00
return 0 ;
2012-05-15 20:50:30 +00:00
}
static void
2014-10-25 17:16:34 +02:00
fakelb_hw_stop ( struct ieee802154_hw * hw ) {
2015-05-17 21:45:00 +02:00
struct fakelb_phy * phy = hw - > priv ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:05 +02:00
write_lock_bh ( & fakelb_ifup_phys_lock ) ;
list_del ( & phy - > list_ifup ) ;
write_unlock_bh ( & fakelb_ifup_phys_lock ) ;
2012-05-15 20:50:30 +00:00
}
2014-10-28 18:21:18 +01:00
static const struct ieee802154_ops fakelb_ops = {
2012-05-15 20:50:30 +00:00
. owner = THIS_MODULE ,
2014-10-26 09:37:08 +01:00
. xmit_sync = fakelb_hw_xmit ,
2012-05-15 20:50:30 +00:00
. ed = fakelb_hw_ed ,
. set_channel = fakelb_hw_channel ,
. start = fakelb_hw_start ,
. stop = fakelb_hw_stop ,
} ;
/* Number of dummy devices to be set up by this module. */
module_param ( numlbs , int , 0 ) ;
MODULE_PARM_DESC ( numlbs , " number of pseudo devices " ) ;
2015-05-17 21:45:03 +02:00
static int fakelb_add_one ( struct device * dev )
2012-05-15 20:50:30 +00:00
{
2015-05-17 21:45:00 +02:00
struct fakelb_phy * phy ;
2012-05-15 20:50:30 +00:00
int err ;
2014-10-25 17:16:34 +02:00
struct ieee802154_hw * hw ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:00 +02:00
hw = ieee802154_alloc_hw ( sizeof ( * phy ) , & fakelb_ops ) ;
2014-10-25 17:16:34 +02:00
if ( ! hw )
2012-05-15 20:50:30 +00:00
return - ENOMEM ;
2015-05-17 21:45:00 +02:00
phy = hw - > priv ;
phy - > hw = hw ;
2012-05-15 20:50:30 +00:00
/* 868 MHz BPSK 802.15.4-2003 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 0 ] | = 1 ;
2012-05-15 20:50:30 +00:00
/* 915 MHz BPSK 802.15.4-2003 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 0 ] | = 0x7fe ;
2012-05-15 20:50:30 +00:00
/* 2.4 GHz O-QPSK 802.15.4-2003 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 0 ] | = 0x7FFF800 ;
2012-05-15 20:50:30 +00:00
/* 868 MHz ASK 802.15.4-2006 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 1 ] | = 1 ;
2012-05-15 20:50:30 +00:00
/* 915 MHz ASK 802.15.4-2006 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 1 ] | = 0x7fe ;
2012-05-15 20:50:30 +00:00
/* 868 MHz O-QPSK 802.15.4-2006 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 2 ] | = 1 ;
2012-05-15 20:50:30 +00:00
/* 915 MHz O-QPSK 802.15.4-2006 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 2 ] | = 0x7fe ;
2012-05-15 20:50:30 +00:00
/* 2.4 GHz CSS 802.15.4a-2007 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 3 ] | = 0x3fff ;
2012-05-15 20:50:30 +00:00
/* UWB Sub-gigahertz 802.15.4a-2007 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 4 ] | = 1 ;
2012-05-15 20:50:30 +00:00
/* UWB Low band 802.15.4a-2007 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 4 ] | = 0x1e ;
2012-05-15 20:50:30 +00:00
/* UWB High band 802.15.4a-2007 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 4 ] | = 0xffe0 ;
2012-05-15 20:50:30 +00:00
/* 750 MHz O-QPSK 802.15.4c-2009 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 5 ] | = 0xf ;
2012-05-15 20:50:30 +00:00
/* 750 MHz MPSK 802.15.4c-2009 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 5 ] | = 0xf0 ;
2012-05-15 20:50:30 +00:00
/* 950 MHz BPSK 802.15.4d-2009 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 6 ] | = 0x3ff ;
2012-05-15 20:50:30 +00:00
/* 950 MHz GFSK 802.15.4d-2009 */
2015-05-17 21:44:42 +02:00
hw - > phy - > supported . channels [ 6 ] | = 0x3ffc00 ;
2012-05-15 20:50:30 +00:00
2014-10-25 17:16:34 +02:00
hw - > parent = dev ;
2012-05-15 20:50:30 +00:00
2014-10-25 17:16:34 +02:00
err = ieee802154_register_hw ( hw ) ;
2012-05-15 20:50:30 +00:00
if ( err )
goto err_reg ;
2015-05-17 21:45:05 +02:00
spin_lock ( & fakelb_phys_lock ) ;
2015-05-17 21:45:03 +02:00
list_add_tail ( & phy - > list , & fakelb_phys ) ;
2015-05-17 21:45:05 +02:00
spin_unlock ( & fakelb_phys_lock ) ;
2012-05-15 20:50:30 +00:00
return 0 ;
err_reg :
2015-05-17 21:45:00 +02:00
ieee802154_free_hw ( phy - > hw ) ;
2012-05-15 20:50:30 +00:00
return err ;
}
2015-05-17 21:45:00 +02:00
static void fakelb_del ( struct fakelb_phy * phy )
2012-05-15 20:50:30 +00:00
{
2015-05-17 21:45:00 +02:00
list_del ( & phy - > list ) ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:00 +02:00
ieee802154_unregister_hw ( phy - > hw ) ;
ieee802154_free_hw ( phy - > hw ) ;
2012-05-15 20:50:30 +00:00
}
2012-12-03 09:24:12 -05:00
static int fakelb_probe ( struct platform_device * pdev )
2012-05-15 20:50:30 +00:00
{
2015-05-17 21:45:00 +02:00
struct fakelb_phy * phy , * tmp ;
2012-05-15 20:50:30 +00:00
int err = - ENOMEM ;
int i ;
for ( i = 0 ; i < numlbs ; i + + ) {
2015-05-17 21:45:03 +02:00
err = fakelb_add_one ( & pdev - > dev ) ;
2012-05-15 20:50:30 +00:00
if ( err < 0 )
goto err_slave ;
}
dev_info ( & pdev - > dev , " added ieee802154 hardware \n " ) ;
return 0 ;
err_slave :
2015-05-17 21:45:05 +02:00
spin_lock ( & fakelb_phys_lock ) ;
2015-05-17 21:45:03 +02:00
list_for_each_entry_safe ( phy , tmp , & fakelb_phys , list )
2015-05-17 21:45:00 +02:00
fakelb_del ( phy ) ;
2015-05-17 21:45:05 +02:00
spin_unlock ( & fakelb_phys_lock ) ;
2012-05-15 20:50:30 +00:00
return err ;
}
2012-12-03 09:24:12 -05:00
static int fakelb_remove ( struct platform_device * pdev )
2012-05-15 20:50:30 +00:00
{
2015-05-17 21:45:00 +02:00
struct fakelb_phy * phy , * temp ;
2012-05-15 20:50:30 +00:00
2015-05-17 21:45:05 +02:00
spin_lock ( & fakelb_phys_lock ) ;
2015-05-17 21:45:03 +02:00
list_for_each_entry_safe ( phy , temp , & fakelb_phys , list )
2015-05-17 21:45:00 +02:00
fakelb_del ( phy ) ;
2015-05-17 21:45:05 +02:00
spin_unlock ( & fakelb_phys_lock ) ;
2012-05-15 20:50:30 +00:00
return 0 ;
}
static struct platform_device * ieee802154fake_dev ;
static struct platform_driver ieee802154fake_driver = {
. probe = fakelb_probe ,
2012-12-03 09:24:12 -05:00
. remove = fakelb_remove ,
2012-05-15 20:50:30 +00:00
. driver = {
. name = " ieee802154fakelb " ,
} ,
} ;
static __init int fakelb_init_module ( void )
{
ieee802154fake_dev = platform_device_register_simple (
" ieee802154fakelb " , - 1 , NULL , 0 ) ;
return platform_driver_register ( & ieee802154fake_driver ) ;
}
static __exit void fake_remove_module ( void )
{
platform_driver_unregister ( & ieee802154fake_driver ) ;
platform_device_unregister ( ieee802154fake_dev ) ;
}
module_init ( fakelb_init_module ) ;
module_exit ( fake_remove_module ) ;
MODULE_LICENSE ( " GPL " ) ;