2012-05-15 20:50:22 +00:00
/*
* 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 :
* Dmitry Eremin - Solenikov < dbaryshkov @ gmail . com >
* Sergey Lapin < slapin @ ossfans . org >
* Maxim Gorbachyov < maxim . gorbachev @ siemens . com >
* Alexander Smirnov < alex . bluesman . smirnov @ gmail . com >
*/
# include <linux/netdevice.h>
# include <linux/if_arp.h>
# include <linux/crc-ccitt.h>
2013-04-03 04:00:56 +00:00
# include <net/ieee802154_netdev.h>
2012-05-15 20:50:22 +00:00
# include <net/mac802154.h>
2014-10-25 09:41:02 +02:00
# include <net/cfg802154.h>
2012-05-15 20:50:22 +00:00
2014-10-25 09:41:00 +02:00
# include "ieee802154_i.h"
2012-05-15 20:50:22 +00:00
/* IEEE 802.15.4 transceivers can sleep during the xmit session, so process
* packets through the workqueue .
*/
struct xmit_work {
struct sk_buff * skb ;
struct work_struct work ;
2014-10-25 17:16:35 +02:00
struct ieee802154_local * local ;
2012-05-15 20:50:22 +00:00
u8 chan ;
u8 page ;
} ;
static void mac802154_xmit_worker ( struct work_struct * work )
{
struct xmit_work * xw = container_of ( work , struct xmit_work , work ) ;
2014-10-25 17:16:36 +02:00
struct ieee802154_sub_if_data * sdata ;
2012-05-15 20:50:22 +00:00
int res ;
2014-10-25 17:16:35 +02:00
mutex_lock ( & xw - > local - > phy - > pib_lock ) ;
if ( xw - > local - > phy - > current_channel ! = xw - > chan | |
xw - > local - > phy - > current_page ! = xw - > page ) {
res = xw - > local - > ops - > set_channel ( & xw - > local - > hw ,
2012-05-15 20:50:22 +00:00
xw - > page ,
xw - > chan ) ;
if ( res ) {
pr_debug ( " set_channel failed \n " ) ;
goto out ;
}
2013-04-05 13:03:10 +00:00
2014-10-25 17:16:35 +02:00
xw - > local - > phy - > current_channel = xw - > chan ;
xw - > local - > phy - > current_page = xw - > page ;
2012-05-15 20:50:22 +00:00
}
2014-10-25 17:16:35 +02:00
res = xw - > local - > ops - > xmit ( & xw - > local - > hw , xw - > skb ) ;
2013-04-03 04:00:55 +00:00
if ( res )
pr_debug ( " transmission failed \n " ) ;
2012-05-15 20:50:22 +00:00
out :
2014-10-25 17:16:35 +02:00
mutex_unlock ( & xw - > local - > phy - > pib_lock ) ;
2012-05-15 20:50:22 +00:00
2013-04-03 04:00:56 +00:00
/* Restart the netif queue on each sub_if_data object. */
rcu_read_lock ( ) ;
2014-10-25 17:16:38 +02:00
list_for_each_entry_rcu ( sdata , & xw - > local - > interfaces , list )
2013-04-03 04:00:56 +00:00
netif_wake_queue ( sdata - > dev ) ;
rcu_read_unlock ( ) ;
2012-05-15 20:50:22 +00:00
dev_kfree_skb ( xw - > skb ) ;
kfree ( xw ) ;
}
2014-10-25 17:16:35 +02:00
netdev_tx_t mac802154_tx ( struct ieee802154_local * local , struct sk_buff * skb ,
2012-05-15 20:50:22 +00:00
u8 page , u8 chan )
{
struct xmit_work * work ;
2014-10-25 17:16:36 +02:00
struct ieee802154_sub_if_data * sdata ;
2012-05-15 20:50:22 +00:00
2014-10-25 17:16:35 +02:00
if ( ! ( local - > phy - > channels_supported [ page ] & ( 1 < < chan ) ) ) {
2012-05-15 20:50:22 +00:00
WARN_ON ( 1 ) ;
2014-08-11 13:25:10 +02:00
goto err_tx ;
2012-06-25 03:30:13 +00:00
}
2012-05-15 20:50:22 +00:00
2014-10-25 17:16:35 +02:00
mac802154_monitors_rx ( mac802154_to_priv ( & local - > hw ) , skb ) ;
2012-06-25 23:24:54 +00:00
2014-10-25 17:16:35 +02:00
if ( ! ( local - > hw . flags & IEEE802154_HW_OMIT_CKSUM ) ) {
2012-05-15 20:50:22 +00:00
u16 crc = crc_ccitt ( 0 , skb - > data , skb - > len ) ;
u8 * data = skb_put ( skb , 2 ) ;
2014-07-02 09:01:09 +05:30
2012-05-15 20:50:22 +00:00
data [ 0 ] = crc & 0xff ;
data [ 1 ] = crc > > 8 ;
}
2014-10-25 17:16:35 +02:00
if ( skb_cow_head ( skb , local - > hw . extra_tx_headroom ) )
2014-08-11 13:25:10 +02:00
goto err_tx ;
2012-05-15 20:50:22 +00:00
2014-08-11 13:25:10 +02:00
work = kzalloc ( sizeof ( * work ) , GFP_ATOMIC ) ;
2012-11-29 18:25:10 +00:00
if ( ! work ) {
kfree_skb ( skb ) ;
2012-05-15 20:50:22 +00:00
return NETDEV_TX_BUSY ;
2012-11-29 18:25:10 +00:00
}
2012-05-15 20:50:22 +00:00
2013-04-03 04:00:56 +00:00
/* Stop the netif queue on each sub_if_data object. */
rcu_read_lock ( ) ;
2014-10-25 17:16:38 +02:00
list_for_each_entry_rcu ( sdata , & local - > interfaces , list )
2013-04-03 04:00:56 +00:00
netif_stop_queue ( sdata - > dev ) ;
rcu_read_unlock ( ) ;
2012-05-15 20:50:22 +00:00
INIT_WORK ( & work - > work , mac802154_xmit_worker ) ;
work - > skb = skb ;
2014-10-25 17:16:35 +02:00
work - > local = local ;
2012-05-15 20:50:22 +00:00
work - > page = page ;
work - > chan = chan ;
2014-10-25 17:16:35 +02:00
queue_work ( local - > dev_workqueue , & work - > work ) ;
2012-05-15 20:50:22 +00:00
return NETDEV_TX_OK ;
2014-08-11 13:25:10 +02:00
err_tx :
kfree_skb ( skb ) ;
return NETDEV_TX_OK ;
2012-05-15 20:50:22 +00:00
}