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 .
*/
2014-10-26 09:37:02 +01:00
struct wpan_xmit_cb {
2012-05-15 20:50:22 +00:00
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 ;
} ;
2014-10-26 09:37:02 +01:00
static inline struct wpan_xmit_cb * wpan_xmit_cb ( const struct sk_buff * skb )
{
BUILD_BUG_ON ( sizeof ( skb - > cb ) < sizeof ( struct wpan_xmit_cb ) ) ;
return ( struct wpan_xmit_cb * ) skb - > cb ;
}
2012-05-15 20:50:22 +00:00
static void mac802154_xmit_worker ( struct work_struct * work )
{
2014-10-26 09:37:02 +01:00
struct wpan_xmit_cb * cb = container_of ( work , struct wpan_xmit_cb , work ) ;
2014-10-26 09:37:03 +01:00
struct ieee802154_local * local = cb - > local ;
2014-10-25 17:16:36 +02:00
struct ieee802154_sub_if_data * sdata ;
2014-10-26 09:37:03 +01:00
struct sk_buff * skb = cb - > skb ;
2012-05-15 20:50:22 +00:00
int res ;
2014-10-26 09:37:03 +01:00
mutex_lock ( & local - > phy - > pib_lock ) ;
if ( local - > phy - > current_channel ! = cb - > chan | |
local - > phy - > current_page ! = cb - > page ) {
res = local - > ops - > set_channel ( & local - > hw , cb - > page , cb - > chan ) ;
2012-05-15 20:50:22 +00:00
if ( res ) {
pr_debug ( " set_channel failed \n " ) ;
goto out ;
}
2013-04-05 13:03:10 +00:00
2014-10-26 09:37:03 +01:00
local - > phy - > current_channel = cb - > chan ;
local - > phy - > current_page = cb - > page ;
2012-05-15 20:50:22 +00:00
}
2014-10-26 09:37:03 +01:00
res = local - > ops - > xmit ( & local - > hw , 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-26 09:37:03 +01:00
mutex_unlock ( & 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-26 09:37:03 +01:00
list_for_each_entry_rcu ( sdata , & 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
2014-10-26 09:37:03 +01:00
dev_kfree_skb ( skb ) ;
2012-05-15 20:50:22 +00:00
}
2014-10-26 09:37:01 +01:00
static netdev_tx_t mac802154_tx ( struct ieee802154_local * local ,
struct sk_buff * skb , u8 page , u8 chan )
2012-05-15 20:50:22 +00:00
{
2014-10-25 17:16:36 +02:00
struct ieee802154_sub_if_data * sdata ;
2014-10-26 09:37:02 +01:00
struct wpan_xmit_cb * cb = wpan_xmit_cb ( skb ) ;
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:39 +02:00
mac802154_monitors_rx ( local , 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
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 ( ) ;
2014-10-26 09:37:02 +01:00
INIT_WORK ( & cb - > work , mac802154_xmit_worker ) ;
cb - > skb = skb ;
cb - > local = local ;
cb - > page = page ;
cb - > chan = chan ;
2012-05-15 20:50:22 +00:00
2014-10-26 09:37:02 +01:00
queue_work ( local - > workqueue , & cb - > 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
}
2014-10-26 09:37:01 +01:00
netdev_tx_t mac802154_monitor_xmit ( struct sk_buff * skb , struct net_device * dev )
{
struct ieee802154_sub_if_data * sdata = IEEE802154_DEV_TO_SUB_IF ( dev ) ;
u8 chan , page ;
/* FIXME: locking */
chan = sdata - > local - > phy - > current_channel ;
page = sdata - > local - > phy - > current_page ;
if ( chan = = MAC802154_CHAN_NONE ) /* not initialized */
return NETDEV_TX_OK ;
if ( WARN_ON ( page > = WPAN_NUM_PAGES ) | |
WARN_ON ( chan > = WPAN_NUM_CHANNELS ) )
return NETDEV_TX_OK ;
skb - > skb_iif = dev - > ifindex ;
dev - > stats . tx_packets + + ;
dev - > stats . tx_bytes + = skb - > len ;
return mac802154_tx ( sdata - > local , skb , page , chan ) ;
}
netdev_tx_t mac802154_wpan_xmit ( struct sk_buff * skb , struct net_device * dev )
{
struct ieee802154_sub_if_data * sdata = IEEE802154_DEV_TO_SUB_IF ( dev ) ;
u8 chan , page ;
int rc ;
spin_lock_bh ( & sdata - > mib_lock ) ;
chan = sdata - > chan ;
page = sdata - > page ;
spin_unlock_bh ( & sdata - > mib_lock ) ;
if ( chan = = MAC802154_CHAN_NONE | |
page > = WPAN_NUM_PAGES | |
chan > = WPAN_NUM_CHANNELS ) {
kfree_skb ( skb ) ;
return NETDEV_TX_OK ;
}
rc = mac802154_llsec_encrypt ( & sdata - > sec , skb ) ;
if ( rc ) {
pr_warn ( " encryption failed: %i \n " , rc ) ;
kfree_skb ( skb ) ;
return NETDEV_TX_OK ;
}
skb - > skb_iif = dev - > ifindex ;
dev - > stats . tx_packets + + ;
dev - > stats . tx_bytes + = skb - > len ;
return mac802154_tx ( sdata - > local , skb , page , chan ) ;
}