Commit 34e0b945 authored by David S. Miller's avatar David S. Miller

Merge tag 'ieee802154-for-net-next-2022-10-25' of...

Merge tag 'ieee802154-for-net-next-2022-10-25' of git://git.kernel.org/pub/scm/linux/kernel/git/sschmidt/wpan-next

Stefan Schmidt says:

====================

==
One of the biggest cycles for ieee802154 in a long time. We are landing the
first pieces of a big enhancements in managing PAN's. We might have another pull
request ready for this cycle later on, but I want to get this one out first.

Miquel Raynal added support for sending frames synchronously as a dependency
to handle MLME commands. Also introducing more filtering levels to match with
the needs of a device when scanning or operating as a pan coordinator.
To support development and testing the hwsim driver for ieee802154 was also
enhanced for the new filtering levels and to update the PIB attributes.

Alexander Aring fixed quite a few bugs spotted during reviewing changes. He
also added support for TRAC in the atusb driver to have better failure
handling if the firmware provides the needed information.

Jilin Yuan fixed a comment with a repeated word in it.
==================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 9c8dddab 4161634b
...@@ -191,7 +191,7 @@ static void atusb_work_urbs(struct work_struct *work) ...@@ -191,7 +191,7 @@ static void atusb_work_urbs(struct work_struct *work)
/* ----- Asynchronous USB -------------------------------------------------- */ /* ----- Asynchronous USB -------------------------------------------------- */
static void atusb_tx_done(struct atusb *atusb, u8 seq) static void atusb_tx_done(struct atusb *atusb, u8 seq, int reason)
{ {
struct usb_device *usb_dev = atusb->usb_dev; struct usb_device *usb_dev = atusb->usb_dev;
u8 expect = atusb->tx_ack_seq; u8 expect = atusb->tx_ack_seq;
...@@ -199,7 +199,10 @@ static void atusb_tx_done(struct atusb *atusb, u8 seq) ...@@ -199,7 +199,10 @@ static void atusb_tx_done(struct atusb *atusb, u8 seq)
dev_dbg(&usb_dev->dev, "%s (0x%02x/0x%02x)\n", __func__, seq, expect); dev_dbg(&usb_dev->dev, "%s (0x%02x/0x%02x)\n", __func__, seq, expect);
if (seq == expect) { if (seq == expect) {
/* TODO check for ifs handling in firmware */ /* TODO check for ifs handling in firmware */
ieee802154_xmit_complete(atusb->hw, atusb->tx_skb, false); if (reason == IEEE802154_SUCCESS)
ieee802154_xmit_complete(atusb->hw, atusb->tx_skb, false);
else
ieee802154_xmit_error(atusb->hw, atusb->tx_skb, reason);
} else { } else {
/* TODO I experience this case when atusb has a tx complete /* TODO I experience this case when atusb has a tx complete
* irq before probing, we should fix the firmware it's an * irq before probing, we should fix the firmware it's an
...@@ -215,7 +218,8 @@ static void atusb_in_good(struct urb *urb) ...@@ -215,7 +218,8 @@ static void atusb_in_good(struct urb *urb)
struct usb_device *usb_dev = urb->dev; struct usb_device *usb_dev = urb->dev;
struct sk_buff *skb = urb->context; struct sk_buff *skb = urb->context;
struct atusb *atusb = SKB_ATUSB(skb); struct atusb *atusb = SKB_ATUSB(skb);
u8 len, lqi; int result = IEEE802154_SUCCESS;
u8 len, lqi, trac;
if (!urb->actual_length) { if (!urb->actual_length) {
dev_dbg(&usb_dev->dev, "atusb_in: zero-sized URB ?\n"); dev_dbg(&usb_dev->dev, "atusb_in: zero-sized URB ?\n");
...@@ -224,8 +228,27 @@ static void atusb_in_good(struct urb *urb) ...@@ -224,8 +228,27 @@ static void atusb_in_good(struct urb *urb)
len = *skb->data; len = *skb->data;
if (urb->actual_length == 1) { switch (urb->actual_length) {
atusb_tx_done(atusb, len); case 2:
trac = TRAC_MASK(*(skb->data + 1));
switch (trac) {
case TRAC_SUCCESS:
case TRAC_SUCCESS_DATA_PENDING:
/* already IEEE802154_SUCCESS */
break;
case TRAC_CHANNEL_ACCESS_FAILURE:
result = IEEE802154_CHANNEL_ACCESS_FAILURE;
break;
case TRAC_NO_ACK:
result = IEEE802154_NO_ACK;
break;
default:
result = IEEE802154_SYSTEM_ERROR;
}
fallthrough;
case 1:
atusb_tx_done(atusb, len, result);
return; return;
} }
......
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include <linux/netdevice.h> #include <linux/netdevice.h>
#include <linux/device.h> #include <linux/device.h>
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <net/ieee802154_netdev.h>
#include <net/mac802154.h> #include <net/mac802154.h>
#include <net/cfg802154.h> #include <net/cfg802154.h>
#include <net/genetlink.h> #include <net/genetlink.h>
...@@ -47,6 +48,8 @@ static const struct genl_multicast_group hwsim_mcgrps[] = { ...@@ -47,6 +48,8 @@ static const struct genl_multicast_group hwsim_mcgrps[] = {
struct hwsim_pib { struct hwsim_pib {
u8 page; u8 page;
u8 channel; u8 channel;
struct ieee802154_hw_addr_filt filt;
enum ieee802154_filtering_level filt_level;
struct rcu_head rcu; struct rcu_head rcu;
}; };
...@@ -88,24 +91,168 @@ static int hwsim_hw_ed(struct ieee802154_hw *hw, u8 *level) ...@@ -88,24 +91,168 @@ static int hwsim_hw_ed(struct ieee802154_hw *hw, u8 *level)
return 0; return 0;
} }
static int hwsim_hw_channel(struct ieee802154_hw *hw, u8 page, u8 channel) static int hwsim_update_pib(struct ieee802154_hw *hw, u8 page, u8 channel,
struct ieee802154_hw_addr_filt *filt,
enum ieee802154_filtering_level filt_level)
{ {
struct hwsim_phy *phy = hw->priv; struct hwsim_phy *phy = hw->priv;
struct hwsim_pib *pib, *pib_old; struct hwsim_pib *pib, *pib_old;
pib = kzalloc(sizeof(*pib), GFP_KERNEL); pib = kzalloc(sizeof(*pib), GFP_ATOMIC);
if (!pib) if (!pib)
return -ENOMEM; return -ENOMEM;
pib_old = rtnl_dereference(phy->pib);
pib->page = page; pib->page = page;
pib->channel = channel; pib->channel = channel;
pib->filt.short_addr = filt->short_addr;
pib->filt.pan_id = filt->pan_id;
pib->filt.ieee_addr = filt->ieee_addr;
pib->filt.pan_coord = filt->pan_coord;
pib->filt_level = filt_level;
pib_old = rtnl_dereference(phy->pib);
rcu_assign_pointer(phy->pib, pib); rcu_assign_pointer(phy->pib, pib);
kfree_rcu(pib_old, rcu); kfree_rcu(pib_old, rcu);
return 0; return 0;
} }
static int hwsim_hw_channel(struct ieee802154_hw *hw, u8 page, u8 channel)
{
struct hwsim_phy *phy = hw->priv;
struct hwsim_pib *pib;
int ret;
rcu_read_lock();
pib = rcu_dereference(phy->pib);
ret = hwsim_update_pib(hw, page, channel, &pib->filt, pib->filt_level);
rcu_read_unlock();
return ret;
}
static int hwsim_hw_addr_filt(struct ieee802154_hw *hw,
struct ieee802154_hw_addr_filt *filt,
unsigned long changed)
{
struct hwsim_phy *phy = hw->priv;
struct hwsim_pib *pib;
int ret;
rcu_read_lock();
pib = rcu_dereference(phy->pib);
ret = hwsim_update_pib(hw, pib->page, pib->channel, filt, pib->filt_level);
rcu_read_unlock();
return ret;
}
static void hwsim_hw_receive(struct ieee802154_hw *hw, struct sk_buff *skb,
u8 lqi)
{
struct ieee802154_hdr hdr;
struct hwsim_phy *phy = hw->priv;
struct hwsim_pib *pib;
rcu_read_lock();
pib = rcu_dereference(phy->pib);
if (!pskb_may_pull(skb, 3)) {
dev_dbg(hw->parent, "invalid frame\n");
goto drop;
}
memcpy(&hdr, skb->data, 3);
/* Level 4 filtering: Frame fields validity */
if (pib->filt_level == IEEE802154_FILTERING_4_FRAME_FIELDS) {
/* a) Drop reserved frame types */
switch (mac_cb(skb)->type) {
case IEEE802154_FC_TYPE_BEACON:
case IEEE802154_FC_TYPE_DATA:
case IEEE802154_FC_TYPE_ACK:
case IEEE802154_FC_TYPE_MAC_CMD:
break;
default:
dev_dbg(hw->parent, "unrecognized frame type 0x%x\n",
mac_cb(skb)->type);
goto drop;
}
/* b) Drop reserved frame versions */
switch (hdr.fc.version) {
case IEEE802154_2003_STD:
case IEEE802154_2006_STD:
case IEEE802154_STD:
break;
default:
dev_dbg(hw->parent,
"unrecognized frame version 0x%x\n",
hdr.fc.version);
goto drop;
}
/* c) PAN ID constraints */
if ((mac_cb(skb)->dest.mode == IEEE802154_ADDR_LONG ||
mac_cb(skb)->dest.mode == IEEE802154_ADDR_SHORT) &&
mac_cb(skb)->dest.pan_id != pib->filt.pan_id &&
mac_cb(skb)->dest.pan_id != cpu_to_le16(IEEE802154_PANID_BROADCAST)) {
dev_dbg(hw->parent,
"unrecognized PAN ID %04x\n",
le16_to_cpu(mac_cb(skb)->dest.pan_id));
goto drop;
}
/* d1) Short address constraints */
if (mac_cb(skb)->dest.mode == IEEE802154_ADDR_SHORT &&
mac_cb(skb)->dest.short_addr != pib->filt.short_addr &&
mac_cb(skb)->dest.short_addr != cpu_to_le16(IEEE802154_ADDR_BROADCAST)) {
dev_dbg(hw->parent,
"unrecognized short address %04x\n",
le16_to_cpu(mac_cb(skb)->dest.short_addr));
goto drop;
}
/* d2) Extended address constraints */
if (mac_cb(skb)->dest.mode == IEEE802154_ADDR_LONG &&
mac_cb(skb)->dest.extended_addr != pib->filt.ieee_addr) {
dev_dbg(hw->parent,
"unrecognized long address 0x%016llx\n",
mac_cb(skb)->dest.extended_addr);
goto drop;
}
/* d4) Specific PAN coordinator case (no parent) */
if ((mac_cb(skb)->type == IEEE802154_FC_TYPE_DATA ||
mac_cb(skb)->type == IEEE802154_FC_TYPE_MAC_CMD) &&
mac_cb(skb)->dest.mode == IEEE802154_ADDR_NONE) {
dev_dbg(hw->parent,
"relaying is not supported\n");
goto drop;
}
/* e) Beacon frames follow specific PAN ID rules */
if (mac_cb(skb)->type == IEEE802154_FC_TYPE_BEACON &&
pib->filt.pan_id != cpu_to_le16(IEEE802154_PANID_BROADCAST) &&
mac_cb(skb)->dest.pan_id != pib->filt.pan_id) {
dev_dbg(hw->parent,
"invalid beacon PAN ID %04x\n",
le16_to_cpu(mac_cb(skb)->dest.pan_id));
goto drop;
}
}
rcu_read_unlock();
ieee802154_rx_irqsafe(hw, skb, lqi);
return;
drop:
rcu_read_unlock();
kfree_skb(skb);
}
static int hwsim_hw_xmit(struct ieee802154_hw *hw, struct sk_buff *skb) static int hwsim_hw_xmit(struct ieee802154_hw *hw, struct sk_buff *skb)
{ {
struct hwsim_phy *current_phy = hw->priv; struct hwsim_phy *current_phy = hw->priv;
...@@ -133,8 +280,7 @@ static int hwsim_hw_xmit(struct ieee802154_hw *hw, struct sk_buff *skb) ...@@ -133,8 +280,7 @@ static int hwsim_hw_xmit(struct ieee802154_hw *hw, struct sk_buff *skb)
einfo = rcu_dereference(e->info); einfo = rcu_dereference(e->info);
if (newskb) if (newskb)
ieee802154_rx_irqsafe(e->endpoint->hw, newskb, hwsim_hw_receive(e->endpoint->hw, newskb, einfo->lqi);
einfo->lqi);
} }
} }
rcu_read_unlock(); rcu_read_unlock();
...@@ -148,6 +294,7 @@ static int hwsim_hw_start(struct ieee802154_hw *hw) ...@@ -148,6 +294,7 @@ static int hwsim_hw_start(struct ieee802154_hw *hw)
struct hwsim_phy *phy = hw->priv; struct hwsim_phy *phy = hw->priv;
phy->suspended = false; phy->suspended = false;
return 0; return 0;
} }
...@@ -161,7 +308,22 @@ static void hwsim_hw_stop(struct ieee802154_hw *hw) ...@@ -161,7 +308,22 @@ static void hwsim_hw_stop(struct ieee802154_hw *hw)
static int static int
hwsim_set_promiscuous_mode(struct ieee802154_hw *hw, const bool on) hwsim_set_promiscuous_mode(struct ieee802154_hw *hw, const bool on)
{ {
return 0; enum ieee802154_filtering_level filt_level;
struct hwsim_phy *phy = hw->priv;
struct hwsim_pib *pib;
int ret;
if (on)
filt_level = IEEE802154_FILTERING_NONE;
else
filt_level = IEEE802154_FILTERING_4_FRAME_FIELDS;
rcu_read_lock();
pib = rcu_dereference(phy->pib);
ret = hwsim_update_pib(hw, pib->page, pib->channel, &pib->filt, filt_level);
rcu_read_unlock();
return ret;
} }
static const struct ieee802154_ops hwsim_ops = { static const struct ieee802154_ops hwsim_ops = {
...@@ -172,6 +334,7 @@ static const struct ieee802154_ops hwsim_ops = { ...@@ -172,6 +334,7 @@ static const struct ieee802154_ops hwsim_ops = {
.start = hwsim_hw_start, .start = hwsim_hw_start,
.stop = hwsim_hw_stop, .stop = hwsim_hw_stop,
.set_promiscuous_mode = hwsim_set_promiscuous_mode, .set_promiscuous_mode = hwsim_set_promiscuous_mode,
.set_hw_addr_filt = hwsim_hw_addr_filt,
}; };
static int hwsim_new_radio_nl(struct sk_buff *msg, struct genl_info *info) static int hwsim_new_radio_nl(struct sk_buff *msg, struct genl_info *info)
...@@ -788,11 +951,13 @@ static int hwsim_add_one(struct genl_info *info, struct device *dev, ...@@ -788,11 +951,13 @@ static int hwsim_add_one(struct genl_info *info, struct device *dev,
} }
pib->channel = 13; pib->channel = 13;
pib->filt.short_addr = cpu_to_le16(IEEE802154_ADDR_BROADCAST);
pib->filt.pan_id = cpu_to_le16(IEEE802154_PANID_BROADCAST);
rcu_assign_pointer(phy->pib, pib); rcu_assign_pointer(phy->pib, pib);
phy->idx = idx; phy->idx = idx;
INIT_LIST_HEAD(&phy->edges); INIT_LIST_HEAD(&phy->edges);
hw->flags = IEEE802154_HW_PROMISCUOUS | IEEE802154_HW_RX_DROP_BAD_CKSUM; hw->flags = IEEE802154_HW_PROMISCUOUS;
hw->parent = dev; hw->parent = dev;
err = ieee802154_register_hw(hw); err = ieee802154_register_hw(hw);
......
...@@ -1233,12 +1233,9 @@ mcr20a_probe(struct spi_device *spi) ...@@ -1233,12 +1233,9 @@ mcr20a_probe(struct spi_device *spi)
} }
rst_b = devm_gpiod_get(&spi->dev, "rst_b", GPIOD_OUT_HIGH); rst_b = devm_gpiod_get(&spi->dev, "rst_b", GPIOD_OUT_HIGH);
if (IS_ERR(rst_b)) { if (IS_ERR(rst_b))
ret = PTR_ERR(rst_b); return dev_err_probe(&spi->dev, PTR_ERR(rst_b),
if (ret != -EPROBE_DEFER) "Failed to get 'rst_b' gpio");
dev_err(&spi->dev, "Failed to get 'rst_b' gpio: %d", ret);
return ret;
}
/* reset mcr20a */ /* reset mcr20a */
usleep_range(10, 20); usleep_range(10, 20);
......
...@@ -276,6 +276,30 @@ enum { ...@@ -276,6 +276,30 @@ enum {
IEEE802154_SYSTEM_ERROR = 0xff, IEEE802154_SYSTEM_ERROR = 0xff,
}; };
/**
* enum ieee802154_filtering_level - Filtering levels applicable to a PHY
*
* @IEEE802154_FILTERING_NONE: No filtering at all, what is received is
* forwarded to the softMAC
* @IEEE802154_FILTERING_1_FCS: First filtering level, frames with an invalid
* FCS should be dropped
* @IEEE802154_FILTERING_2_PROMISCUOUS: Second filtering level, promiscuous
* mode as described in the spec, identical in terms of filtering to the
* level one on PHY side, but at the MAC level the frame should be
* forwarded to the upper layer directly
* @IEEE802154_FILTERING_3_SCAN: Third filtering level, scan related, where
* only beacons must be processed, all remaining traffic gets dropped
* @IEEE802154_FILTERING_4_FRAME_FIELDS: Fourth filtering level actually
* enforcing the validity of the content of the frame with various checks
*/
enum ieee802154_filtering_level {
IEEE802154_FILTERING_NONE,
IEEE802154_FILTERING_1_FCS,
IEEE802154_FILTERING_2_PROMISCUOUS,
IEEE802154_FILTERING_3_SCAN,
IEEE802154_FILTERING_4_FRAME_FIELDS,
};
/* frame control handling */ /* frame control handling */
#define IEEE802154_FCTL_FTYPE 0x0003 #define IEEE802154_FCTL_FTYPE 0x0003
#define IEEE802154_FCTL_ACKREQ 0x0020 #define IEEE802154_FCTL_ACKREQ 0x0020
......
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
#include <linux/ieee802154.h> #include <linux/ieee802154.h>
#include <linux/netdevice.h> #include <linux/netdevice.h>
#include <linux/mutex.h> #include <linux/spinlock.h>
#include <linux/bug.h> #include <linux/bug.h>
#include <net/nl802154.h> #include <net/nl802154.h>
...@@ -166,11 +166,14 @@ wpan_phy_cca_cmp(const struct wpan_phy_cca *a, const struct wpan_phy_cca *b) ...@@ -166,11 +166,14 @@ wpan_phy_cca_cmp(const struct wpan_phy_cca *a, const struct wpan_phy_cca *b)
* level setting. * level setting.
* @WPAN_PHY_FLAG_CCA_MODE: Indicates that transceiver will support cca mode * @WPAN_PHY_FLAG_CCA_MODE: Indicates that transceiver will support cca mode
* setting. * setting.
* @WPAN_PHY_FLAG_STATE_QUEUE_STOPPED: Indicates that the transmit queue was
* temporarily stopped.
*/ */
enum wpan_phy_flags { enum wpan_phy_flags {
WPAN_PHY_FLAG_TXPOWER = BIT(1), WPAN_PHY_FLAG_TXPOWER = BIT(1),
WPAN_PHY_FLAG_CCA_ED_LEVEL = BIT(2), WPAN_PHY_FLAG_CCA_ED_LEVEL = BIT(2),
WPAN_PHY_FLAG_CCA_MODE = BIT(3), WPAN_PHY_FLAG_CCA_MODE = BIT(3),
WPAN_PHY_FLAG_STATE_QUEUE_STOPPED = BIT(4),
}; };
struct wpan_phy { struct wpan_phy {
...@@ -182,7 +185,7 @@ struct wpan_phy { ...@@ -182,7 +185,7 @@ struct wpan_phy {
*/ */
const void *privid; const void *privid;
u32 flags; unsigned long flags;
/* /*
* This is a PIB according to 802.15.4-2011. * This is a PIB according to 802.15.4-2011.
...@@ -214,6 +217,17 @@ struct wpan_phy { ...@@ -214,6 +217,17 @@ struct wpan_phy {
/* the network namespace this phy lives in currently */ /* the network namespace this phy lives in currently */
possible_net_t _net; possible_net_t _net;
/* Transmission monitoring and control */
spinlock_t queue_lock;
atomic_t ongoing_txs;
atomic_t hold_txs;
wait_queue_head_t sync_txq;
/* Current filtering level on reception.
* Only allowed to be changed if phy is not operational.
*/
enum ieee802154_filtering_level filtering;
char priv[] __aligned(NETDEV_ALIGN); char priv[] __aligned(NETDEV_ALIGN);
}; };
...@@ -365,8 +379,6 @@ struct wpan_dev { ...@@ -365,8 +379,6 @@ struct wpan_dev {
bool lbt; bool lbt;
bool promiscuous_mode;
/* fallback for acknowledgment bit setting */ /* fallback for acknowledgment bit setting */
bool ackreq; bool ackreq;
}; };
......
...@@ -85,6 +85,14 @@ struct ieee802154_hdr_fc { ...@@ -85,6 +85,14 @@ struct ieee802154_hdr_fc {
#endif #endif
}; };
enum ieee802154_frame_version {
IEEE802154_2003_STD,
IEEE802154_2006_STD,
IEEE802154_STD,
IEEE802154_RESERVED_STD,
IEEE802154_MULTIPURPOSE_STD = IEEE802154_2003_STD,
};
struct ieee802154_hdr { struct ieee802154_hdr {
struct ieee802154_hdr_fc fc; struct ieee802154_hdr_fc fc;
u8 seq; u8 seq;
......
...@@ -111,9 +111,6 @@ struct ieee802154_hw { ...@@ -111,9 +111,6 @@ struct ieee802154_hw {
* promiscuous mode setting. * promiscuous mode setting.
* *
* @IEEE802154_HW_RX_OMIT_CKSUM: Indicates that receiver omits FCS. * @IEEE802154_HW_RX_OMIT_CKSUM: Indicates that receiver omits FCS.
*
* @IEEE802154_HW_RX_DROP_BAD_CKSUM: Indicates that receiver will not filter
* frames with bad checksum.
*/ */
enum ieee802154_hw_flags { enum ieee802154_hw_flags {
IEEE802154_HW_TX_OMIT_CKSUM = BIT(0), IEEE802154_HW_TX_OMIT_CKSUM = BIT(0),
...@@ -123,7 +120,6 @@ enum ieee802154_hw_flags { ...@@ -123,7 +120,6 @@ enum ieee802154_hw_flags {
IEEE802154_HW_AFILT = BIT(4), IEEE802154_HW_AFILT = BIT(4),
IEEE802154_HW_PROMISCUOUS = BIT(5), IEEE802154_HW_PROMISCUOUS = BIT(5),
IEEE802154_HW_RX_OMIT_CKSUM = BIT(6), IEEE802154_HW_RX_OMIT_CKSUM = BIT(6),
IEEE802154_HW_RX_DROP_BAD_CKSUM = BIT(7),
}; };
/* Indicates that receiver omits FCS and xmitter will add FCS on it's own. */ /* Indicates that receiver omits FCS and xmitter will add FCS on it's own. */
...@@ -460,33 +456,6 @@ void ieee802154_unregister_hw(struct ieee802154_hw *hw); ...@@ -460,33 +456,6 @@ void ieee802154_unregister_hw(struct ieee802154_hw *hw);
*/ */
void ieee802154_rx_irqsafe(struct ieee802154_hw *hw, struct sk_buff *skb, void ieee802154_rx_irqsafe(struct ieee802154_hw *hw, struct sk_buff *skb,
u8 lqi); u8 lqi);
/**
* ieee802154_wake_queue - wake ieee802154 queue
* @hw: pointer as obtained from ieee802154_alloc_hw().
*
* Tranceivers usually have either one transmit framebuffer or one framebuffer
* for both transmitting and receiving. Hence, the core currently only handles
* one frame at a time for each phy, which means we had to stop the queue to
* avoid new skb to come during the transmission. The queue then needs to be
* woken up after the operation.
*
* Drivers should use this function instead of netif_wake_queue.
*/
void ieee802154_wake_queue(struct ieee802154_hw *hw);
/**
* ieee802154_stop_queue - stop ieee802154 queue
* @hw: pointer as obtained from ieee802154_alloc_hw().
*
* Tranceivers usually have either one transmit framebuffer or one framebuffer
* for both transmitting and receiving. Hence, the core currently only handles
* one frame at a time for each phy, which means we need to tell upper layers to
* stop giving us new skbs while we are busy with the transmitted one. The queue
* must then be stopped before transmitting.
*
* Drivers should use this function instead of netif_stop_queue.
*/
void ieee802154_stop_queue(struct ieee802154_hw *hw);
/** /**
* ieee802154_xmit_complete - frame transmission complete * ieee802154_xmit_complete - frame transmission complete
......
...@@ -129,6 +129,9 @@ wpan_phy_new(const struct cfg802154_ops *ops, size_t priv_size) ...@@ -129,6 +129,9 @@ wpan_phy_new(const struct cfg802154_ops *ops, size_t priv_size)
wpan_phy_net_set(&rdev->wpan_phy, &init_net); wpan_phy_net_set(&rdev->wpan_phy, &init_net);
init_waitqueue_head(&rdev->dev_wait); init_waitqueue_head(&rdev->dev_wait);
init_waitqueue_head(&rdev->wpan_phy.sync_txq);
spin_lock_init(&rdev->wpan_phy.queue_lock);
return &rdev->wpan_phy; return &rdev->wpan_phy;
} }
......
...@@ -46,7 +46,7 @@ static int ieee802154_suspend(struct wpan_phy *wpan_phy) ...@@ -46,7 +46,7 @@ static int ieee802154_suspend(struct wpan_phy *wpan_phy)
if (!local->open_count) if (!local->open_count)
goto suspend; goto suspend;
ieee802154_stop_queue(&local->hw); ieee802154_sync_and_hold_queue(local);
synchronize_net(); synchronize_net();
/* stop hardware - this must stop RX */ /* stop hardware - this must stop RX */
...@@ -67,12 +67,12 @@ static int ieee802154_resume(struct wpan_phy *wpan_phy) ...@@ -67,12 +67,12 @@ static int ieee802154_resume(struct wpan_phy *wpan_phy)
goto wake_up; goto wake_up;
/* restart hardware */ /* restart hardware */
ret = drv_start(local); ret = drv_start(local, local->phy->filtering, &local->addr_filt);
if (ret) if (ret)
return ret; return ret;
wake_up: wake_up:
ieee802154_wake_queue(&local->hw); ieee802154_release_queue(local);
local->suspended = false; local->suspended = false;
return 0; return 0;
} }
......
This diff is collapsed.
...@@ -26,6 +26,8 @@ struct ieee802154_local { ...@@ -26,6 +26,8 @@ struct ieee802154_local {
struct ieee802154_hw hw; struct ieee802154_hw hw;
const struct ieee802154_ops *ops; const struct ieee802154_ops *ops;
/* hardware address filter */
struct ieee802154_hw_addr_filt addr_filt;
/* ieee802154 phy */ /* ieee802154 phy */
struct wpan_phy *phy; struct wpan_phy *phy;
...@@ -55,7 +57,7 @@ struct ieee802154_local { ...@@ -55,7 +57,7 @@ struct ieee802154_local {
struct sk_buff_head skb_queue; struct sk_buff_head skb_queue;
struct sk_buff *tx_skb; struct sk_buff *tx_skb;
struct work_struct tx_work; struct work_struct sync_tx_work;
/* A negative Linux error code or a null/positive MLME error status */ /* A negative Linux error code or a null/positive MLME error status */
int tx_result; int tx_result;
}; };
...@@ -82,6 +84,16 @@ struct ieee802154_sub_if_data { ...@@ -82,6 +84,16 @@ struct ieee802154_sub_if_data {
struct ieee802154_local *local; struct ieee802154_local *local;
struct net_device *dev; struct net_device *dev;
/* Each interface starts and works in nominal state at a given filtering
* level given by iface_default_filtering, which is set once for all at
* the interface creation and should not evolve over time. For some MAC
* operations however, the filtering level may change temporarily, as
* reflected in the required_filtering field. The actual filtering at
* the PHY level may be different and is shown in struct wpan_phy.
*/
enum ieee802154_filtering_level iface_default_filtering;
enum ieee802154_filtering_level required_filtering;
unsigned long state; unsigned long state;
char name[IFNAMSIZ]; char name[IFNAMSIZ];
...@@ -123,13 +135,53 @@ ieee802154_sdata_running(struct ieee802154_sub_if_data *sdata) ...@@ -123,13 +135,53 @@ ieee802154_sdata_running(struct ieee802154_sub_if_data *sdata)
extern struct ieee802154_mlme_ops mac802154_mlme_wpan; extern struct ieee802154_mlme_ops mac802154_mlme_wpan;
void ieee802154_rx(struct ieee802154_local *local, struct sk_buff *skb); void ieee802154_rx(struct ieee802154_local *local, struct sk_buff *skb);
void ieee802154_xmit_worker(struct work_struct *work); void ieee802154_xmit_sync_worker(struct work_struct *work);
int ieee802154_sync_and_hold_queue(struct ieee802154_local *local);
int ieee802154_mlme_op_pre(struct ieee802154_local *local);
int ieee802154_mlme_tx(struct ieee802154_local *local,
struct ieee802154_sub_if_data *sdata,
struct sk_buff *skb);
void ieee802154_mlme_op_post(struct ieee802154_local *local);
int ieee802154_mlme_tx_one(struct ieee802154_local *local,
struct ieee802154_sub_if_data *sdata,
struct sk_buff *skb);
netdev_tx_t netdev_tx_t
ieee802154_monitor_start_xmit(struct sk_buff *skb, struct net_device *dev); ieee802154_monitor_start_xmit(struct sk_buff *skb, struct net_device *dev);
netdev_tx_t netdev_tx_t
ieee802154_subif_start_xmit(struct sk_buff *skb, struct net_device *dev); ieee802154_subif_start_xmit(struct sk_buff *skb, struct net_device *dev);
enum hrtimer_restart ieee802154_xmit_ifs_timer(struct hrtimer *timer); enum hrtimer_restart ieee802154_xmit_ifs_timer(struct hrtimer *timer);
/**
* ieee802154_hold_queue - hold ieee802154 queue
* @local: main mac object
*
* Hold a queue by incrementing an atomic counter and requesting the netif
* queues to be stopped. The queues cannot be woken up while the counter has not
* been reset with as any ieee802154_release_queue() calls as needed.
*/
void ieee802154_hold_queue(struct ieee802154_local *local);
/**
* ieee802154_release_queue - release ieee802154 queue
* @local: main mac object
*
* Release a queue which is held by decrementing an atomic counter and wake it
* up only if the counter reaches 0.
*/
void ieee802154_release_queue(struct ieee802154_local *local);
/**
* ieee802154_disable_queue - disable ieee802154 queue
* @local: main mac object
*
* When trying to sync the Tx queue, we cannot just stop the queue
* (which is basically a bit being set without proper lock handling)
* because it would be racy. We actually need to call netif_tx_disable()
* instead, which is done by this helper. Restarting the queue can
* however still be done with a regular wake call.
*/
void ieee802154_disable_queue(struct ieee802154_local *local);
/* MIB callbacks */ /* MIB callbacks */
void mac802154_dev_set_page_channel(struct net_device *dev, u8 page, u8 chan); void mac802154_dev_set_page_channel(struct net_device *dev, u8 page, u8 chan);
......
...@@ -147,25 +147,12 @@ static int ieee802154_setup_hw(struct ieee802154_sub_if_data *sdata) ...@@ -147,25 +147,12 @@ static int ieee802154_setup_hw(struct ieee802154_sub_if_data *sdata)
struct wpan_dev *wpan_dev = &sdata->wpan_dev; struct wpan_dev *wpan_dev = &sdata->wpan_dev;
int ret; int ret;
if (local->hw.flags & IEEE802154_HW_PROMISCUOUS) { sdata->required_filtering = sdata->iface_default_filtering;
ret = drv_set_promiscuous_mode(local,
wpan_dev->promiscuous_mode);
if (ret < 0)
return ret;
}
if (local->hw.flags & IEEE802154_HW_AFILT) { if (local->hw.flags & IEEE802154_HW_AFILT) {
ret = drv_set_pan_id(local, wpan_dev->pan_id); local->addr_filt.pan_id = wpan_dev->pan_id;
if (ret < 0) local->addr_filt.ieee_addr = wpan_dev->extended_addr;
return ret; local->addr_filt.short_addr = wpan_dev->short_addr;
ret = drv_set_extended_addr(local, wpan_dev->extended_addr);
if (ret < 0)
return ret;
ret = drv_set_short_addr(local, wpan_dev->short_addr);
if (ret < 0)
return ret;
} }
if (local->hw.flags & IEEE802154_HW_LBT) { if (local->hw.flags & IEEE802154_HW_LBT) {
...@@ -206,7 +193,8 @@ static int mac802154_slave_open(struct net_device *dev) ...@@ -206,7 +193,8 @@ static int mac802154_slave_open(struct net_device *dev)
if (res) if (res)
goto err; goto err;
res = drv_start(local); res = drv_start(local, sdata->required_filtering,
&local->addr_filt);
if (res) if (res)
goto err; goto err;
} }
...@@ -223,15 +211,16 @@ static int mac802154_slave_open(struct net_device *dev) ...@@ -223,15 +211,16 @@ static int mac802154_slave_open(struct net_device *dev)
static int static int
ieee802154_check_mac_settings(struct ieee802154_local *local, ieee802154_check_mac_settings(struct ieee802154_local *local,
struct wpan_dev *wpan_dev, struct ieee802154_sub_if_data *sdata,
struct wpan_dev *nwpan_dev) struct ieee802154_sub_if_data *nsdata)
{ {
struct wpan_dev *nwpan_dev = &nsdata->wpan_dev;
struct wpan_dev *wpan_dev = &sdata->wpan_dev;
ASSERT_RTNL(); ASSERT_RTNL();
if (local->hw.flags & IEEE802154_HW_PROMISCUOUS) { if (sdata->iface_default_filtering != nsdata->iface_default_filtering)
if (wpan_dev->promiscuous_mode != nwpan_dev->promiscuous_mode) return -EBUSY;
return -EBUSY;
}
if (local->hw.flags & IEEE802154_HW_AFILT) { if (local->hw.flags & IEEE802154_HW_AFILT) {
if (wpan_dev->pan_id != nwpan_dev->pan_id || if (wpan_dev->pan_id != nwpan_dev->pan_id ||
...@@ -285,8 +274,7 @@ ieee802154_check_concurrent_iface(struct ieee802154_sub_if_data *sdata, ...@@ -285,8 +274,7 @@ ieee802154_check_concurrent_iface(struct ieee802154_sub_if_data *sdata,
/* check all phy mac sublayer settings are the same. /* check all phy mac sublayer settings are the same.
* We have only one phy, different values makes trouble. * We have only one phy, different values makes trouble.
*/ */
ret = ieee802154_check_mac_settings(local, wpan_dev, ret = ieee802154_check_mac_settings(local, sdata, nsdata);
&nsdata->wpan_dev);
if (ret < 0) if (ret < 0)
return ret; return ret;
} }
...@@ -586,7 +574,7 @@ ieee802154_setup_sdata(struct ieee802154_sub_if_data *sdata, ...@@ -586,7 +574,7 @@ ieee802154_setup_sdata(struct ieee802154_sub_if_data *sdata,
sdata->dev->priv_destructor = mac802154_wpan_free; sdata->dev->priv_destructor = mac802154_wpan_free;
sdata->dev->netdev_ops = &mac802154_wpan_ops; sdata->dev->netdev_ops = &mac802154_wpan_ops;
sdata->dev->ml_priv = &mac802154_mlme_wpan; sdata->dev->ml_priv = &mac802154_mlme_wpan;
wpan_dev->promiscuous_mode = false; sdata->iface_default_filtering = IEEE802154_FILTERING_4_FRAME_FIELDS;
wpan_dev->header_ops = &ieee802154_header_ops; wpan_dev->header_ops = &ieee802154_header_ops;
mutex_init(&sdata->sec_mtx); mutex_init(&sdata->sec_mtx);
...@@ -600,7 +588,7 @@ ieee802154_setup_sdata(struct ieee802154_sub_if_data *sdata, ...@@ -600,7 +588,7 @@ ieee802154_setup_sdata(struct ieee802154_sub_if_data *sdata,
case NL802154_IFTYPE_MONITOR: case NL802154_IFTYPE_MONITOR:
sdata->dev->needs_free_netdev = true; sdata->dev->needs_free_netdev = true;
sdata->dev->netdev_ops = &mac802154_monitor_ops; sdata->dev->netdev_ops = &mac802154_monitor_ops;
wpan_dev->promiscuous_mode = true; sdata->iface_default_filtering = IEEE802154_FILTERING_NONE;
break; break;
default: default:
BUG(); BUG();
......
...@@ -95,7 +95,7 @@ ieee802154_alloc_hw(size_t priv_data_len, const struct ieee802154_ops *ops) ...@@ -95,7 +95,7 @@ ieee802154_alloc_hw(size_t priv_data_len, const struct ieee802154_ops *ops)
skb_queue_head_init(&local->skb_queue); skb_queue_head_init(&local->skb_queue);
INIT_WORK(&local->tx_work, ieee802154_xmit_worker); INIT_WORK(&local->sync_tx_work, ieee802154_xmit_sync_worker);
/* init supported flags with 802.15.4 default ranges */ /* init supported flags with 802.15.4 default ranges */
phy->supported.max_minbe = 8; phy->supported.max_minbe = 8;
......
...@@ -34,6 +34,7 @@ ieee802154_subif_frame(struct ieee802154_sub_if_data *sdata, ...@@ -34,6 +34,7 @@ ieee802154_subif_frame(struct ieee802154_sub_if_data *sdata,
struct sk_buff *skb, const struct ieee802154_hdr *hdr) struct sk_buff *skb, const struct ieee802154_hdr *hdr)
{ {
struct wpan_dev *wpan_dev = &sdata->wpan_dev; struct wpan_dev *wpan_dev = &sdata->wpan_dev;
struct wpan_phy *wpan_phy = sdata->local->hw.phy;
__le16 span, sshort; __le16 span, sshort;
int rc; int rc;
...@@ -42,6 +43,17 @@ ieee802154_subif_frame(struct ieee802154_sub_if_data *sdata, ...@@ -42,6 +43,17 @@ ieee802154_subif_frame(struct ieee802154_sub_if_data *sdata,
span = wpan_dev->pan_id; span = wpan_dev->pan_id;
sshort = wpan_dev->short_addr; sshort = wpan_dev->short_addr;
/* Level 3 filtering: Only beacons are accepted during scans */
if (sdata->required_filtering == IEEE802154_FILTERING_3_SCAN &&
sdata->required_filtering > wpan_phy->filtering) {
if (mac_cb(skb)->type != IEEE802154_FC_TYPE_BEACON) {
dev_dbg(&sdata->dev->dev,
"drop non-beacon frame (0x%x) during scan\n",
mac_cb(skb)->type);
goto fail;
}
}
switch (mac_cb(skb)->dest.mode) { switch (mac_cb(skb)->dest.mode) {
case IEEE802154_ADDR_NONE: case IEEE802154_ADDR_NONE:
if (hdr->source.mode != IEEE802154_ADDR_NONE) if (hdr->source.mode != IEEE802154_ADDR_NONE)
...@@ -114,8 +126,10 @@ ieee802154_subif_frame(struct ieee802154_sub_if_data *sdata, ...@@ -114,8 +126,10 @@ ieee802154_subif_frame(struct ieee802154_sub_if_data *sdata,
static void static void
ieee802154_print_addr(const char *name, const struct ieee802154_addr *addr) ieee802154_print_addr(const char *name, const struct ieee802154_addr *addr)
{ {
if (addr->mode == IEEE802154_ADDR_NONE) if (addr->mode == IEEE802154_ADDR_NONE) {
pr_debug("%s not present\n", name); pr_debug("%s not present\n", name);
return;
}
pr_debug("%s PAN ID: %04x\n", name, le16_to_cpu(addr->pan_id)); pr_debug("%s PAN ID: %04x\n", name, le16_to_cpu(addr->pan_id));
if (addr->mode == IEEE802154_ADDR_SHORT) { if (addr->mode == IEEE802154_ADDR_SHORT) {
...@@ -209,6 +223,13 @@ __ieee802154_rx_handle_packet(struct ieee802154_local *local, ...@@ -209,6 +223,13 @@ __ieee802154_rx_handle_packet(struct ieee802154_local *local,
if (!ieee802154_sdata_running(sdata)) if (!ieee802154_sdata_running(sdata))
continue; continue;
/* Do not deliver packets received on interfaces expecting
* AACK=1 if the address filters where disabled.
*/
if (local->hw.phy->filtering < IEEE802154_FILTERING_4_FRAME_FIELDS &&
sdata->required_filtering == IEEE802154_FILTERING_4_FRAME_FIELDS)
continue;
ieee802154_subif_frame(sdata, skb, &hdr); ieee802154_subif_frame(sdata, skb, &hdr);
skb = NULL; skb = NULL;
break; break;
...@@ -268,10 +289,8 @@ void ieee802154_rx(struct ieee802154_local *local, struct sk_buff *skb) ...@@ -268,10 +289,8 @@ void ieee802154_rx(struct ieee802154_local *local, struct sk_buff *skb)
ieee802154_monitors_rx(local, skb); ieee802154_monitors_rx(local, skb);
/* Check if transceiver doesn't validate the checksum. /* Level 1 filtering: Check the FCS by software when relevant */
* If not we validate the checksum here. if (local->hw.phy->filtering == IEEE802154_FILTERING_NONE) {
*/
if (local->hw.flags & IEEE802154_HW_RX_DROP_BAD_CKSUM) {
crc = crc_ccitt(0, skb->data, skb->len); crc = crc_ccitt(0, skb->data, skb->len);
if (crc) { if (crc) {
rcu_read_unlock(); rcu_read_unlock();
......
...@@ -22,10 +22,10 @@ ...@@ -22,10 +22,10 @@
#include "ieee802154_i.h" #include "ieee802154_i.h"
#include "driver-ops.h" #include "driver-ops.h"
void ieee802154_xmit_worker(struct work_struct *work) void ieee802154_xmit_sync_worker(struct work_struct *work)
{ {
struct ieee802154_local *local = struct ieee802154_local *local =
container_of(work, struct ieee802154_local, tx_work); container_of(work, struct ieee802154_local, sync_tx_work);
struct sk_buff *skb = local->tx_skb; struct sk_buff *skb = local->tx_skb;
struct net_device *dev = skb->dev; struct net_device *dev = skb->dev;
int res; int res;
...@@ -43,7 +43,9 @@ void ieee802154_xmit_worker(struct work_struct *work) ...@@ -43,7 +43,9 @@ void ieee802154_xmit_worker(struct work_struct *work)
err_tx: err_tx:
/* Restart the netif queue on each sub_if_data object. */ /* Restart the netif queue on each sub_if_data object. */
ieee802154_wake_queue(&local->hw); ieee802154_release_queue(local);
if (atomic_dec_and_test(&local->phy->ongoing_txs))
wake_up(&local->phy->sync_txq);
kfree_skb(skb); kfree_skb(skb);
netdev_dbg(dev, "transmission failed\n"); netdev_dbg(dev, "transmission failed\n");
} }
...@@ -65,7 +67,7 @@ ieee802154_tx(struct ieee802154_local *local, struct sk_buff *skb) ...@@ -65,7 +67,7 @@ ieee802154_tx(struct ieee802154_local *local, struct sk_buff *skb)
consume_skb(skb); consume_skb(skb);
skb = nskb; skb = nskb;
} else { } else {
goto err_tx; goto err_free_skb;
} }
} }
...@@ -74,32 +76,134 @@ ieee802154_tx(struct ieee802154_local *local, struct sk_buff *skb) ...@@ -74,32 +76,134 @@ ieee802154_tx(struct ieee802154_local *local, struct sk_buff *skb)
} }
/* Stop the netif queue on each sub_if_data object. */ /* Stop the netif queue on each sub_if_data object. */
ieee802154_stop_queue(&local->hw); ieee802154_hold_queue(local);
atomic_inc(&local->phy->ongoing_txs);
/* async is priority, otherwise sync is fallback */ /* Drivers should preferably implement the async callback. In some rare
* cases they only provide a sync callback which we will use as a
* fallback.
*/
if (local->ops->xmit_async) { if (local->ops->xmit_async) {
unsigned int len = skb->len; unsigned int len = skb->len;
ret = drv_xmit_async(local, skb); ret = drv_xmit_async(local, skb);
if (ret) { if (ret)
ieee802154_wake_queue(&local->hw); goto err_wake_netif_queue;
goto err_tx;
}
dev->stats.tx_packets++; dev->stats.tx_packets++;
dev->stats.tx_bytes += len; dev->stats.tx_bytes += len;
} else { } else {
local->tx_skb = skb; local->tx_skb = skb;
queue_work(local->workqueue, &local->tx_work); queue_work(local->workqueue, &local->sync_tx_work);
} }
return NETDEV_TX_OK; return NETDEV_TX_OK;
err_tx: err_wake_netif_queue:
ieee802154_release_queue(local);
if (atomic_dec_and_test(&local->phy->ongoing_txs))
wake_up(&local->phy->sync_txq);
err_free_skb:
kfree_skb(skb); kfree_skb(skb);
return NETDEV_TX_OK; return NETDEV_TX_OK;
} }
static int ieee802154_sync_queue(struct ieee802154_local *local)
{
int ret;
ieee802154_hold_queue(local);
ieee802154_disable_queue(local);
wait_event(local->phy->sync_txq, !atomic_read(&local->phy->ongoing_txs));
ret = local->tx_result;
ieee802154_release_queue(local);
return ret;
}
int ieee802154_sync_and_hold_queue(struct ieee802154_local *local)
{
int ret;
ieee802154_hold_queue(local);
ret = ieee802154_sync_queue(local);
set_bit(WPAN_PHY_FLAG_STATE_QUEUE_STOPPED, &local->phy->flags);
return ret;
}
int ieee802154_mlme_op_pre(struct ieee802154_local *local)
{
return ieee802154_sync_and_hold_queue(local);
}
int ieee802154_mlme_tx(struct ieee802154_local *local,
struct ieee802154_sub_if_data *sdata,
struct sk_buff *skb)
{
int ret;
/* Avoid possible calls to ->ndo_stop() when we asynchronously perform
* MLME transmissions.
*/
rtnl_lock();
/* Ensure the device was not stopped, otherwise error out */
if (!local->open_count) {
rtnl_unlock();
return -ENETDOWN;
}
/* Warn if the ieee802154 core thinks MLME frames can be sent while the
* net interface expects this cannot happen.
*/
if (WARN_ON_ONCE(!netif_running(sdata->dev))) {
rtnl_unlock();
return -ENETDOWN;
}
ieee802154_tx(local, skb);
ret = ieee802154_sync_queue(local);
rtnl_unlock();
return ret;
}
void ieee802154_mlme_op_post(struct ieee802154_local *local)
{
ieee802154_release_queue(local);
}
int ieee802154_mlme_tx_one(struct ieee802154_local *local,
struct ieee802154_sub_if_data *sdata,
struct sk_buff *skb)
{
int ret;
ieee802154_mlme_op_pre(local);
ret = ieee802154_mlme_tx(local, sdata, skb);
ieee802154_mlme_op_post(local);
return ret;
}
static bool ieee802154_queue_is_stopped(struct ieee802154_local *local)
{
return test_bit(WPAN_PHY_FLAG_STATE_QUEUE_STOPPED, &local->phy->flags);
}
static netdev_tx_t
ieee802154_hot_tx(struct ieee802154_local *local, struct sk_buff *skb)
{
/* Warn if the net interface tries to transmit frames while the
* ieee802154 core assumes the queue is stopped.
*/
WARN_ON_ONCE(ieee802154_queue_is_stopped(local));
return ieee802154_tx(local, skb);
}
netdev_tx_t netdev_tx_t
ieee802154_monitor_start_xmit(struct sk_buff *skb, struct net_device *dev) ieee802154_monitor_start_xmit(struct sk_buff *skb, struct net_device *dev)
{ {
...@@ -107,7 +211,7 @@ ieee802154_monitor_start_xmit(struct sk_buff *skb, struct net_device *dev) ...@@ -107,7 +211,7 @@ ieee802154_monitor_start_xmit(struct sk_buff *skb, struct net_device *dev)
skb->skb_iif = dev->ifindex; skb->skb_iif = dev->ifindex;
return ieee802154_tx(sdata->local, skb); return ieee802154_hot_tx(sdata->local, skb);
} }
netdev_tx_t netdev_tx_t
...@@ -129,5 +233,5 @@ ieee802154_subif_start_xmit(struct sk_buff *skb, struct net_device *dev) ...@@ -129,5 +233,5 @@ ieee802154_subif_start_xmit(struct sk_buff *skb, struct net_device *dev)
skb->skb_iif = dev->ifindex; skb->skb_iif = dev->ifindex;
return ieee802154_tx(sdata->local, skb); return ieee802154_hot_tx(sdata->local, skb);
} }
...@@ -13,12 +13,23 @@ ...@@ -13,12 +13,23 @@
/* privid for wpan_phys to determine whether they belong to us or not */ /* privid for wpan_phys to determine whether they belong to us or not */
const void *const mac802154_wpan_phy_privid = &mac802154_wpan_phy_privid; const void *const mac802154_wpan_phy_privid = &mac802154_wpan_phy_privid;
void ieee802154_wake_queue(struct ieee802154_hw *hw) /**
* ieee802154_wake_queue - wake ieee802154 queue
* @local: main mac object
*
* Tranceivers usually have either one transmit framebuffer or one framebuffer
* for both transmitting and receiving. Hence, the core currently only handles
* one frame at a time for each phy, which means we had to stop the queue to
* avoid new skb to come during the transmission. The queue then needs to be
* woken up after the operation.
*/
static void ieee802154_wake_queue(struct ieee802154_hw *hw)
{ {
struct ieee802154_local *local = hw_to_local(hw); struct ieee802154_local *local = hw_to_local(hw);
struct ieee802154_sub_if_data *sdata; struct ieee802154_sub_if_data *sdata;
rcu_read_lock(); rcu_read_lock();
clear_bit(WPAN_PHY_FLAG_STATE_QUEUE_STOPPED, &local->phy->flags);
list_for_each_entry_rcu(sdata, &local->interfaces, list) { list_for_each_entry_rcu(sdata, &local->interfaces, list) {
if (!sdata->dev) if (!sdata->dev)
continue; continue;
...@@ -27,9 +38,18 @@ void ieee802154_wake_queue(struct ieee802154_hw *hw) ...@@ -27,9 +38,18 @@ void ieee802154_wake_queue(struct ieee802154_hw *hw)
} }
rcu_read_unlock(); rcu_read_unlock();
} }
EXPORT_SYMBOL(ieee802154_wake_queue);
void ieee802154_stop_queue(struct ieee802154_hw *hw) /**
* ieee802154_stop_queue - stop ieee802154 queue
* @local: main mac object
*
* Tranceivers usually have either one transmit framebuffer or one framebuffer
* for both transmitting and receiving. Hence, the core currently only handles
* one frame at a time for each phy, which means we need to tell upper layers to
* stop giving us new skbs while we are busy with the transmitted one. The queue
* must then be stopped before transmitting.
*/
static void ieee802154_stop_queue(struct ieee802154_hw *hw)
{ {
struct ieee802154_local *local = hw_to_local(hw); struct ieee802154_local *local = hw_to_local(hw);
struct ieee802154_sub_if_data *sdata; struct ieee802154_sub_if_data *sdata;
...@@ -43,14 +63,47 @@ void ieee802154_stop_queue(struct ieee802154_hw *hw) ...@@ -43,14 +63,47 @@ void ieee802154_stop_queue(struct ieee802154_hw *hw)
} }
rcu_read_unlock(); rcu_read_unlock();
} }
EXPORT_SYMBOL(ieee802154_stop_queue);
void ieee802154_hold_queue(struct ieee802154_local *local)
{
unsigned long flags;
spin_lock_irqsave(&local->phy->queue_lock, flags);
if (!atomic_fetch_inc(&local->phy->hold_txs))
ieee802154_stop_queue(&local->hw);
spin_unlock_irqrestore(&local->phy->queue_lock, flags);
}
void ieee802154_release_queue(struct ieee802154_local *local)
{
unsigned long flags;
spin_lock_irqsave(&local->phy->queue_lock, flags);
if (atomic_dec_and_test(&local->phy->hold_txs))
ieee802154_wake_queue(&local->hw);
spin_unlock_irqrestore(&local->phy->queue_lock, flags);
}
void ieee802154_disable_queue(struct ieee802154_local *local)
{
struct ieee802154_sub_if_data *sdata;
rcu_read_lock();
list_for_each_entry_rcu(sdata, &local->interfaces, list) {
if (!sdata->dev)
continue;
netif_tx_disable(sdata->dev);
}
rcu_read_unlock();
}
enum hrtimer_restart ieee802154_xmit_ifs_timer(struct hrtimer *timer) enum hrtimer_restart ieee802154_xmit_ifs_timer(struct hrtimer *timer)
{ {
struct ieee802154_local *local = struct ieee802154_local *local =
container_of(timer, struct ieee802154_local, ifs_timer); container_of(timer, struct ieee802154_local, ifs_timer);
ieee802154_wake_queue(&local->hw); ieee802154_release_queue(local);
return HRTIMER_NORESTART; return HRTIMER_NORESTART;
} }
...@@ -84,10 +137,12 @@ void ieee802154_xmit_complete(struct ieee802154_hw *hw, struct sk_buff *skb, ...@@ -84,10 +137,12 @@ void ieee802154_xmit_complete(struct ieee802154_hw *hw, struct sk_buff *skb,
hw->phy->sifs_period * NSEC_PER_USEC, hw->phy->sifs_period * NSEC_PER_USEC,
HRTIMER_MODE_REL); HRTIMER_MODE_REL);
} else { } else {
ieee802154_wake_queue(hw); ieee802154_release_queue(local);
} }
dev_consume_skb_any(skb); dev_consume_skb_any(skb);
if (atomic_dec_and_test(&hw->phy->ongoing_txs))
wake_up(&hw->phy->sync_txq);
} }
EXPORT_SYMBOL(ieee802154_xmit_complete); EXPORT_SYMBOL(ieee802154_xmit_complete);
...@@ -97,8 +152,10 @@ void ieee802154_xmit_error(struct ieee802154_hw *hw, struct sk_buff *skb, ...@@ -97,8 +152,10 @@ void ieee802154_xmit_error(struct ieee802154_hw *hw, struct sk_buff *skb,
struct ieee802154_local *local = hw_to_local(hw); struct ieee802154_local *local = hw_to_local(hw);
local->tx_result = reason; local->tx_result = reason;
ieee802154_wake_queue(hw); ieee802154_release_queue(local);
dev_kfree_skb_any(skb); dev_kfree_skb_any(skb);
if (atomic_dec_and_test(&hw->phy->ongoing_txs))
wake_up(&hw->phy->sync_txq);
} }
EXPORT_SYMBOL(ieee802154_xmit_error); EXPORT_SYMBOL(ieee802154_xmit_error);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment