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>
2014-10-27 17:13:28 +01:00
# include <asm/unaligned.h>
2012-05-15 20:50:22 +00:00
2014-10-26 09:37:09 +01:00
# include <net/rtnetlink.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"
2014-10-28 18:21:21 +01:00
# include "driver-ops.h"
2012-05-15 20:50:22 +00:00
2015-07-21 17:44:47 +03:00
void ieee802154_xmit_worker ( struct work_struct * work )
2012-05-15 20:50:22 +00:00
{
2015-07-21 17:44:47 +03:00
struct ieee802154_local * local =
container_of ( work , struct ieee802154_local , tx_work ) ;
struct sk_buff * skb = local - > tx_skb ;
2014-10-26 09:37:12 +01:00
struct net_device * dev = skb - > dev ;
2012-05-15 20:50:22 +00:00
int res ;
2014-10-28 18:21:21 +01:00
res = drv_xmit_sync ( local , skb ) ;
2014-10-26 09:37:09 +01:00
if ( res )
goto err_tx ;
2014-11-12 19:51:56 +01:00
ieee802154_xmit_complete ( & local - > hw , skb , false ) ;
2014-10-26 09:37:09 +01:00
2014-10-26 09:37:12 +01:00
dev - > stats . tx_packets + + ;
dev - > stats . tx_bytes + = skb - > len ;
2014-10-26 09:37:09 +01:00
return ;
err_tx :
/* Restart the netif queue on each sub_if_data object. */
ieee802154_wake_queue ( & local - > hw ) ;
kfree_skb ( skb ) ;
2014-10-26 09:37:12 +01:00
netdev_dbg ( dev , " transmission failed \n " ) ;
2012-05-15 20:50:22 +00:00
}
2014-10-26 09:37:04 +01:00
static netdev_tx_t
2014-10-26 09:37:13 +01:00
ieee802154_tx ( struct ieee802154_local * local , struct sk_buff * skb )
2012-05-15 20:50:22 +00:00
{
2014-10-26 09:37:12 +01:00
struct net_device * dev = skb - > dev ;
2014-10-26 09:37:08 +01:00
int ret ;
2012-05-15 20:50:22 +00:00
2014-10-29 21:34:34 +01:00
if ( ! ( local - > hw . flags & IEEE802154_HW_TX_OMIT_CKSUM ) ) {
2018-07-02 16:32:03 -04:00
struct sk_buff * nskb ;
u16 crc ;
2014-07-02 09:01:09 +05:30
2018-07-02 16:32:03 -04:00
if ( unlikely ( skb_tailroom ( skb ) < IEEE802154_FCS_LEN ) ) {
nskb = skb_copy_expand ( skb , 0 , IEEE802154_FCS_LEN ,
GFP_ATOMIC ) ;
if ( likely ( nskb ) ) {
consume_skb ( skb ) ;
skb = nskb ;
} else {
goto err_tx ;
}
}
crc = crc_ccitt ( 0 , skb - > data , skb - > len ) ;
2014-10-27 17:13:28 +01:00
put_unaligned_le16 ( crc , skb_put ( skb , 2 ) ) ;
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. */
2014-10-26 09:37:06 +01:00
ieee802154_stop_queue ( & local - > hw ) ;
2013-04-03 04:00:56 +00:00
2014-10-26 09:37:08 +01:00
/* async is priority, otherwise sync is fallback */
if ( local - > ops - > xmit_async ) {
2014-10-28 18:21:21 +01:00
ret = drv_xmit_async ( local , skb ) ;
2014-10-26 09:37:08 +01:00
if ( ret ) {
ieee802154_wake_queue ( & local - > hw ) ;
goto err_tx ;
}
2014-10-26 09:37:12 +01:00
dev - > stats . tx_packets + + ;
dev - > stats . tx_bytes + = skb - > len ;
2014-10-26 09:37:08 +01:00
} else {
2015-07-21 17:44:47 +03:00
local - > tx_skb = skb ;
queue_work ( local - > workqueue , & local - > tx_work ) ;
2014-10-26 09:37:08 +01:00
}
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
2014-10-26 09:37:13 +01:00
netdev_tx_t
ieee802154_monitor_start_xmit ( struct sk_buff * skb , struct net_device * dev )
2014-10-26 09:37:01 +01:00
{
struct ieee802154_sub_if_data * sdata = IEEE802154_DEV_TO_SUB_IF ( dev ) ;
skb - > skb_iif = dev - > ifindex ;
2014-10-26 09:37:13 +01:00
return ieee802154_tx ( sdata - > local , skb ) ;
2014-10-26 09:37:01 +01:00
}
2014-10-26 09:37:13 +01:00
netdev_tx_t
ieee802154_subif_start_xmit ( struct sk_buff * skb , struct net_device * dev )
2014-10-26 09:37:01 +01:00
{
struct ieee802154_sub_if_data * sdata = IEEE802154_DEV_TO_SUB_IF ( dev ) ;
int rc ;
2015-09-28 09:00:26 +02:00
/* TODO we should move it to wpan_dev_hard_header and dev_hard_header
* functions . The reason is wireshark will show a mac header which is
* with security fields but the payload is not encrypted .
*/
2014-10-26 09:37:01 +01:00
rc = mac802154_llsec_encrypt ( & sdata - > sec , skb ) ;
if ( rc ) {
2014-10-26 09:37:10 +01:00
netdev_warn ( dev , " encryption failed: %i \n " , rc ) ;
2014-10-26 09:37:01 +01:00
kfree_skb ( skb ) ;
return NETDEV_TX_OK ;
}
skb - > skb_iif = dev - > ifindex ;
2014-10-26 09:37:13 +01:00
return ieee802154_tx ( sdata - > local , skb ) ;
2014-10-26 09:37:01 +01:00
}