Compare commits
44 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 64fc309e26 | |||
| cdb6bd07d3 | |||
| fc327b2ff6 | |||
| d95453c98e | |||
| 8cd10f487c | |||
| 093a5038b8 | |||
| 87a3d65960 | |||
| f68fd5530f | |||
| 0750df2611 | |||
| 0ec58c0ad5 | |||
| 42fd0ceab6 | |||
| 4be43770fd | |||
| 3dbabf3092 | |||
| 44b296647b | |||
| 25c0ed8c57 | |||
| 8dd79199f8 | |||
| 979d5436ee | |||
| 1e9eb4581f | |||
| 6a6aa243a4 | |||
| b717251598 | |||
| 65a4c39914 | |||
| 4bc0a34c94 | |||
| 90f50b375f | |||
| f2cf586f89 | |||
| d0f14e3ba7 | |||
| 80e5e68c22 | |||
| 983bd62dd0 | |||
| b7c4b0fc39 | |||
| 6168e9d340 | |||
| 0dde479994 | |||
| 8fc1bacded | |||
| aff632ea64 | |||
| f31c57adf7 | |||
| 8855718511 | |||
| ebb5c57988 | |||
| ef24cdb891 | |||
| 64eae76f4e | |||
| 315986ea27 | |||
| 9012b74eea | |||
| 8539460bf1 | |||
| 19feb8181a | |||
| 20d349e2b5 | |||
| 98c6e363f0 | |||
| b76c9904f8 |
+3
-4
@@ -14,7 +14,6 @@
|
||||
#include <linux/umh.h>
|
||||
#include "epta_request.h"
|
||||
#include "epta_coex.h"
|
||||
#include "txrx_opt.h"
|
||||
|
||||
#ifdef AP_HT_CAP_UPDATE
|
||||
#define HT_INFO_OFFSET 4
|
||||
@@ -1163,7 +1162,7 @@ void bes2600_multicast_stop_work(struct work_struct *work)
|
||||
container_of(work, struct bes2600_vif, multicast_stop_work);
|
||||
|
||||
if (priv->aid0_bit_set) {
|
||||
timer_delete_sync(&priv->mcast_timeout);
|
||||
del_timer_sync(&priv->mcast_timeout);
|
||||
wsm_lock_tx(priv->hw_priv);
|
||||
priv->aid0_bit_set = false;
|
||||
bes2600_set_tim_impl(priv, false);
|
||||
@@ -1173,7 +1172,7 @@ void bes2600_multicast_stop_work(struct work_struct *work)
|
||||
|
||||
void bes2600_mcast_timeout(struct timer_list *t)
|
||||
{
|
||||
struct bes2600_vif *priv = timer_container_of(priv, t, mcast_timeout);
|
||||
struct bes2600_vif *priv = from_timer(priv, t, mcast_timeout);
|
||||
|
||||
wiphy_warn(priv->hw->wiphy,
|
||||
"Multicast delivery timeout.\n");
|
||||
@@ -1247,7 +1246,7 @@ void bes2600_suspend_resume(struct bes2600_vif *priv,
|
||||
}
|
||||
spin_unlock_bh(&priv->ps_state_lock);
|
||||
if (cancel_tmo)
|
||||
timer_delete_sync(&priv->mcast_timeout);
|
||||
del_timer_sync(&priv->mcast_timeout);
|
||||
} else {
|
||||
spin_lock_bh(&priv->ps_state_lock);
|
||||
bes2600_ps_notify(priv, arg->link_id, arg->stop);
|
||||
|
||||
@@ -1,509 +0,0 @@
|
||||
// SPDX-License-Identifier: GPL-2.0-only
|
||||
/*
|
||||
* Bestechnic BES2600 BT UART driver
|
||||
*
|
||||
* Copyright (c) 2025 Dang Huynh <dang.huynh@mainlining.org>
|
||||
*/
|
||||
|
||||
#include <linux/module.h>
|
||||
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/types.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/skbuff.h>
|
||||
|
||||
#include <linux/of.h>
|
||||
#include <linux/of_irq.h>
|
||||
|
||||
#include <linux/device.h>
|
||||
#include <linux/pm.h>
|
||||
#include <linux/pm_wakeirq.h>
|
||||
|
||||
#include <linux/gpio/consumer.h>
|
||||
#include <linux/pinctrl/consumer.h>
|
||||
#include <linux/serdev.h>
|
||||
|
||||
#include <net/bluetooth/bluetooth.h>
|
||||
#include <net/bluetooth/hci_core.h>
|
||||
|
||||
#include "bes2600.h"
|
||||
#include "bes_chardev.h"
|
||||
|
||||
#include "h4_recv.h"
|
||||
|
||||
struct bes2600_btuart_data {
|
||||
struct serdev_device *serdev;
|
||||
struct hci_dev *hdev;
|
||||
|
||||
struct pinctrl *pinctrl;
|
||||
struct pinctrl_state *pinctrl_default;
|
||||
|
||||
/* Device tree properties */
|
||||
struct gpio_desc *power;
|
||||
struct gpio_desc *wakeup;
|
||||
int speed;
|
||||
|
||||
unsigned long state;
|
||||
|
||||
int wake_irq;
|
||||
bool opened;
|
||||
|
||||
int err_count; /* workaround for broken firmware */
|
||||
|
||||
struct sk_buff_head txq;
|
||||
struct work_struct tx_work;
|
||||
rwlock_t lock;
|
||||
|
||||
struct sk_buff *rx_skb;
|
||||
};
|
||||
|
||||
static const struct h4_recv_pkt bes2600_btuart_recv_pkts[] = {
|
||||
{ H4_RECV_ACL, .recv = hci_recv_frame },
|
||||
{ H4_RECV_SCO, .recv = hci_recv_frame },
|
||||
{ H4_RECV_EVENT, .recv = hci_recv_frame },
|
||||
};
|
||||
|
||||
static int bes2600_btuart_power(struct bes2600_btuart_data *data, bool enable)
|
||||
{
|
||||
int ret, val;
|
||||
|
||||
if (enable)
|
||||
val = 1;
|
||||
else
|
||||
val = 0;
|
||||
|
||||
ret = gpiod_set_value_cansleep(data->power, val);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = bes2600_chrdev_switch_subsys_glb(-1, val);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* HACK: Potential DOS issue, not sure if the firmware can filter it but
|
||||
* there needs to be a better way to detect this. Even better, fix the firmware.
|
||||
*/
|
||||
static int bes2600_btuart_bug_recovery(struct bes2600_btuart_data *data)
|
||||
{
|
||||
struct hci_dev *hdev = data->hdev;
|
||||
struct serdev_device *serdev = data->serdev;
|
||||
|
||||
bt_dev_info(hdev, "Attempting to recover from Firmware bug (might not work)");
|
||||
|
||||
serdev_device_set_flow_control(serdev, false);
|
||||
bes2600_btuart_power(data, false);
|
||||
msleep(20);
|
||||
bes2600_btuart_power(data, true);
|
||||
serdev_device_set_flow_control(serdev, true);
|
||||
msleep(100);
|
||||
|
||||
hci_reset_dev(hdev);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_recv(struct bes2600_btuart_data *data, const void *buffer, int count)
|
||||
{
|
||||
struct hci_dev *hdev = data->hdev;
|
||||
int err = 0;
|
||||
|
||||
data->rx_skb = h4_recv_buf(hdev, data->rx_skb, buffer, count,
|
||||
bes2600_btuart_recv_pkts, ARRAY_SIZE(bes2600_btuart_recv_pkts));
|
||||
if (IS_ERR(data->rx_skb))
|
||||
err = PTR_ERR(data->rx_skb);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static size_t bes2600_btuart_receive_buf(struct serdev_device *serdev,
|
||||
const u8 *buf, size_t count)
|
||||
{
|
||||
struct bes2600_btuart_data *data = serdev_device_get_drvdata(serdev);
|
||||
struct hci_dev *hdev = data->hdev;
|
||||
int ret;
|
||||
|
||||
read_lock(&data->lock);
|
||||
ret = bes2600_btuart_recv(data, buf, count);
|
||||
if (ret) {
|
||||
bt_dev_err(hdev, "Failed to receive packet (%d)", ret);
|
||||
data->err_count++;
|
||||
ret = 0;
|
||||
} else {
|
||||
hdev->stat.byte_rx += count;
|
||||
ret = count;
|
||||
}
|
||||
read_unlock(&data->lock);
|
||||
|
||||
/* Attempting to recovery from a bugged firmware */
|
||||
if (data->err_count > 10) {
|
||||
bes2600_btuart_bug_recovery(data);
|
||||
data->err_count = 0;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static const struct serdev_device_ops bes2600_btuart_serdev_client_ops = {
|
||||
.receive_buf = bes2600_btuart_receive_buf,
|
||||
.write_wakeup = serdev_device_write_wakeup,
|
||||
};
|
||||
|
||||
static void bes2600_btuart_wake(struct hci_dev *hdev, bool enable)
|
||||
{
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
|
||||
if (enable)
|
||||
gpiod_set_value_cansleep(data->wakeup, 1);
|
||||
else
|
||||
gpiod_set_value_cansleep(data->wakeup, 0);
|
||||
}
|
||||
|
||||
static int bes2600_btuart_open(struct hci_dev *hdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
struct serdev_device *serdev = data->serdev;
|
||||
int ret;
|
||||
|
||||
ret = serdev_device_open(serdev);
|
||||
if (ret < 0) {
|
||||
bt_dev_err(hdev, "Failed to open serial device");
|
||||
return ret;
|
||||
}
|
||||
|
||||
serdev_device_set_flow_control(serdev, false);
|
||||
|
||||
ret = serdev_device_set_baudrate(serdev, data->speed);
|
||||
if (ret < 0) {
|
||||
bt_dev_err(hdev, "Failed to set baud rate");
|
||||
return ret;
|
||||
}
|
||||
|
||||
serdev_device_set_flow_control(serdev, true);
|
||||
|
||||
data->opened = true;
|
||||
|
||||
bes2600_btuart_wake(hdev, true);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_flush(struct hci_dev *hdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
struct serdev_device *serdev = data->serdev;
|
||||
|
||||
serdev_device_write_flush(serdev);
|
||||
|
||||
read_lock(&data->lock);
|
||||
skb_queue_purge(&data->txq);
|
||||
cancel_work_sync(&data->tx_work);
|
||||
read_unlock(&data->lock);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_close(struct hci_dev *hdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
struct serdev_device *serdev = data->serdev;
|
||||
|
||||
bes2600_btuart_wake(hdev, false);
|
||||
|
||||
serdev_device_set_flow_control(serdev, false);
|
||||
serdev_device_close(serdev);
|
||||
|
||||
data->opened = false;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_setup(struct hci_dev *hdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
int ret;
|
||||
|
||||
if (bes2600_chrdev_is_bus_error())
|
||||
return -EFAULT;
|
||||
|
||||
/* Set pinctrl state to default */
|
||||
ret = pinctrl_select_state(data->pinctrl, data->pinctrl_default);
|
||||
if (ret < 0) {
|
||||
bt_dev_err(hdev, "Failed to setup pinctrl state");
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Set power GPIO to high and tell chardev to switch to BT */
|
||||
bes2600_btuart_power(data, true);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_shutdown(struct hci_dev *hdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
|
||||
if (bes2600_chrdev_is_bus_error())
|
||||
return -EFAULT;
|
||||
|
||||
bes2600_btuart_power(data, false);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_send_frame(struct hci_dev *hdev, struct sk_buff *skb)
|
||||
{
|
||||
char *pkt_type = NULL;
|
||||
struct bes2600_btuart_data *data = hci_get_drvdata(hdev);
|
||||
|
||||
switch (hci_skb_pkt_type(skb)) {
|
||||
case HCI_COMMAND_PKT:
|
||||
hdev->stat.cmd_tx++;
|
||||
break;
|
||||
case HCI_ACLDATA_PKT:
|
||||
hdev->stat.acl_tx++;
|
||||
break;
|
||||
case HCI_SCODATA_PKT:
|
||||
hdev->stat.sco_tx++;
|
||||
break;
|
||||
default:
|
||||
return -EILSEQ;
|
||||
}
|
||||
pkt_type = skb_push(skb, 1);
|
||||
pkt_type[0] = hci_skb_pkt_type(skb);
|
||||
read_lock(&data->lock);
|
||||
skb_queue_tail(&data->txq, skb);
|
||||
read_unlock(&data->lock);
|
||||
|
||||
schedule_work(&data->tx_work);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static ssize_t bes2600_btuart_send(struct bes2600_btuart_data *data, struct sk_buff *skb)
|
||||
{
|
||||
struct hci_dev *hdev = data->hdev;
|
||||
int len;
|
||||
|
||||
len = serdev_device_write_buf(data->serdev, skb->data, skb->len);
|
||||
if (len < 0) {
|
||||
bt_dev_err(data->hdev, "Failed to write buffer to TTY");
|
||||
return -EIO;
|
||||
}
|
||||
serdev_device_wait_until_sent(data->serdev, 0);
|
||||
|
||||
hdev->stat.byte_tx += len;
|
||||
|
||||
skb_pull(skb, len);
|
||||
|
||||
return skb->len;
|
||||
}
|
||||
|
||||
static void bes2600_btuart_work(struct work_struct *work)
|
||||
{
|
||||
struct bes2600_btuart_data *data = container_of(work, struct bes2600_btuart_data, tx_work);
|
||||
struct sk_buff *skb;
|
||||
|
||||
while ((skb = skb_dequeue(&data->txq))) {
|
||||
if (bes2600_btuart_send(data, skb) > 0) {
|
||||
skb_queue_head(&data->txq, skb);
|
||||
break;
|
||||
}
|
||||
|
||||
kfree_skb(skb);
|
||||
}
|
||||
}
|
||||
|
||||
static void bes2600_btuart_parse_dt(struct serdev_device *serdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = serdev_device_get_drvdata(serdev);
|
||||
struct device_node *node = serdev->dev.of_node;
|
||||
int speed = 1500000;
|
||||
|
||||
of_property_read_u32(node, "current-speed", &speed);
|
||||
data->speed = speed;
|
||||
}
|
||||
|
||||
static irqreturn_t bes2600_btuart_wakeup_handler(int irq, void *priv)
|
||||
{
|
||||
struct bes2600_btuart_data *data = (struct bes2600_btuart_data *)priv;
|
||||
|
||||
dev_info(&data->serdev->dev, "IRQ wake\n");
|
||||
|
||||
return IRQ_HANDLED;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_PM_SLEEP
|
||||
static int bes2600_btuart_suspend(struct device *dev)
|
||||
{
|
||||
struct serdev_device *serdev = to_serdev_device(dev);
|
||||
struct bes2600_btuart_data *data = serdev_device_get_drvdata(serdev);
|
||||
|
||||
if (data->opened) {
|
||||
bes2600_btuart_wake(data->hdev, false);
|
||||
|
||||
if (device_may_wakeup(dev) && data->wake_irq)
|
||||
enable_irq_wake(data->wake_irq);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int bes2600_btuart_resume(struct device *dev)
|
||||
{
|
||||
struct serdev_device *serdev = to_serdev_device(dev);
|
||||
struct bes2600_btuart_data *data = serdev_device_get_drvdata(serdev);
|
||||
|
||||
if (data->opened) {
|
||||
bes2600_btuart_wake(data->hdev, true);
|
||||
|
||||
if (device_may_wakeup(dev) && data->wake_irq)
|
||||
disable_irq_wake(data->wake_irq);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
static SIMPLE_DEV_PM_OPS(bes2600_btuart_pm_ops,
|
||||
bes2600_btuart_suspend, bes2600_btuart_resume);
|
||||
|
||||
static int bes2600_btuart_probe(struct serdev_device *serdev)
|
||||
{
|
||||
struct device *dev = &serdev->dev;
|
||||
struct bes2600_btuart_data *data;
|
||||
struct hci_dev *hdev;
|
||||
int ret;
|
||||
|
||||
data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
|
||||
if (!data)
|
||||
return -ENOMEM;
|
||||
|
||||
skb_queue_head_init(&data->txq);
|
||||
|
||||
data->pinctrl = devm_pinctrl_get(dev);
|
||||
if (IS_ERR(data->pinctrl))
|
||||
return dev_err_probe(dev, PTR_ERR(data->pinctrl), "Cannot get pinctrl\n");
|
||||
|
||||
data->pinctrl_default = pinctrl_lookup_state(data->pinctrl, PINCTRL_STATE_DEFAULT);
|
||||
if (IS_ERR(data->pinctrl_default))
|
||||
return dev_err_probe(dev, PTR_ERR(data->pinctrl_default),
|
||||
"Cannot get default pinctrl state\n");
|
||||
|
||||
data->power = devm_gpiod_get_index(dev, "power", 0, GPIOD_OUT_HIGH);
|
||||
if (IS_ERR(data->power))
|
||||
return dev_err_probe(dev, PTR_ERR(data->power), "Failed to get power gpio\n");
|
||||
|
||||
data->wakeup = devm_gpiod_get_index_optional(dev, "wakeup", 0, GPIOD_OUT_HIGH);
|
||||
if (!data->wakeup)
|
||||
dev_err(dev, "Failed to get wakeup gpio\n");
|
||||
|
||||
data->wake_irq = of_irq_get_byname(dev->of_node, "wakeup");
|
||||
if (data->wake_irq <= 0) {
|
||||
ret = data->wake_irq;
|
||||
dev_err(dev, "Failed to get IRQ number, host will not wake up (ret=%d)\n", ret);
|
||||
}
|
||||
|
||||
if (data->wake_irq > 0) {
|
||||
ret = devm_device_init_wakeup(dev);
|
||||
if (ret)
|
||||
return dev_err_probe(dev, ret, "Failed to initialize device wakeup\n");
|
||||
|
||||
ret = devm_request_threaded_irq(dev, data->wake_irq, NULL,
|
||||
bes2600_btuart_wakeup_handler,
|
||||
IRQF_ONESHOT | IRQF_TRIGGER_HIGH,
|
||||
"bes2600_btwakeup_irq", data);
|
||||
if (ret < 0)
|
||||
return dev_err_probe(dev, ret, "Cannot request IRQ\n");
|
||||
|
||||
ret = devm_pm_set_wake_irq(dev, data->wake_irq);
|
||||
if (ret < 0)
|
||||
return dev_err_probe(dev, ret, "Failed to set wake irq\n");
|
||||
}
|
||||
|
||||
data->err_count = 0;
|
||||
|
||||
hdev = hci_alloc_dev();
|
||||
if (!hdev)
|
||||
return -ENOMEM;
|
||||
|
||||
rwlock_init(&data->lock);
|
||||
|
||||
data->serdev = serdev;
|
||||
data->hdev = hdev;
|
||||
|
||||
serdev_device_set_drvdata(serdev, data);
|
||||
serdev_device_set_client_ops(serdev, &bes2600_btuart_serdev_client_ops);
|
||||
|
||||
bes2600_btuart_parse_dt(serdev);
|
||||
|
||||
INIT_WORK(&data->tx_work, bes2600_btuart_work);
|
||||
|
||||
hdev->bus = HCI_UART;
|
||||
hci_set_drvdata(hdev, data);
|
||||
|
||||
hdev->open = bes2600_btuart_open;
|
||||
hdev->close = bes2600_btuart_close;
|
||||
hdev->flush = bes2600_btuart_flush;
|
||||
hdev->setup = bes2600_btuart_setup;
|
||||
hdev->shutdown = bes2600_btuart_shutdown;
|
||||
hdev->send = bes2600_btuart_send_frame;
|
||||
hdev->manufacturer = 0x02B0;
|
||||
|
||||
SET_HCIDEV_DEV(hdev, &serdev->dev);
|
||||
|
||||
hci_set_quirk(hdev, HCI_QUIRK_BROKEN_STORED_LINK_KEY);
|
||||
|
||||
/* Firmware claims to support HCI_OP_LE_READ_BUFFER_SIZE_V2 but broken */
|
||||
hci_set_quirk(hdev, HCI_QUIRK_BROKEN_BUFFER_SIZE_V2);
|
||||
|
||||
/*
|
||||
* Once shutdown() is ran, it turns off the Bluetooth regulator and
|
||||
* would not power back on unless setup() is run again.
|
||||
*/
|
||||
hci_set_quirk(hdev, HCI_QUIRK_NON_PERSISTENT_SETUP);
|
||||
|
||||
ret = hci_register_dev(hdev);
|
||||
if (ret < 0) {
|
||||
hci_free_dev(hdev);
|
||||
return dev_err_probe(dev, ret, "Failed to register HCI device\n");
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bes2600_btuart_remove(struct serdev_device *serdev)
|
||||
{
|
||||
struct bes2600_btuart_data *data = serdev_device_get_drvdata(serdev);
|
||||
struct hci_dev *hdev = data->hdev;
|
||||
|
||||
cancel_work_sync(&data->tx_work);
|
||||
if (hdev) {
|
||||
hci_unregister_dev(hdev);
|
||||
hci_free_dev(hdev);
|
||||
}
|
||||
}
|
||||
|
||||
static const struct of_device_id bes2600_btuart_of_match[] = {
|
||||
{ .compatible = "bestechnic,bes2600-btuart", },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, bes2600_btuart_of_match);
|
||||
|
||||
static struct serdev_device_driver bes2600_btuart_driver = {
|
||||
.probe = bes2600_btuart_probe,
|
||||
.remove = bes2600_btuart_remove,
|
||||
.driver = {
|
||||
.name = "bes2600_btuart",
|
||||
.pm = &bes2600_btuart_pm_ops,
|
||||
.of_match_table = of_match_ptr(bes2600_btuart_of_match),
|
||||
},
|
||||
};
|
||||
module_serdev_device_driver(bes2600_btuart_driver);
|
||||
|
||||
MODULE_AUTHOR("Dang Huynh <dang.huynh@mainlining.org>");
|
||||
MODULE_DESCRIPTION("BES2600 BT UART driver");
|
||||
MODULE_LICENSE("GPL");
|
||||
@@ -1997,6 +1997,7 @@ static int bes2600_sdio_probe(struct sdio_func *func,
|
||||
|
||||
out:
|
||||
bes2600_chrdev_set_sbus_priv_data(self, false);
|
||||
bes2600_switch_bt(true);
|
||||
bes2600_gpio_allow_mcu_sleep(self, GPIO_WAKE_FLAG_SDIO_PROBE);
|
||||
return 0;
|
||||
|
||||
@@ -2031,8 +2032,8 @@ int bes2600_unregister_net_dev(struct sbus_priv *bus_priv)
|
||||
BUG_ON(!bus_priv);
|
||||
if (bus_priv->core && !bus_priv->unregister_in_process) {
|
||||
bus_priv->unregister_in_process = true;
|
||||
bes2600_pwr_unregister_en_lp_cb(bus_priv->core, bes2600_sdio_en_lp_cb);
|
||||
bes2600_core_release(bus_priv->core);
|
||||
bes2600_pwr_unregister_en_lp_cb(bus_priv->core, bes2600_sdio_en_lp_cb);
|
||||
bus_priv->core = NULL;
|
||||
|
||||
if (bus_priv->sdio_wq) {
|
||||
|
||||
+1
-32
@@ -181,7 +181,7 @@ static int bes2600_switch_wifi(bool on)
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_switch_bt(bool on)
|
||||
int bes2600_switch_bt(bool on)
|
||||
{
|
||||
int ret = 0;
|
||||
long status = 0;
|
||||
@@ -234,36 +234,6 @@ static int bes2600_switch_bt(bool on)
|
||||
return ret;
|
||||
}
|
||||
|
||||
/*
|
||||
* Re-added for danctnix's bes2600_btuart.c (a danctnix-only file) which
|
||||
* relies on the chardev utility API for BT power switching and bus-error
|
||||
* checks. The userspace /dev/bes2600 chardev itself is removed by the
|
||||
* remove-chardev-user-interface series; these in-kernel helpers stay.
|
||||
*
|
||||
* @wifi: 1 to turn on, 0 to turn off. Otherwise, leave unchanged
|
||||
* @bt: 1 to turn on, 0 to turn off. Otherwise, leave unchanged
|
||||
*/
|
||||
int bes2600_chrdev_switch_subsys_glb(int wifi, int bt)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
switch (wifi) {
|
||||
case 0: ret = bes2600_switch_wifi(false); break;
|
||||
case 1: ret = bes2600_switch_wifi(true); break;
|
||||
default: break;
|
||||
}
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
switch (bt) {
|
||||
case 0: ret = bes2600_switch_bt(false); break;
|
||||
case 1: ret = bes2600_switch_bt(true); break;
|
||||
default: break;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bes2600_chrdev_switch_subsys_glb);
|
||||
|
||||
|
||||
|
||||
|
||||
@@ -592,7 +562,6 @@ bool bes2600_chrdev_is_bus_error(void)
|
||||
|
||||
return error;
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(bes2600_chrdev_is_bus_error);
|
||||
|
||||
void bes2600_chrdev_update_signal_mode(void)
|
||||
{
|
||||
|
||||
@@ -62,7 +62,6 @@ int bes2600_chrdev_trigger_bus_reset(void);
|
||||
void bes2600_chrdev_wakeup_bt(void);
|
||||
void bes2600_chrdev_wifi_force_close(struct bes2600_common *hw_priv, bool halt_dev);
|
||||
void bes2600_chrdev_usb_remove(struct bes2600_common *hw_priv);
|
||||
int bes2600_chrdev_switch_subsys_glb(int wifi, int bt);
|
||||
|
||||
/* get and set internal state */
|
||||
bool bes2600_chrdev_is_wifi_opened(void);
|
||||
@@ -91,4 +90,6 @@ u8* bes2600_alloc_dpd_log_buffer(u16 len);
|
||||
void bes2600_get_dpd_log(char **data, size_t *len);
|
||||
#endif
|
||||
|
||||
int bes2600_switch_bt(bool);
|
||||
|
||||
#endif /* __BES_CHARDEV_H__ */
|
||||
|
||||
+23
-568
@@ -22,7 +22,6 @@
|
||||
#include "debug.h"
|
||||
#include "epta_coex.h"
|
||||
#include "bes_chardev.h"
|
||||
#include "txrx_opt.h"
|
||||
#include "sta.h"
|
||||
#include "bes_log.h"
|
||||
|
||||
@@ -317,83 +316,6 @@ int wsm_release_buffer_to_fw(struct bes2600_vif *priv, int count)
|
||||
}
|
||||
#endif
|
||||
|
||||
#if 0
|
||||
static struct sk_buff *bes2600_get_skb(struct bes2600_common *hw_priv, size_t len)
|
||||
{
|
||||
struct sk_buff *skb;
|
||||
size_t alloc_len = (len > SDIO_BLOCK_SIZE) ? len : SDIO_BLOCK_SIZE;
|
||||
|
||||
if (len > SDIO_BLOCK_SIZE || !hw_priv->skb_cache) {
|
||||
skb = dev_alloc_skb(alloc_len
|
||||
+ WSM_TX_EXTRA_HEADROOM
|
||||
+ 8 /* TKIP IV */
|
||||
+ 12 /* TKIP ICV + MIC */
|
||||
- 2 /* Piggyback */);
|
||||
/* In AP mode RXed SKB can be looped back as a broadcast.
|
||||
* Here we reserve enough space for headers. */
|
||||
skb_reserve(skb, WSM_TX_EXTRA_HEADROOM
|
||||
+ 8 /* TKIP IV */
|
||||
- WSM_RX_EXTRA_HEADROOM);
|
||||
} else {
|
||||
skb = hw_priv->skb_cache;
|
||||
hw_priv->skb_cache = NULL;
|
||||
}
|
||||
return skb;
|
||||
}
|
||||
|
||||
static void bes2600_put_skb(struct bes2600_common *hw_priv, struct sk_buff *skb)
|
||||
{
|
||||
if (hw_priv->skb_cache)
|
||||
dev_kfree_skb(skb);
|
||||
else
|
||||
hw_priv->skb_cache = skb;
|
||||
}
|
||||
|
||||
static int bes2600_bh_read_ctrl_reg(struct bes2600_common *hw_priv,
|
||||
u16 *ctrl_reg)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = bes2600_reg_read_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, ctrl_reg);
|
||||
if (ret) {
|
||||
ret = bes2600_reg_read_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, ctrl_reg);
|
||||
if (ret)
|
||||
bes_err("[BH] Failed to read control register.\n");
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_device_wakeup(struct bes2600_common *hw_priv)
|
||||
{
|
||||
u16 ctrl_reg;
|
||||
int ret;
|
||||
|
||||
bes_devel("[BH] Device wakeup.\n");
|
||||
|
||||
/* To force the device to be always-on, the host sets WLAN_UP to 1 */
|
||||
ret = bes2600_reg_write_16(hw_priv, ST90TDS_CONTROL_REG_ID,
|
||||
ST90TDS_CONT_WUP_BIT);
|
||||
if (WARN_ON(ret))
|
||||
return ret;
|
||||
|
||||
ret = bes2600_bh_read_ctrl_reg(hw_priv, &ctrl_reg);
|
||||
if (WARN_ON(ret))
|
||||
return ret;
|
||||
|
||||
/* If the device returns WLAN_RDY as 1, the device is active and will
|
||||
* remain active. */
|
||||
if (ctrl_reg & ST90TDS_CONT_RDY_BIT) {
|
||||
bes_devel("[BH] Device awake.\n");
|
||||
return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/* Must be called from BH thraed. */
|
||||
void bes2600_enable_powersave(struct bes2600_vif *priv,
|
||||
@@ -403,475 +325,6 @@ void bes2600_enable_powersave(struct bes2600_vif *priv,
|
||||
priv->powersave_enabled = enable;
|
||||
}
|
||||
|
||||
#if 0
|
||||
#define INTERRUPT_WORKAROUND
|
||||
static int bes2600_bh(void *arg)
|
||||
{
|
||||
struct bes2600_common *hw_priv = arg;
|
||||
struct bes2600_vif *priv = NULL;
|
||||
struct sk_buff *skb_rx = NULL;
|
||||
size_t read_len = 0;
|
||||
int rx, tx, term, suspend;
|
||||
struct wsm_hdr *wsm;
|
||||
size_t wsm_len;
|
||||
int wsm_id;
|
||||
u8 wsm_seq;
|
||||
int rx_resync = 1;
|
||||
u16 ctrl_reg = 0;
|
||||
int tx_allowed;
|
||||
int pending_tx = 0;
|
||||
int tx_burst;
|
||||
int rx_burst = 0;
|
||||
long status;
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS)
|
||||
size_t wsm_dump_max = -1;
|
||||
#endif
|
||||
u32 dummy;
|
||||
bool powersave_enabled;
|
||||
int i;
|
||||
int vif_selected;
|
||||
|
||||
for (;;) {
|
||||
powersave_enabled = 1;
|
||||
spin_lock(&hw_priv->vif_list_lock);
|
||||
bes2600_for_each_vif(hw_priv, priv, i) {
|
||||
#ifdef P2P_MULTIVIF
|
||||
if ((i = (CW12XX_MAX_VIFS - 1)) || !priv)
|
||||
#else
|
||||
if (!priv)
|
||||
#endif
|
||||
continue;
|
||||
powersave_enabled &= !!priv->powersave_enabled;
|
||||
}
|
||||
spin_unlock(&hw_priv->vif_list_lock);
|
||||
if (!hw_priv->hw_bufs_used
|
||||
&& powersave_enabled
|
||||
&& !hw_priv->device_can_sleep
|
||||
&& !atomic_read(&hw_priv->recent_scan)) {
|
||||
status = HZ/8;
|
||||
bes_devel("[BH] No Device wakedown.\n");
|
||||
#ifndef FPGA_SETUP
|
||||
WARN_ON(bes2600_reg_write_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, 0));
|
||||
hw_priv->device_can_sleep = true;
|
||||
#endif
|
||||
} else if (hw_priv->hw_bufs_used)
|
||||
/* Interrupt loss detection */
|
||||
status = HZ/8;
|
||||
else
|
||||
status = HZ/8;
|
||||
|
||||
/* Dummy Read for SDIO retry mechanism*/
|
||||
if (((atomic_read(&hw_priv->bh_rx) == 0) &&
|
||||
(atomic_read(&hw_priv->bh_tx) == 0)))
|
||||
bes2600_reg_read(hw_priv, ST90TDS_CONFIG_REG_ID,
|
||||
&dummy, sizeof(dummy));
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS_SHORT)
|
||||
wsm_dump_max = hw_priv->wsm_dump_max_size;
|
||||
#endif /* CONFIG_BES2600_WSM_DUMPS_SHORT */
|
||||
|
||||
#ifdef INTERRUPT_WORKAROUND
|
||||
/* If a packet has already been txed to the device then read the
|
||||
control register for a probable interrupt miss before going
|
||||
further to wait for interrupt; if the read length is non-zero
|
||||
then it means there is some data to be received */
|
||||
if (hw_priv->hw_bufs_used) {
|
||||
bes2600_bh_read_ctrl_reg(hw_priv, &ctrl_reg);
|
||||
if(ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK)
|
||||
{
|
||||
rx = 1;
|
||||
goto test;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
status = wait_event_interruptible_timeout(hw_priv->bh_wq, ({
|
||||
rx = atomic_xchg(&hw_priv->bh_rx, 0);
|
||||
tx = atomic_xchg(&hw_priv->bh_tx, 0);
|
||||
term = atomic_xchg(&hw_priv->bh_term, 0);
|
||||
suspend = pending_tx ?
|
||||
0 : atomic_read(&hw_priv->bh_suspend);
|
||||
(rx || tx || term || suspend || hw_priv->bh_error);
|
||||
}), status);
|
||||
|
||||
if (status < 0 || term || hw_priv->bh_error)
|
||||
break;
|
||||
|
||||
#ifdef INTERRUPT_WORKAROUND
|
||||
if (!status) {
|
||||
bes2600_bh_read_ctrl_reg(hw_priv, &ctrl_reg);
|
||||
if(ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK)
|
||||
{
|
||||
bes_err("MISS 1\n");
|
||||
rx = 1;
|
||||
goto test;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
if (!status && hw_priv->hw_bufs_used) {
|
||||
unsigned long timestamp = jiffies;
|
||||
long timeout;
|
||||
bool pending = false;
|
||||
int i;
|
||||
|
||||
wiphy_warn(hw_priv->hw->wiphy, "Missed interrupt?\n");
|
||||
rx = 1;
|
||||
|
||||
/* Get a timestamp of "oldest" frame */
|
||||
for (i = 0; i < 4; ++i)
|
||||
pending |= bes2600_queue_get_xmit_timestamp(
|
||||
&hw_priv->tx_queue[i],
|
||||
×tamp, -1,
|
||||
hw_priv->pending_frame_id);
|
||||
|
||||
/* Check if frame transmission is timed out.
|
||||
* Add an extra second with respect to possible
|
||||
* interrupt loss. */
|
||||
timeout = timestamp +
|
||||
WSM_CMD_LAST_CHANCE_TIMEOUT +
|
||||
1 * HZ -
|
||||
jiffies;
|
||||
|
||||
/* And terminate BH tread if the frame is "stuck" */
|
||||
if (pending && timeout < 0) {
|
||||
//wiphy_warn(priv->hw->wiphy,
|
||||
// "Timeout waiting for TX confirm.\n");
|
||||
bes_devel("bes2600_bh: Timeout waiting for TX confirm.\n");
|
||||
break;
|
||||
}
|
||||
|
||||
#if defined(CONFIG_BES2600_DUMP_ON_ERROR)
|
||||
BUG_ON(1);
|
||||
#endif /* CONFIG_BES2600_DUMP_ON_ERROR */
|
||||
} else if (!status) {
|
||||
if (!hw_priv->device_can_sleep
|
||||
&& !atomic_read(&hw_priv->recent_scan)) {
|
||||
bes_devel("[BH] Device wakedown. Timeout.\n");
|
||||
#ifndef FPGA_SETUP
|
||||
WARN_ON(bes2600_reg_write_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, 0));
|
||||
hw_priv->device_can_sleep = true;
|
||||
#endif
|
||||
}
|
||||
continue;
|
||||
} else if (suspend) {
|
||||
bes_devel("[BH] Device suspend.\n");
|
||||
powersave_enabled = 1;
|
||||
spin_lock(&hw_priv->vif_list_lock);
|
||||
bes2600_for_each_vif(hw_priv, priv, i) {
|
||||
#ifdef P2P_MULTIVIF
|
||||
if ((i = (CW12XX_MAX_VIFS - 1)) || !priv)
|
||||
#else
|
||||
if (!priv)
|
||||
#endif
|
||||
continue;
|
||||
powersave_enabled &= !!priv->powersave_enabled;
|
||||
}
|
||||
spin_unlock(&hw_priv->vif_list_lock);
|
||||
if (powersave_enabled) {
|
||||
bes_devel("[BH] No Device wakedown. Suspend.\n");
|
||||
#ifndef FPGA_SETUP
|
||||
WARN_ON(bes2600_reg_write_16(hw_priv,
|
||||
ST90TDS_CONTROL_REG_ID, 0));
|
||||
hw_priv->device_can_sleep = true;
|
||||
#endif
|
||||
}
|
||||
|
||||
atomic_set(&hw_priv->bh_suspend, BES2600_BH_SUSPENDED);
|
||||
wake_up(&hw_priv->bh_evt_wq);
|
||||
status = wait_event_interruptible(hw_priv->bh_wq,
|
||||
BES2600_BH_RESUME == atomic_read(
|
||||
&hw_priv->bh_suspend));
|
||||
if (status < 0) {
|
||||
wiphy_err(hw_priv->hw->wiphy,
|
||||
"%s: Failed to wait for resume: %ld.\n",
|
||||
__func__, status);
|
||||
break;
|
||||
}
|
||||
bes_devel("[BH] Device resume.\n");
|
||||
atomic_set(&hw_priv->bh_suspend, BES2600_BH_RESUMED);
|
||||
wake_up(&hw_priv->bh_evt_wq);
|
||||
atomic_inc(&hw_priv->bh_rx);
|
||||
continue;
|
||||
}
|
||||
|
||||
test:
|
||||
tx += pending_tx;
|
||||
pending_tx = 0;
|
||||
|
||||
if (rx) {
|
||||
size_t alloc_len;
|
||||
u8 *data;
|
||||
|
||||
#ifdef INTERRUPT_WORKAROUND
|
||||
if(!(ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK))
|
||||
#endif
|
||||
if (WARN_ON(bes2600_bh_read_ctrl_reg(
|
||||
hw_priv, &ctrl_reg)))
|
||||
break;
|
||||
rx:
|
||||
read_len = (ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK) * 2;
|
||||
if (!read_len) {
|
||||
rx_burst = 0;
|
||||
goto tx;
|
||||
}
|
||||
|
||||
if (WARN_ON((read_len < sizeof(struct wsm_hdr)) ||
|
||||
(read_len > EFFECTIVE_BUF_SIZE))) {
|
||||
bes_devel("Invalid read len: %d", read_len);
|
||||
break;
|
||||
}
|
||||
|
||||
/* Add SIZE of PIGGYBACK reg (CONTROL Reg)
|
||||
* to the NEXT Message length + 2 Bytes for SKB */
|
||||
read_len = read_len + 2;
|
||||
|
||||
#if defined(CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES)
|
||||
alloc_len = hw_priv->sbus_ops->align_size(
|
||||
hw_priv->sbus_priv, read_len);
|
||||
#else /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
/* Platform's SDIO workaround */
|
||||
alloc_len = read_len & ~(SDIO_BLOCK_SIZE - 1);
|
||||
if (read_len & (SDIO_BLOCK_SIZE - 1))
|
||||
alloc_len += SDIO_BLOCK_SIZE;
|
||||
#endif /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
|
||||
/* Check if not exceeding BES2600 capabilities */
|
||||
if (WARN_ON_ONCE(alloc_len > EFFECTIVE_BUF_SIZE))
|
||||
bes_devel("Read aligned len: %d\n", alloc_len);
|
||||
|
||||
skb_rx = bes2600_get_skb(hw_priv, alloc_len);
|
||||
if (WARN_ON(!skb_rx))
|
||||
break;
|
||||
|
||||
skb_trim(skb_rx, 0);
|
||||
skb_put(skb_rx, read_len);
|
||||
data = skb_rx->data;
|
||||
if (WARN_ON(!data))
|
||||
break;
|
||||
|
||||
if (WARN_ON(bes2600_data_read(hw_priv, data, alloc_len)))
|
||||
break;
|
||||
|
||||
/* Piggyback */
|
||||
ctrl_reg = __le16_to_cpu(
|
||||
((__le16 *)data)[alloc_len / 2 - 1]);
|
||||
|
||||
wsm = (struct wsm_hdr *)data;
|
||||
wsm_len = __le32_to_cpu(wsm->len);
|
||||
if (WARN_ON(wsm_len > read_len))
|
||||
break;
|
||||
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS)
|
||||
if (unlikely(hw_priv->wsm_enable_wsm_dumps)) {
|
||||
u16 msgid, ifid;
|
||||
u16 *p = (u16 *)data;
|
||||
msgid = (*(p + 1)) & 0xC3F;
|
||||
ifid = (*(p + 1)) >> 6;
|
||||
ifid &= 0xF;
|
||||
bes_devel("[DUMP] <<< msgid 0x%.4X ifid %d len %d\n", msgid, ifid, *p);
|
||||
print_hex_dump(KERN_DEBUG, "<-- ", DUMP_PREFIX_NONE, data, min(wsm_len, wsm_dump_max));
|
||||
}
|
||||
#endif /* CONFIG_BES2600_WSM_DUMPS */
|
||||
|
||||
wsm_id = __le32_to_cpu(wsm->id) & 0xFFF;
|
||||
wsm_seq = (__le32_to_cpu(wsm->id) >> 13) & 7;
|
||||
|
||||
skb_trim(skb_rx, wsm_len);
|
||||
|
||||
if (unlikely(wsm_id == 0x0800)) {
|
||||
wsm_handle_exception(hw_priv,
|
||||
&data[sizeof(*wsm)],
|
||||
wsm_len - sizeof(*wsm));
|
||||
break;
|
||||
} else if (unlikely(!rx_resync)) {
|
||||
if (WARN_ON(wsm_seq != hw_priv->wsm_rx_seq)) {
|
||||
#if defined(CONFIG_BES2600_DUMP_ON_ERROR)
|
||||
BUG_ON(1);
|
||||
#endif /* CONFIG_BES2600_DUMP_ON_ERROR */
|
||||
break;
|
||||
}
|
||||
}
|
||||
hw_priv->wsm_rx_seq = (wsm_seq + 1) & 7;
|
||||
rx_resync = 0;
|
||||
|
||||
if (wsm_id & 0x0400) {
|
||||
int rc = wsm_release_tx_buffer(hw_priv, 1);
|
||||
if (WARN_ON(rc < 0))
|
||||
break;
|
||||
else if (rc > 0)
|
||||
tx = 1;
|
||||
}
|
||||
|
||||
/* bes2600_wsm_rx takes care on SKB livetime */
|
||||
if (WARN_ON(wsm_handle_rx(hw_priv, wsm_id, wsm,
|
||||
&skb_rx)))
|
||||
break;
|
||||
|
||||
if (skb_rx) {
|
||||
bes2600_put_skb(hw_priv, skb_rx);
|
||||
skb_rx = NULL;
|
||||
}
|
||||
|
||||
read_len = 0;
|
||||
|
||||
if (rx_burst) {
|
||||
bes2600_debug_rx_burst(hw_priv);
|
||||
--rx_burst;
|
||||
goto rx;
|
||||
}
|
||||
}
|
||||
|
||||
tx:
|
||||
BUG_ON(hw_priv->hw_bufs_used > hw_priv->wsm_caps.numInpChBufs);
|
||||
tx_burst = hw_priv->wsm_caps.numInpChBufs -
|
||||
hw_priv->hw_bufs_used;
|
||||
tx_allowed = tx_burst > 0;
|
||||
if (tx && tx_allowed) {
|
||||
size_t tx_len;
|
||||
u8 *data;
|
||||
int ret;
|
||||
|
||||
if (hw_priv->device_can_sleep) {
|
||||
ret = bes2600_device_wakeup(hw_priv);
|
||||
if (WARN_ON(ret < 0))
|
||||
break;
|
||||
else if (ret)
|
||||
hw_priv->device_can_sleep = false;
|
||||
else {
|
||||
/* Wait for "awake" interrupt */
|
||||
pending_tx = tx;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
wsm_alloc_tx_buffer(hw_priv);
|
||||
ret = wsm_get_tx(hw_priv, &data, &tx_len, &tx_burst,
|
||||
&vif_selected);
|
||||
if (ret <= 0) {
|
||||
wsm_release_tx_buffer(hw_priv, 1);
|
||||
if (WARN_ON(ret < 0))
|
||||
break;
|
||||
} else {
|
||||
wsm = (struct wsm_hdr *)data;
|
||||
BUG_ON(tx_len < sizeof(*wsm));
|
||||
BUG_ON(__le32_to_cpu(wsm->len) != tx_len);
|
||||
|
||||
#if 0 /* count is not implemented */
|
||||
if (ret > 1)
|
||||
atomic_inc(&hw_priv->bh_tx);
|
||||
#else
|
||||
atomic_inc(&hw_priv->bh_tx);
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES)
|
||||
if (tx_len <= 8)
|
||||
tx_len = 16;
|
||||
tx_len = hw_priv->sbus_ops->align_size(
|
||||
hw_priv->sbus_priv, tx_len);
|
||||
#else /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
/* HACK!!! Platform limitation.
|
||||
* It is also supported by upper layer:
|
||||
* there is always enough space at the
|
||||
* end of the buffer. */
|
||||
if (tx_len & (SDIO_BLOCK_SIZE - 1)) {
|
||||
tx_len &= ~(SDIO_BLOCK_SIZE - 1);
|
||||
tx_len += SDIO_BLOCK_SIZE;
|
||||
}
|
||||
#endif /* CONFIG_BES2600_NON_POWER_OF_TWO_BLOCKSIZES */
|
||||
|
||||
/* Check if not exceeding BES2600
|
||||
capabilities */
|
||||
if (WARN_ON_ONCE(tx_len > EFFECTIVE_BUF_SIZE))
|
||||
bes_devel("Write aligned len: %d\n", tx_len);
|
||||
|
||||
wsm->id &= __cpu_to_le32(
|
||||
~WSM_TX_SEQ(WSM_TX_SEQ_MAX));
|
||||
wsm->id |= cpu_to_le32(WSM_TX_SEQ(
|
||||
hw_priv->wsm_tx_seq));
|
||||
|
||||
if (WARN_ON(bes2600_data_write(hw_priv,
|
||||
data, tx_len))) {
|
||||
wsm_release_tx_buffer(hw_priv, 1);
|
||||
break;
|
||||
}
|
||||
|
||||
if (vif_selected != -1) {
|
||||
hw_priv->hw_bufs_used_vif[
|
||||
vif_selected]++;
|
||||
}
|
||||
|
||||
#if defined(CONFIG_BES2600_WSM_DUMPS)
|
||||
if (unlikely(hw_priv->wsm_enable_wsm_dumps)) {
|
||||
u16 msgid, ifid;
|
||||
u16 *p = (u16 *)data;
|
||||
msgid = (*(p + 1)) & 0x3F;
|
||||
ifid = (*(p + 1)) >> 6;
|
||||
ifid &= 0xF;
|
||||
if (msgid == 0x0006)
|
||||
bes_devel("[DUMP] >>> msgid 0x%.4X ifid %d len %d MIB 0x%.4X\n", msgid, ifid, *p, *(p + 2));
|
||||
else
|
||||
bes_devel("[DUMP] >>> msgid 0x%.4X ifid %d len %d\n", msgid, ifid, *p);
|
||||
print_hex_dump(KERN_DEBUG, "--> ", DUMP_PREFIX_NONE, data, min(__le32_to_cpu(wsm->len), wsm_dump_max));
|
||||
}
|
||||
#endif /* CONFIG_BES2600_WSM_DUMPS */
|
||||
|
||||
wsm_txed(hw_priv, data);
|
||||
hw_priv->wsm_tx_seq = (hw_priv->wsm_tx_seq + 1)
|
||||
& WSM_TX_SEQ_MAX;
|
||||
|
||||
if (tx_burst > 1) {
|
||||
bes2600_debug_tx_burst(hw_priv);
|
||||
++rx_burst;
|
||||
goto tx;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (ctrl_reg & ST90TDS_CONT_NEXT_LEN_MASK)
|
||||
goto rx;
|
||||
}
|
||||
|
||||
if (skb_rx) {
|
||||
bes2600_put_skb(hw_priv, skb_rx);
|
||||
skb_rx = NULL;
|
||||
}
|
||||
|
||||
|
||||
if (!term) {
|
||||
bes_devel("[BH] Fatal error, exitting.\n");
|
||||
#if defined(CONFIG_BES2600_DUMP_ON_ERROR)
|
||||
BUG_ON(1);
|
||||
#endif /* CONFIG_BES2600_DUMP_ON_ERROR */
|
||||
hw_priv->bh_error = 1;
|
||||
#if defined(CONFIG_BES2600_USE_STE_EXTENSIONS)
|
||||
spin_lock(&hw_priv->vif_list_lock);
|
||||
bes2600_for_each_vif(hw_priv, priv, i) {
|
||||
if (!priv)
|
||||
continue;
|
||||
ieee80211_driver_hang_notify(priv->vif, GFP_KERNEL);
|
||||
}
|
||||
spin_unlock(&hw_priv->vif_list_lock);
|
||||
bes2600_pm_stay_awake(&hw_priv->pm_state, 3*HZ);
|
||||
#endif
|
||||
/* TODO: schedule_work(recovery) */
|
||||
#ifndef HAS_PUT_TASK_STRUCT
|
||||
/* The only reason of having this stupid code here is
|
||||
* that __put_task_struct is not exported by kernel. */
|
||||
for (;;) {
|
||||
int status = wait_event_interruptible(hw_priv->bh_wq, ({
|
||||
term = atomic_xchg(&hw_priv->bh_term, 0);
|
||||
(term);
|
||||
}));
|
||||
|
||||
if (status || term)
|
||||
break;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#else
|
||||
|
||||
extern int bes2600_bh_read_ctrl_reg(struct bes2600_common *priv, u32 *ctrl_reg);
|
||||
|
||||
@@ -1381,8 +834,6 @@ int bes2600_bh_sw_process(struct bes2600_common *hw_priv,
|
||||
delta_time = jiffies + ((unsigned long)0xffffffff - timestamp);
|
||||
else
|
||||
delta_time = jiffies - timestamp;
|
||||
bes2600_add_tx_delta_time(delta_time);
|
||||
bes2600_add_tx_ac_delta_time(queue_id, delta_time);
|
||||
|
||||
if (bes2600_need_retry_type(skb, tx_confirm->status) == 0)
|
||||
return -1;
|
||||
@@ -1391,12 +842,8 @@ int bes2600_bh_sw_process(struct bes2600_common *hw_priv,
|
||||
return -1;
|
||||
|
||||
if (txpriv->retry_count < CW1200_MAX_SW_RETRY_CNT ) {
|
||||
struct bes2600_vif *priv =
|
||||
__cw12xx_hwpriv_to_vifpriv(hw_priv, txpriv->if_id);
|
||||
txpriv->retry_count++;
|
||||
|
||||
bes2600_tx_status(priv,skb);
|
||||
|
||||
bes2600_pwr_set_busy_event_with_timeout_async(
|
||||
hw_priv, BES_PWR_LOCK_ON_TX, BES_PWR_EVENT_TX_TIMEOUT);
|
||||
|
||||
@@ -1432,14 +879,14 @@ void bes2600_bh_dec_pending_count(struct bes2600_common *hw_priv, int idx)
|
||||
}
|
||||
|
||||
if (--hw_priv->wsm_tx_pending[idx] == 0)
|
||||
timer_delete_sync(timer);
|
||||
del_timer_sync(timer);
|
||||
else
|
||||
mod_timer(timer, jiffies + 3 * HZ);
|
||||
}
|
||||
|
||||
void bes2600_bh_mcu_active_monitor(struct timer_list* t)
|
||||
{
|
||||
struct bes2600_common *hw_priv = timer_container_of(hw_priv, t, mcu_mon_timer);
|
||||
struct bes2600_common *hw_priv = from_timer(hw_priv, t, mcu_mon_timer);
|
||||
|
||||
bes_err("link break between mcu and host, hw_buf_used:%d pending:%d\n",
|
||||
hw_priv->hw_bufs_used, hw_priv->wsm_tx_pending[1]);
|
||||
@@ -1448,7 +895,7 @@ void bes2600_bh_mcu_active_monitor(struct timer_list* t)
|
||||
|
||||
void bes2600_bh_lmac_active_monitor(struct timer_list* t)
|
||||
{
|
||||
struct bes2600_common *hw_priv = timer_container_of(hw_priv, t, lmac_mon_timer);
|
||||
struct bes2600_common *hw_priv = from_timer(hw_priv, t, lmac_mon_timer);
|
||||
|
||||
bes_err("link break between lmac and host, hw_buf_used:%d pending:%d\n",
|
||||
hw_priv->hw_bufs_used, hw_priv->wsm_tx_pending[0]);
|
||||
@@ -1599,7 +1046,15 @@ static int bes2600_bh(void *arg)
|
||||
|
||||
tx = 0;
|
||||
|
||||
BUG_ON(hw_priv->hw_bufs_used > hw_priv->wsm_caps.numInpChBufs);
|
||||
/*
|
||||
* Patch H: BUG_ON -> WARN_ON_ONCE in the steady-state
|
||||
* hot path. The original BUG_ON ran every bh-loop
|
||||
* iteration; tripping it on a bookkeeping bug locks
|
||||
* the kernel up during normal operation, which is
|
||||
* the wrong response. WARN_ON_ONCE surfaces the
|
||||
* issue without taking the system down.
|
||||
*/
|
||||
WARN_ON_ONCE(hw_priv->hw_bufs_used > hw_priv->wsm_caps.numInpChBufs);
|
||||
tx_burst = hw_priv->wsm_caps.numInpChBufs - hw_priv->hw_bufs_used;
|
||||
tx_allowed = tx_burst > 0;
|
||||
|
||||
@@ -1643,18 +1098,19 @@ static int bes2600_bh(void *arg)
|
||||
goto tx;
|
||||
|
||||
done:
|
||||
/* Re-enable device interrupts */
|
||||
//hw_priv->sbus_ops->lock(hw_priv->sbus_priv);
|
||||
//__bes2600_irq_enable(1);
|
||||
//hw_priv->sbus_ops->unlock(hw_priv->sbus_priv);
|
||||
asm volatile ("nop");
|
||||
/*
|
||||
* Patch H: dropped the dead `__bes2600_irq_enable(1)` /
|
||||
* `asm volatile("nop")` placeholder that used to sit here.
|
||||
* `__bes2600_irq_enable()` is a stub that returns 0 on
|
||||
* bes2600 silicon — the IRQ is managed by sdio_claim_irq
|
||||
* and chip-side firmware, not by a driver-side enable bit.
|
||||
* (cw1200 inherited the function from a different chip
|
||||
* shape; bes2600 kept the stub but the call sites are
|
||||
* meaningless.)
|
||||
*/
|
||||
;
|
||||
}
|
||||
|
||||
/* Explicitly disable device interrupts */
|
||||
hw_priv->sbus_ops->lock(hw_priv->sbus_priv);
|
||||
__bes2600_irq_enable(0);
|
||||
hw_priv->sbus_ops->unlock(hw_priv->sbus_priv);
|
||||
|
||||
if (!term) {
|
||||
bes_err("[BH] Fatal error, exiting.\n");
|
||||
sdio_work_debug(hw_priv->sbus_priv);
|
||||
@@ -1663,4 +1119,3 @@ static int bes2600_bh(void *arg)
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
@@ -1,154 +0,0 @@
|
||||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* Copy of drivers/bluetooth/h4_recv.h
|
||||
*
|
||||
* Generic Bluetooth HCI UART driver
|
||||
*
|
||||
* Copyright (C) 2015-2018 Intel Corporation
|
||||
*/
|
||||
|
||||
#include <linux/unaligned.h>
|
||||
|
||||
struct h4_recv_pkt {
|
||||
u8 type; /* Packet type */
|
||||
u8 hlen; /* Header length */
|
||||
u8 loff; /* Data length offset in header */
|
||||
u8 lsize; /* Data length field size */
|
||||
u16 maxlen; /* Max overall packet length */
|
||||
int (*recv)(struct hci_dev *hdev, struct sk_buff *skb);
|
||||
};
|
||||
|
||||
#define H4_RECV_ACL \
|
||||
.type = HCI_ACLDATA_PKT, \
|
||||
.hlen = HCI_ACL_HDR_SIZE, \
|
||||
.loff = 2, \
|
||||
.lsize = 2, \
|
||||
.maxlen = HCI_MAX_FRAME_SIZE \
|
||||
|
||||
#define H4_RECV_SCO \
|
||||
.type = HCI_SCODATA_PKT, \
|
||||
.hlen = HCI_SCO_HDR_SIZE, \
|
||||
.loff = 2, \
|
||||
.lsize = 1, \
|
||||
.maxlen = HCI_MAX_SCO_SIZE
|
||||
|
||||
#define H4_RECV_EVENT \
|
||||
.type = HCI_EVENT_PKT, \
|
||||
.hlen = HCI_EVENT_HDR_SIZE, \
|
||||
.loff = 1, \
|
||||
.lsize = 1, \
|
||||
.maxlen = HCI_MAX_EVENT_SIZE
|
||||
|
||||
#define H4_RECV_ISO \
|
||||
.type = HCI_ISODATA_PKT, \
|
||||
.hlen = HCI_ISO_HDR_SIZE, \
|
||||
.loff = 2, \
|
||||
.lsize = 2, \
|
||||
.maxlen = HCI_MAX_FRAME_SIZE
|
||||
|
||||
static inline struct sk_buff *h4_recv_buf(struct hci_dev *hdev,
|
||||
struct sk_buff *skb,
|
||||
const unsigned char *buffer,
|
||||
int count,
|
||||
const struct h4_recv_pkt *pkts,
|
||||
int pkts_count)
|
||||
{
|
||||
/* Check for error from previous call */
|
||||
if (IS_ERR(skb))
|
||||
skb = NULL;
|
||||
|
||||
while (count) {
|
||||
int i, len;
|
||||
|
||||
if (!skb) {
|
||||
for (i = 0; i < pkts_count; i++) {
|
||||
if (buffer[0] != (&pkts[i])->type)
|
||||
continue;
|
||||
|
||||
skb = bt_skb_alloc((&pkts[i])->maxlen,
|
||||
GFP_ATOMIC);
|
||||
if (!skb)
|
||||
return ERR_PTR(-ENOMEM);
|
||||
|
||||
hci_skb_pkt_type(skb) = (&pkts[i])->type;
|
||||
hci_skb_expect(skb) = (&pkts[i])->hlen;
|
||||
break;
|
||||
}
|
||||
|
||||
/* Check for invalid packet type */
|
||||
if (!skb)
|
||||
return ERR_PTR(-EILSEQ);
|
||||
|
||||
count -= 1;
|
||||
buffer += 1;
|
||||
}
|
||||
|
||||
len = min_t(uint, hci_skb_expect(skb) - skb->len, count);
|
||||
skb_put_data(skb, buffer, len);
|
||||
|
||||
count -= len;
|
||||
buffer += len;
|
||||
|
||||
/* Check for partial packet */
|
||||
if (skb->len < hci_skb_expect(skb))
|
||||
continue;
|
||||
|
||||
for (i = 0; i < pkts_count; i++) {
|
||||
if (hci_skb_pkt_type(skb) == (&pkts[i])->type)
|
||||
break;
|
||||
}
|
||||
|
||||
if (i >= pkts_count) {
|
||||
kfree_skb(skb);
|
||||
return ERR_PTR(-EILSEQ);
|
||||
}
|
||||
|
||||
if (skb->len == (&pkts[i])->hlen) {
|
||||
u16 dlen;
|
||||
|
||||
switch ((&pkts[i])->lsize) {
|
||||
case 0:
|
||||
/* No variable data length */
|
||||
dlen = 0;
|
||||
break;
|
||||
case 1:
|
||||
/* Single octet variable length */
|
||||
dlen = skb->data[(&pkts[i])->loff];
|
||||
hci_skb_expect(skb) += dlen;
|
||||
|
||||
if (skb_tailroom(skb) < dlen) {
|
||||
kfree_skb(skb);
|
||||
return ERR_PTR(-EMSGSIZE);
|
||||
}
|
||||
break;
|
||||
case 2:
|
||||
/* Double octet variable length */
|
||||
dlen = get_unaligned_le16(skb->data +
|
||||
(&pkts[i])->loff);
|
||||
hci_skb_expect(skb) += dlen;
|
||||
|
||||
if (skb_tailroom(skb) < dlen) {
|
||||
kfree_skb(skb);
|
||||
return ERR_PTR(-EMSGSIZE);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
/* Unsupported variable length */
|
||||
kfree_skb(skb);
|
||||
return ERR_PTR(-EILSEQ);
|
||||
}
|
||||
|
||||
if (!dlen) {
|
||||
/* No more data, complete frame */
|
||||
(&pkts[i])->recv(hdev, skb);
|
||||
skb = NULL;
|
||||
}
|
||||
} else {
|
||||
/* Complete frame */
|
||||
(&pkts[i])->recv(hdev, skb);
|
||||
skb = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
return skb;
|
||||
}
|
||||
+7
-4
@@ -324,7 +324,10 @@ out:
|
||||
}
|
||||
#endif
|
||||
|
||||
int __bes2600_irq_enable(int enable)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
/*
|
||||
* Patch H: __bes2600_irq_enable stub removed. It was a no-op
|
||||
* (always returned 0) inherited from cw1200 where the analogous
|
||||
* function manipulates the chip's IRQ-enable register. bes2600
|
||||
* silicon manages SDIO IRQ via sdio_claim_irq and chip-side
|
||||
* firmware — there is no driver-side enable register to write.
|
||||
*/
|
||||
|
||||
+4
-9
@@ -38,7 +38,6 @@
|
||||
#include "pm.h"
|
||||
#include "bes2600_factory.h"
|
||||
#include "bes_chardev.h"
|
||||
#include "txrx_opt.h"
|
||||
|
||||
MODULE_AUTHOR("Dmitry Tarnyagin <dmitry.tarnyagin@stericsson.com>");
|
||||
MODULE_DESCRIPTION("Softmac BES2600 common code");
|
||||
@@ -205,11 +204,7 @@ static const struct ieee80211_iface_limit bes2600_if_limits[] = {
|
||||
BIT(NL80211_IFTYPE_P2P_CLIENT) |
|
||||
BIT(NL80211_IFTYPE_P2P_GO) },
|
||||
#ifdef P2P_MULTIVIF
|
||||
/*
|
||||
* HACK: Disable P2P_DEVICE implementation for BES2600
|
||||
* as the code is a little buggy.
|
||||
*/
|
||||
//{ .max = 1, .types = BIT(NL80211_IFTYPE_P2P_DEVICE) },
|
||||
{ .max = 1, .types = BIT(NL80211_IFTYPE_P2P_DEVICE) },
|
||||
#endif
|
||||
};
|
||||
|
||||
@@ -608,7 +603,7 @@ static void bes2600_unregister_common(struct ieee80211_hw *dev)
|
||||
|
||||
ieee80211_unregister_hw(dev);
|
||||
|
||||
timer_delete_sync(&hw_priv->ba_timer);
|
||||
del_timer_sync(&hw_priv->ba_timer);
|
||||
|
||||
hw_priv->sbus_ops->irq_unsubscribe(hw_priv->sbus_priv);
|
||||
bes2600_unregister_bh(hw_priv);
|
||||
@@ -876,8 +871,8 @@ int bes2600_wifi_stop(struct bes2600_common *hw_priv)
|
||||
hw_priv->wsm_tx_seq[1] = 0;
|
||||
hw_priv->wsm_tx_pending[0] = 0;
|
||||
hw_priv->wsm_tx_pending[1] = 0;
|
||||
timer_delete_sync(&hw_priv->mcu_mon_timer);
|
||||
timer_delete_sync(&hw_priv->lmac_mon_timer);
|
||||
del_timer_sync(&hw_priv->mcu_mon_timer);
|
||||
del_timer_sync(&hw_priv->lmac_mon_timer);
|
||||
#ifdef CONFIG_BES2600_STATIC_SDD
|
||||
hw_priv->sdd = NULL;
|
||||
#else
|
||||
|
||||
+13
-11
@@ -119,9 +119,10 @@ static void bes2600_queue_register_post_gc(struct list_head *gc_list,
|
||||
struct bes2600_queue_item *item)
|
||||
{
|
||||
struct bes2600_queue_item *gc_item;
|
||||
gc_item = kmemdup(item, sizeof(struct bes2600_queue_item),
|
||||
gc_item = kmalloc(sizeof(struct bes2600_queue_item),
|
||||
GFP_ATOMIC);
|
||||
BUG_ON(!gc_item);
|
||||
memcpy(gc_item, item, sizeof(struct bes2600_queue_item));
|
||||
list_add_tail(&gc_item->head, gc_list);
|
||||
}
|
||||
|
||||
@@ -130,9 +131,9 @@ static void bes2600_queue_pending_record(struct list_head *pending_record_list,
|
||||
{
|
||||
struct bes2600_queue_item *record_item;
|
||||
|
||||
record_item = kmemdup(pending_item, sizeof(struct bes2600_queue_item),
|
||||
GFP_ATOMIC);
|
||||
record_item = kmalloc(sizeof(struct bes2600_queue_item),GFP_ATOMIC);
|
||||
BUG_ON(!record_item);
|
||||
memcpy(record_item, pending_item, sizeof(struct bes2600_queue_item));
|
||||
record_item->skb = skb_clone(pending_item->skb, GFP_ATOMIC);
|
||||
list_add_tail(&record_item->head, pending_record_list);
|
||||
}
|
||||
@@ -195,7 +196,7 @@ static void __bes2600_queue_gc(struct bes2600_queue *queue,
|
||||
static void bes2600_queue_gc(struct timer_list *t)
|
||||
{
|
||||
LIST_HEAD(list);
|
||||
struct bes2600_queue *queue = timer_container_of(queue, t, gc);
|
||||
struct bes2600_queue *queue = from_timer(queue, t, gc);
|
||||
|
||||
spin_lock_bh(&queue->lock);
|
||||
__bes2600_queue_gc(queue, &list, true);
|
||||
@@ -217,7 +218,7 @@ int bes2600_queue_stats_init(struct bes2600_queue_stats *stats,
|
||||
spin_lock_init(&stats->lock);
|
||||
init_waitqueue_head(&stats->wait_link_id_empty);
|
||||
for (i = 0; i < CW12XX_MAX_VIFS; i++) {
|
||||
stats->link_map_cache[i] = kcalloc(map_capacity, sizeof(int),
|
||||
stats->link_map_cache[i] = kzalloc(map_capacity * sizeof(int),
|
||||
GFP_KERNEL);
|
||||
if (!stats->link_map_cache[i]) {
|
||||
for (; i >= 0; i--)
|
||||
@@ -248,14 +249,14 @@ int bes2600_queue_init(struct bes2600_queue *queue,
|
||||
queue->queue_all_lock = false;
|
||||
spin_lock_init(&queue->lock);
|
||||
timer_setup(&queue->gc, bes2600_queue_gc, 0);
|
||||
queue->pool = kcalloc(capacity, sizeof(struct bes2600_queue_item),
|
||||
queue->pool = kzalloc(sizeof(struct bes2600_queue_item) * capacity,
|
||||
GFP_KERNEL);
|
||||
if (!queue->pool)
|
||||
return -ENOMEM;
|
||||
|
||||
for (i = 0; i < CW12XX_MAX_VIFS; i++) {
|
||||
queue->link_map_cache[i] =
|
||||
kcalloc(stats->map_capacity, sizeof(int),
|
||||
kzalloc(stats->map_capacity * sizeof(int),
|
||||
GFP_KERNEL);
|
||||
if (!queue->link_map_cache[i]) {
|
||||
for (; i >= 0; i--)
|
||||
@@ -362,7 +363,7 @@ void bes2600_queue_deinit(struct bes2600_queue *queue)
|
||||
int i;
|
||||
|
||||
bes2600_queue_clear(queue, CW12XX_ALL_IFS);
|
||||
timer_delete_sync(&queue->gc);
|
||||
del_timer_sync(&queue->gc);
|
||||
INIT_LIST_HEAD(&queue->free_pool);
|
||||
kfree(queue->pool);
|
||||
for (i = 0; i < CW12XX_MAX_VIFS; i++) {
|
||||
@@ -409,6 +410,7 @@ int bes2600_queue_put(struct bes2600_queue *queue,
|
||||
struct timespec64 tmval;
|
||||
#endif /*CONFIG_BES2600_TESTMODE*/
|
||||
|
||||
LIST_HEAD(gc_list);
|
||||
struct bes2600_queue_stats *stats = queue->stats;
|
||||
/* TODO:COMBO: Add interface ID info to queue item */
|
||||
|
||||
@@ -827,19 +829,19 @@ int bes2600_queue_get_skb(struct bes2600_queue *queue, u32 packetID,
|
||||
bes2600_queue_parse_id(packetID, &queue_generation, &queue_id,
|
||||
&item_generation, &item_id, &if_id, &link_id);
|
||||
|
||||
spin_lock(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
spin_lock_bh(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
if (!list_empty(&queue->stats->hw_priv->tx_loop.pending_record_list)) {
|
||||
list_for_each_entry_safe(record_item, temp_record_item, &queue->stats->hw_priv->tx_loop.pending_record_list, head) {
|
||||
if (record_item->packetID == packetID) {
|
||||
list_del(&record_item->head);
|
||||
dev_kfree_skb(record_item->skb);
|
||||
kfree(record_item);
|
||||
spin_unlock(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
spin_unlock_bh(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
}
|
||||
spin_unlock(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
spin_unlock_bh(&queue->stats->hw_priv->tx_loop.pending_record_lock);
|
||||
|
||||
item = &queue->pool[item_id];
|
||||
|
||||
|
||||
+1
-2
@@ -95,7 +95,6 @@ struct sbus_ops {
|
||||
|
||||
void bes2600_irq_handler(struct bes2600_common *priv);
|
||||
|
||||
/* This MUST be wrapped with hwbus_ops->lock/unlock! */
|
||||
int __bes2600_irq_enable(int enable);
|
||||
/* Patch H: __bes2600_irq_enable removed (was a stub). */
|
||||
|
||||
#endif /* BES2600_SBUS_H */
|
||||
|
||||
+35
-6
@@ -238,6 +238,36 @@ int bes2600_hw_scan(struct ieee80211_hw *hw,
|
||||
/* Scan when P2P_GO corrupt firmware MiniAP mode */
|
||||
if (priv->join_status == BES2600_JOIN_STATUS_AP)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
/*
|
||||
* Firmware refuses WSM start-scan for 5 GHz with status 2 ("rejected
|
||||
* by policy"); see besser issue #1. mac80211 splits multi-band
|
||||
* hw_scan requests per-band when the driver does not set
|
||||
* IEEE80211_HW_SINGLE_SCAN_ON_ALL_BANDS (we don't -- see
|
||||
* ieee80211_hw_set() calls in bes2600_main.c), so each per-band call
|
||||
* has req->channels[] from one band only (see ieee80211_prep_hw_scan
|
||||
* in net/mac80211/scan.c). Refuse the 5 GHz iteration at the driver
|
||||
* boundary so userspace gets a clean aborted-scan for that portion
|
||||
* rather than waiting for the firmware reject to cascade up.
|
||||
*
|
||||
* Only the multi-channel case is refused (n_channels > 1): that's
|
||||
* the per-band-sweep pattern mac80211 issues internally and the
|
||||
* one that triggers the firmware storm at the per-band loop
|
||||
* boundary. Single-channel 5 GHz scans (BSS verification, NM's
|
||||
* per-freq iteration when 802-11-wireless.band=a is set) pass
|
||||
* through to firmware, which generally accepts them since the
|
||||
* storm is the back-to-back per-band issue, not a blanket 5 GHz
|
||||
* reject. This preserves 5 GHz association via the
|
||||
* "wpa_supplicant iterates freq_list per channel" path.
|
||||
*
|
||||
* Contract: per include/net/mac80211.h struct ieee80211_ops.hw_scan
|
||||
* documentation, a negative return aborts the scan without requiring
|
||||
* ieee80211_scan_completed().
|
||||
*/
|
||||
if (req->n_channels > 1 &&
|
||||
req->channels[0]->band == NL80211_BAND_5GHZ)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
#if 0
|
||||
if (work_pending(&priv->offchannel_work) ||
|
||||
(hw_priv->roc_if_id != -1)) {
|
||||
@@ -592,10 +622,10 @@ void bes2600_scan_work(struct work_struct *work)
|
||||
"[SCAN] Scan failed (%d).\n",
|
||||
hw_priv->scan.status);
|
||||
else if (hw_priv->scan.req)
|
||||
wiphy_dbg(priv->hw->wiphy,
|
||||
wiphy_info(priv->hw->wiphy,
|
||||
"[SCAN] Scan completed.\n");
|
||||
else
|
||||
wiphy_dbg(priv->hw->wiphy,
|
||||
wiphy_info(priv->hw->wiphy,
|
||||
"[SCAN] Scan canceled.\n");
|
||||
|
||||
#ifdef WIFI_BT_COEXIST_EPTA_ENABLE
|
||||
@@ -680,9 +710,8 @@ void bes2600_scan_work(struct work_struct *work)
|
||||
scan.scanType = WSM_SCAN_TYPE_BACKGROUND;
|
||||
scan.scanFlags = WSM_SCAN_FLAG_FORCE_BACKGROUND;
|
||||
}
|
||||
scan.ch = kcalloc((it - hw_priv->scan.curr),
|
||||
sizeof(struct wsm_scan_ch),
|
||||
GFP_KERNEL);
|
||||
scan.ch = kzalloc((it - hw_priv->scan.curr) *
|
||||
sizeof(struct wsm_scan_ch), GFP_KERNEL);
|
||||
if (!scan.ch) {
|
||||
hw_priv->scan.status = -ENOMEM;
|
||||
goto fail;
|
||||
@@ -984,7 +1013,7 @@ void bes2600_scan_complete_cb(struct bes2600_common *hw_priv,
|
||||
// recover EPTA timer after scan wsm msg complete, in case of epta state error
|
||||
// bwifi_change_current_status(hw_priv, BWIFI_STATUS_SCANNING_COMP);
|
||||
#endif
|
||||
wiphy_dbg(hw_priv->hw->wiphy, "bes2600_scan_complete_cb status: %u", arg->status);
|
||||
wiphy_info(hw_priv->hw->wiphy, "bes2600_scan_complete_cb status: %u", arg->status);
|
||||
|
||||
if(hw_priv->scan.status == -ETIMEDOUT)
|
||||
wiphy_warn(hw_priv->hw->wiphy,
|
||||
|
||||
+51
-33
@@ -42,8 +42,6 @@
|
||||
#include "bes2600_factory.h"
|
||||
#endif
|
||||
|
||||
#include "txrx_opt.h"
|
||||
|
||||
#define WEP_ENCRYPT_HDR_SIZE 4
|
||||
#define WEP_ENCRYPT_TAIL_SIZE 4
|
||||
#define WPA_ENCRYPT_HDR_SIZE 8
|
||||
@@ -221,7 +219,7 @@ void bes2600_stop(struct ieee80211_hw *dev, bool suspend)
|
||||
cancel_delayed_work_sync(&hw_priv->advance_scan_timeout);
|
||||
#endif
|
||||
flush_workqueue(hw_priv->workqueue);
|
||||
timer_delete_sync(&hw_priv->ba_timer);
|
||||
del_timer_sync(&hw_priv->ba_timer);
|
||||
|
||||
down(&hw_priv->conf_lock);
|
||||
|
||||
@@ -261,7 +259,7 @@ void bes2600_stop(struct ieee80211_hw *dev, bool suspend)
|
||||
cancel_delayed_work_sync(&priv->bss_loss_work);
|
||||
cancel_delayed_work_sync(&priv->connection_loss_work);
|
||||
cancel_delayed_work_sync(&priv->link_id_gc_work);
|
||||
timer_delete_sync(&priv->mcast_timeout);
|
||||
del_timer_sync(&priv->mcast_timeout);
|
||||
}
|
||||
|
||||
#ifdef WIFI_BT_COEXIST_EPTA_ENABLE
|
||||
@@ -378,23 +376,9 @@ void bes2600_remove_interface(struct ieee80211_hw *dev,
|
||||
atomic_set(&priv->enabled, 0);
|
||||
down(&hw_priv->scan.lock);
|
||||
down(&hw_priv->conf_lock);
|
||||
|
||||
/*
|
||||
* There's a chance remove_interface will run again on the same
|
||||
* (already removed) interface.
|
||||
*
|
||||
* Currently this only happens when NetworkManager creates a P2P_DEVICE
|
||||
* alongside a STA.
|
||||
*
|
||||
* But there can be other cases where this may run as well. So if that
|
||||
* happens, let's throw a warning and decrease the vifs count by one.
|
||||
*/
|
||||
if (WARN_ON(!__cw12xx_hwpriv_to_vifpriv(hw_priv, priv->if_id))) {
|
||||
if (!__cw12xx_hwpriv_to_vifpriv(hw_priv, priv->if_id)) {
|
||||
bes_devel(" !!! %s: interface addr %pM already removed\n",
|
||||
__func__, vif->addr);
|
||||
|
||||
atomic_dec(&hw_priv->num_vifs);
|
||||
|
||||
up(&hw_priv->conf_lock);
|
||||
up(&hw_priv->scan.lock);
|
||||
return;
|
||||
@@ -467,7 +451,7 @@ void bes2600_remove_interface(struct ieee80211_hw *dev,
|
||||
cancel_delayed_work_sync(&priv->pending_offchanneltx_work);
|
||||
cancel_work_sync(&priv->decrypt_storm_recover_work);
|
||||
|
||||
timer_delete_sync(&priv->mcast_timeout);
|
||||
del_timer_sync(&priv->mcast_timeout);
|
||||
/* TODO:COMBO: May be reset of these variables "delayed_link_loss and
|
||||
* join_status to default can be removed as dev_priv will be freed by
|
||||
* mac80211 */
|
||||
@@ -520,7 +504,7 @@ int bes2600_change_interface(struct ieee80211_hw *dev,
|
||||
return ret;
|
||||
}
|
||||
|
||||
int bes2600_config(struct ieee80211_hw *dev, int radio_idx, u32 changed)
|
||||
int bes2600_config(struct ieee80211_hw *dev, u32 changed)
|
||||
{
|
||||
int ret = 0;
|
||||
struct bes2600_common *hw_priv = dev->priv;
|
||||
@@ -1155,7 +1139,7 @@ void bes2600_wep_key_work(struct work_struct *work)
|
||||
wsm_unlock_tx(hw_priv);
|
||||
}
|
||||
|
||||
int bes2600_set_rts_threshold(struct ieee80211_hw *hw, int radio_idx, u32 value)
|
||||
int bes2600_set_rts_threshold(struct ieee80211_hw *hw, u32 value)
|
||||
{
|
||||
struct bes2600_common *hw_priv = hw->priv;
|
||||
int ret;
|
||||
@@ -2225,9 +2209,10 @@ void bes2600_join_work(struct work_struct *work)
|
||||
struct wsm_template_frame probe_tmp = {
|
||||
.frame_type = WSM_FRAME_TYPE_PROBE_REQUEST,
|
||||
};
|
||||
/*struct wsm_reset reset = {
|
||||
.reset_statistics = true,
|
||||
};*/
|
||||
struct wsm_reset join_fail_reset = {
|
||||
.reset_statistics = false,
|
||||
};
|
||||
bool join_failed = false;
|
||||
|
||||
|
||||
BUG_ON(queueId >= 4);
|
||||
@@ -2263,8 +2248,6 @@ void bes2600_join_work(struct work_struct *work)
|
||||
wsm_unlock_tx(hw_priv);
|
||||
return;
|
||||
}
|
||||
|
||||
rcu_read_lock();
|
||||
ssidie = ieee80211_bss_get_ie(bss, WLAN_EID_SSID);
|
||||
dtimie = ieee80211_bss_get_ie(bss, WLAN_EID_TIM);
|
||||
if (dtimie)
|
||||
@@ -2348,8 +2331,6 @@ void bes2600_join_work(struct work_struct *work)
|
||||
bes2600_rate_mask_to_wsm(hw_priv, 0xFF0);
|
||||
}
|
||||
|
||||
rcu_read_unlock();
|
||||
|
||||
bes2600_pwr_set_busy_event(hw_priv, BES_PWR_LOCK_ON_JOIN);
|
||||
wsm_flush_tx(hw_priv);
|
||||
|
||||
@@ -2410,6 +2391,33 @@ void bes2600_join_work(struct work_struct *work)
|
||||
#endif /*CONFIG_BES2600_TESTMODE*/
|
||||
cancel_delayed_work_sync(&priv->join_timeout);
|
||||
bes2600_pwr_clear_busy_event(priv->hw_priv, BES_PWR_LOCK_ON_JOIN);
|
||||
/*
|
||||
* Firmware rejected WSM_JOIN (wsm_join_confirm ret 1).
|
||||
* Issue wsm_reset so the firmware returns to a clean
|
||||
* IDLE state before the next association attempt.
|
||||
*
|
||||
* Without this reset the firmware sits in an
|
||||
* intermediate post-reject state. A rapid second
|
||||
* JOIN (e.g. wpa_supplicant retrying after the
|
||||
* PREV_AUTH_NOT_VALID deauth that follows) hits an
|
||||
* inconsistent firmware context, causing
|
||||
* bes2600_sdio_read_rx_batch to return SDIO error
|
||||
* which cascades into wifi_force_close.
|
||||
*
|
||||
* cw1200 ancestor (drivers/net/wireless/st/cw1200/
|
||||
* sta.c:1339) queues unjoin_work on join failure for
|
||||
* the same reason; bes2600_unjoin_work gates its
|
||||
* wsm_reset on join_status != PASSIVE, so after a
|
||||
* failed JOIN (join_status stays PASSIVE) that path
|
||||
* never fires — call wsm_reset directly here instead.
|
||||
*
|
||||
* Contract: wsm_reset takes only wsm_cmd_lock; safe
|
||||
* to call while conf_lock is held. wsm_oper_unlock
|
||||
* was already called in wsm_join_confirm() before
|
||||
* wsm_join() returned the error.
|
||||
*/
|
||||
WARN_ON(wsm_reset(hw_priv, &join_fail_reset, priv->if_id));
|
||||
join_failed = true;
|
||||
} else {
|
||||
/* Upload keys */
|
||||
#ifdef CONFIG_BES2600_TESTMODE
|
||||
@@ -2434,7 +2442,18 @@ void bes2600_join_work(struct work_struct *work)
|
||||
up(&hw_priv->conf_lock);
|
||||
if (bss)
|
||||
cfg80211_put_bss(hw_priv->hw->wiphy, bss);
|
||||
wsm_unlock_tx(hw_priv);
|
||||
/*
|
||||
* On join failure: queue unjoin_work so the next association
|
||||
* attempt is serialised after any lingering cleanup, matching
|
||||
* cw1200 sta.c:1344 "Tx lock still held, unjoin will clear it."
|
||||
* If unjoin_work is already queued, release TX immediately.
|
||||
*/
|
||||
if (join_failed) {
|
||||
if (queue_work(hw_priv->workqueue, &priv->unjoin_work) <= 0)
|
||||
wsm_unlock_tx(hw_priv);
|
||||
} else {
|
||||
wsm_unlock_tx(hw_priv);
|
||||
}
|
||||
}
|
||||
|
||||
void bes2600_join_timeout(struct work_struct *work)
|
||||
@@ -2461,7 +2480,7 @@ void bes2600_unjoin_work(struct work_struct *work)
|
||||
int i;
|
||||
struct bes2600_vif *tmp_priv;
|
||||
|
||||
timer_delete_sync(&hw_priv->ba_timer);
|
||||
del_timer_sync(&hw_priv->ba_timer);
|
||||
down(&hw_priv->conf_lock);
|
||||
if (unlikely(atomic_read(&hw_priv->scan.in_progress)
|
||||
|| atomic_read(&priv->connect_in_process))) {
|
||||
@@ -2672,7 +2691,7 @@ void bes2600_ba_timer(struct timer_list *t)
|
||||
{
|
||||
bool ba_ena;
|
||||
int cnt, acc, cnt_rx, acc_rx;
|
||||
struct bes2600_common *hw_priv = timer_container_of(hw_priv, t, ba_timer);
|
||||
struct bes2600_common *hw_priv = from_timer(hw_priv, t, ba_timer);
|
||||
|
||||
/*
|
||||
* Patch D: ba_lock removed. Snapshot atomic counters into locals
|
||||
@@ -2904,7 +2923,6 @@ void bes2600_dynamic_opt_txrx_work(struct work_struct *work)
|
||||
if (priv != NULL && priv->join_status > BES2600_JOIN_STATUS_MONITOR) {
|
||||
multivif_connected = true;
|
||||
}
|
||||
bes2600_txrx_opt_multivif_connected_handler(hw_priv, multivif_connected);
|
||||
}
|
||||
|
||||
|
||||
|
||||
+2
-2
@@ -26,7 +26,7 @@ int bes2600_change_interface(struct ieee80211_hw *dev,
|
||||
enum nl80211_iftype new_type,
|
||||
bool p2p);
|
||||
|
||||
int bes2600_config(struct ieee80211_hw *dev, int radio_idx, u32 changed);
|
||||
int bes2600_config(struct ieee80211_hw *dev, u32 changed);
|
||||
int bes2600_change_interface(struct ieee80211_hw *dev,
|
||||
struct ieee80211_vif *vif,
|
||||
enum nl80211_iftype new_type,
|
||||
@@ -48,7 +48,7 @@ int bes2600_set_key(struct ieee80211_hw *dev, enum set_key_cmd cmd,
|
||||
struct ieee80211_vif *vif, struct ieee80211_sta *sta,
|
||||
struct ieee80211_key_conf *key);
|
||||
|
||||
int bes2600_set_rts_threshold(struct ieee80211_hw *hw, int radio_idx, u32 value);
|
||||
int bes2600_set_rts_threshold(struct ieee80211_hw *hw, u32 value);
|
||||
|
||||
void bes2600_flush(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
u32 queues, bool drop);
|
||||
|
||||
+2
-2
@@ -109,9 +109,9 @@ void bes2600_tx_loop_set_enable(struct bes2600_common *hw_priv, bool need_warn)
|
||||
bes2600_queue_iterate_pending_packet(&hw_priv->tx_queue[i],
|
||||
bes2600_tx_loop_item_pending_item);
|
||||
}
|
||||
spin_lock(&hw_priv->tx_loop.pending_record_lock);
|
||||
spin_lock_bh(&hw_priv->tx_loop.pending_record_lock);
|
||||
bes2600_queue_iterate_record_pending_packet(hw_priv, bes2600_tx_loop_item_pending_item);
|
||||
spin_unlock(&hw_priv->tx_loop.pending_record_lock);
|
||||
spin_unlock_bh(&hw_priv->tx_loop.pending_record_lock);
|
||||
|
||||
if (atomic_read(&hw_priv->bh_rx) > 0)
|
||||
wake_up(&hw_priv->bh_wq);
|
||||
|
||||
+1
-24
@@ -21,7 +21,6 @@
|
||||
#include "debug.h"
|
||||
#include "sta.h"
|
||||
#include "sbus.h"
|
||||
#include "txrx_opt.h"
|
||||
#include "bes_log.h"
|
||||
|
||||
#define BES2600_INVALID_RATE_ID (0xFF)
|
||||
@@ -1550,35 +1549,14 @@ void bes2600_skb_dtor(struct bes2600_common *hw_priv,
|
||||
struct bes2600_vif *priv =
|
||||
__cw12xx_hwpriv_to_vifpriv(hw_priv, txpriv->if_id);
|
||||
|
||||
|
||||
if (!skb)
|
||||
return;
|
||||
|
||||
/*
|
||||
* There should be no reason for skb buffer being larger
|
||||
* than the offset..
|
||||
*/
|
||||
if(WARN_ON(txpriv->offset > skb->len)) {
|
||||
ieee80211_free_txskb(hw_priv->hw, skb);
|
||||
return;
|
||||
}
|
||||
|
||||
bes_devel("%s: txpriv->offset: %d - skb->len: %d\n",
|
||||
__func__, txpriv->offset, skb->len);
|
||||
|
||||
skb_pull(skb, txpriv->offset);
|
||||
if (priv && txpriv->rate_id != BES2600_INVALID_RATE_ID) {
|
||||
bes2600_notify_buffered_tx(priv, skb,
|
||||
txpriv->raw_link_id, txpriv->tid);
|
||||
tx_policy_put(hw_priv, txpriv->rate_id);
|
||||
}
|
||||
if (likely(!bes2600_is_itp(hw_priv))) {
|
||||
if (priv) {
|
||||
/* The interface may be already removed */
|
||||
bes2600_tx_status(priv, skb);
|
||||
}
|
||||
if (likely(!bes2600_is_itp(hw_priv)))
|
||||
ieee80211_tx_status_skb(hw_priv->hw, skb);
|
||||
}
|
||||
|
||||
}
|
||||
#ifdef CONFIG_BES2600_TESTMODE
|
||||
@@ -1948,7 +1926,6 @@ void bes2600_rx_cb(struct bes2600_vif *priv,
|
||||
|
||||
if (ieee80211_is_data(frame->frame_control)) {
|
||||
bes2600_rx_h_ba_stat(priv, hdrlen, skb->len);
|
||||
bes2600_rx_status(priv, skb);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_BES2600_TESTMODE
|
||||
|
||||
@@ -1,569 +0,0 @@
|
||||
/***************************************************************************
|
||||
*
|
||||
* Copyright 2015-2022 BES.
|
||||
* All rights reserved. All unpublished rights reserved.
|
||||
*
|
||||
* No part of this work may be used or reproduced in any form or by any
|
||||
* means, or stored in a database or retrieval system, without prior written
|
||||
* permission of BES.
|
||||
*
|
||||
* Use of this work is governed by a license granted by BES.
|
||||
* This work contains confidential and proprietary information of
|
||||
* BES. which is protected by copyright, trade secret,
|
||||
* trademark and other intellectual property rights.
|
||||
*
|
||||
****************************************************************************/
|
||||
#include <net/mac80211.h>
|
||||
#include <linux/etherdevice.h>
|
||||
#include <linux/skbuff.h>
|
||||
#include <linux/timer.h>
|
||||
|
||||
#include "bes2600.h"
|
||||
#include "wsm.h"
|
||||
#include "bh.h"
|
||||
#include "ap.h"
|
||||
#include "debug.h"
|
||||
#include "sta.h"
|
||||
#include "sbus.h"
|
||||
#include "bes_pwr.h"
|
||||
#include "txrx_opt.h"
|
||||
|
||||
#define TXRX_OPT_CLOSE_EDCA 0
|
||||
#define TXRX_OPT_EDCA_MAX_LEVEL 4
|
||||
#define TX_AVG_TIME_COUNT 10
|
||||
#define TXRX_OPT_PEROID 500
|
||||
#define TXRX_OPT_DEBUG 1
|
||||
|
||||
#define TXRX_HIGH_TP_THRESHOLD_2G4 30000 // unit is kbps
|
||||
#define TXRX_HIGH_TP_THRESHOLD_5G 40000 // unit is kbps
|
||||
#define TXRX_HIGH_TP_DELTA_TIME_2G4 8 // unit ms
|
||||
#define TXRX_HIGH_TP_DELTA_TIME_5G 6 // unit ms
|
||||
|
||||
#define TXRX_RTS_PROT_TRIG_THRESH 80 // percent * 100
|
||||
#define TXRX_RTS_PROT_DURATION 10 // unit second
|
||||
|
||||
#define TXRX_RTS_PROT_OPEN(x) (x = 512)
|
||||
#define TXRX_RTS_PROT_CLOSE(x) (x = 2437)
|
||||
#define TXRX_RTS_PROT_OPENED(x) (x < 1536)
|
||||
|
||||
static uint32_t tx_delta_time_arr[4][TX_AVG_TIME_COUNT];
|
||||
static uint32_t tx_queue_arr[4] = {0};
|
||||
static uint32_t tx_delta_time_total = 0;
|
||||
static uint32_t tx_delta_time_total_cnt = 0;
|
||||
static u8 cur_pwr_tbl = 1;
|
||||
static u16 cur_rts_thres = 2437;
|
||||
static unsigned long last_rts_set_time = -1;
|
||||
|
||||
void bes2600_add_tx_delta_time(uint32_t tx_delta_time)
|
||||
{
|
||||
tx_delta_time_total += tx_delta_time;
|
||||
tx_delta_time_total_cnt++;
|
||||
}
|
||||
|
||||
static uint32_t bes2600_get_tx_delta_time(void)
|
||||
{
|
||||
if (tx_delta_time_total_cnt != 0)
|
||||
return tx_delta_time_total / tx_delta_time_total_cnt;
|
||||
else
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void bes2600_clear_tx_delta_time(void)
|
||||
{
|
||||
tx_delta_time_total_cnt = 0;
|
||||
tx_delta_time_total = 0;
|
||||
return ;
|
||||
}
|
||||
|
||||
static uint32_t bes2600_get_tx_ac_delta_time(int ac)
|
||||
{
|
||||
uint32_t avg_time = 0;
|
||||
int i = 0;
|
||||
for (i = 0; i < TX_AVG_TIME_COUNT; i++) {
|
||||
avg_time += tx_delta_time_arr[ac][i];
|
||||
}
|
||||
return avg_time / TX_AVG_TIME_COUNT;
|
||||
}
|
||||
|
||||
static void bes2600_clear_tx_ac_delta_time(int ac)
|
||||
{
|
||||
int i = 0;
|
||||
for (i = 0; i < TX_AVG_TIME_COUNT; i++) {
|
||||
tx_delta_time_arr[ac][i] = 0;
|
||||
}
|
||||
return ;
|
||||
}
|
||||
|
||||
void bes2600_add_tx_ac_delta_time(int ac, uint32_t del_time)
|
||||
{
|
||||
#if 0
|
||||
if (tx_queue_arr[ac] >= (TX_AVG_TIME_COUNT - 1)) {
|
||||
static int num = 0;
|
||||
if ((num ++ % 10) == 0)
|
||||
bes_devel( "%s %d %d %d %d %d del=%d\n\r", __func__, tx_delta_time_arr[ac][0],
|
||||
tx_delta_time_arr[ac][2], tx_delta_time_arr[ac][4], tx_delta_time_arr[ac][6],
|
||||
tx_delta_time_arr[ac][8], del_time);
|
||||
}
|
||||
#endif
|
||||
tx_delta_time_arr[ac][tx_queue_arr[ac]] = del_time;
|
||||
tx_queue_arr[ac] = (tx_queue_arr[ac] >= (TX_AVG_TIME_COUNT - 1)) ? 0 : (tx_queue_arr[ac] + 1);
|
||||
|
||||
}
|
||||
|
||||
|
||||
static int bes2600_set_txrx_opt_param(struct bes2600_common *hw_priv,
|
||||
struct bes2600_vif *priv,
|
||||
MIB_TXRX_OPT_PARAM *para)
|
||||
{
|
||||
|
||||
int ret = 0;
|
||||
ret = WARN_ON(wsm_write_mib(hw_priv,
|
||||
WSM_MIB_ID_EXT_TXRX_OPT_PARAM,
|
||||
(u8 *)para,
|
||||
sizeof(MIB_TXRX_OPT_PARAM),
|
||||
priv->if_id));
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_enable_tx_shortgi(struct bes2600_common *hw_priv,
|
||||
struct bes2600_vif *priv,
|
||||
u8 onoff)
|
||||
{
|
||||
int ret = 0;
|
||||
static u8 en = 0xff;
|
||||
|
||||
bes_devel( "%s onoff=%d\n\r", __func__, onoff);
|
||||
|
||||
if (en != onoff) {
|
||||
en = onoff;
|
||||
ret = WARN_ON(wsm_write_mib(hw_priv,
|
||||
WSM_MIB_ID_EXT_TX_SHORT_GI_ENABLED,
|
||||
(u8 *)&onoff,
|
||||
sizeof(onoff),
|
||||
priv->if_id));
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
void bes2600_rx_status(struct bes2600_vif *priv, struct sk_buff *skb)
|
||||
{
|
||||
priv->dot11ReceivedFragmentCount++;
|
||||
priv->dot11ReceivedDataBytes += skb->len;
|
||||
}
|
||||
|
||||
void bes2600_tx_status(struct bes2600_vif *priv, struct sk_buff *skb)
|
||||
{
|
||||
struct ieee80211_hdr *hdr = (struct ieee80211_hdr *) skb->data;
|
||||
struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
|
||||
int i;
|
||||
int retry_count = -1;
|
||||
|
||||
if(WARN_ON(!priv->hw))
|
||||
return;
|
||||
|
||||
if (!ieee80211_is_data(hdr->frame_control))
|
||||
return;
|
||||
|
||||
for (i = 0; i < IEEE80211_TX_MAX_RATES; i++) {
|
||||
if (info->status.rates[i].idx < 0)
|
||||
break;
|
||||
|
||||
retry_count += info->status.rates[i].count;
|
||||
}
|
||||
if (retry_count < 0)
|
||||
retry_count = 0;
|
||||
|
||||
if (info->flags & IEEE80211_TX_STAT_ACK) {
|
||||
priv->dot11TransmittedFrameCount++;
|
||||
priv->dot11TransmittedDataBytes += (skb->len + 4);
|
||||
if (retry_count > 0)
|
||||
priv->dot11RetryCount += retry_count;
|
||||
|
||||
} else {
|
||||
/* tx fail.*/
|
||||
priv->dot11FailedCount++;
|
||||
}
|
||||
}
|
||||
void bes2600_set_default_params(struct bes2600_common *hw_priv, struct bes2600_vif *priv);
|
||||
|
||||
static int bes2600_set_high_edca_params(struct bes2600_common *hw_priv, struct bes2600_vif *priv, int level)
|
||||
{
|
||||
struct wsm_edca_params arg;
|
||||
int i = 0;
|
||||
static int lev = 0;
|
||||
bes_devel( "set edca level=%d\n\r", level);
|
||||
|
||||
if (lev == level)
|
||||
return 0;
|
||||
|
||||
lev = level;
|
||||
|
||||
memcpy(&arg, &(priv->edca), sizeof(struct wsm_edca_params));
|
||||
|
||||
if (level == 0) {
|
||||
bes2600_set_default_params(hw_priv, priv);
|
||||
return 0;
|
||||
} else if (level == 1) {
|
||||
for ( i = 0; i < 4; i++) {
|
||||
arg.params[i].aifns = 2;
|
||||
arg.params[i].cwMax = 7;
|
||||
arg.params[i].cwMin = 3;
|
||||
/*
|
||||
* tx op must set 0
|
||||
* set other, some AP may not response BA when rx data.
|
||||
*/
|
||||
arg.params[i].txOpLimit = 0;
|
||||
arg.params[i].maxReceiveLifetime = 0xc8;
|
||||
}
|
||||
} else if (level == 2) {
|
||||
for ( i = 0; i < 4; i++) {
|
||||
arg.params[i].aifns = 2;
|
||||
arg.params[i].cwMax = 5;
|
||||
arg.params[i].cwMin = 1;
|
||||
arg.params[i].txOpLimit = 0;
|
||||
arg.params[i].maxReceiveLifetime = 0xc8;
|
||||
}
|
||||
} else if (level == 3) {
|
||||
for ( i = 0; i < 4; i++) {
|
||||
arg.params[i].aifns = 2;
|
||||
arg.params[i].cwMax = 3;
|
||||
arg.params[i].cwMin = 1;
|
||||
arg.params[i].txOpLimit = 0;
|
||||
arg.params[i].maxReceiveLifetime = 0xc8;
|
||||
}
|
||||
} else if (level == 4) {
|
||||
for ( i = 0; i < 4; i++) {
|
||||
arg.params[i].aifns = 1;
|
||||
arg.params[i].cwMax = 3;
|
||||
arg.params[i].cwMin = 1;
|
||||
arg.params[i].txOpLimit = 0;
|
||||
arg.params[i].maxReceiveLifetime = 0xc8;
|
||||
}
|
||||
}
|
||||
wsm_set_edca_params(hw_priv, &arg, priv->if_id);
|
||||
return 0;
|
||||
}
|
||||
void bes2600_set_default_params(struct bes2600_common *hw_priv, struct bes2600_vif *priv)
|
||||
{
|
||||
bes_devel( "set edca default\n\r");
|
||||
wsm_set_edca_params(hw_priv, &priv->edca, priv->if_id);
|
||||
}
|
||||
static void bes2600_set_cca_method(struct bes2600_common *hw_priv, struct bes2600_vif *priv, int value)
|
||||
{
|
||||
// todo set cca alg
|
||||
}
|
||||
|
||||
static void bes2600_set_dynamic_agc(struct bes2600_common *hw_priv, struct bes2600_vif *priv, int value)
|
||||
{
|
||||
// todo set agc alg
|
||||
}
|
||||
|
||||
static int bes2600_update_pwr_table(struct bes2600_common *hw_priv,
|
||||
struct bes2600_vif *priv,
|
||||
u8 pwr_tbl_idx)
|
||||
{
|
||||
int ret = 0;
|
||||
static u8 cur_pwr_tbl_idx = 0xff;
|
||||
|
||||
if (cur_pwr_tbl_idx != pwr_tbl_idx) {
|
||||
cur_pwr_tbl_idx = pwr_tbl_idx;
|
||||
ret = WARN_ON(wsm_write_mib(hw_priv,
|
||||
WSM_MIB_ID_EXT_PWR_TBL_UPDATE,
|
||||
(u8 *)&cur_pwr_tbl_idx,
|
||||
sizeof(cur_pwr_tbl_idx),
|
||||
priv->if_id));
|
||||
bes_devel( "%s pwr_tbl_idx=%d\n\r", __func__, pwr_tbl_idx);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int bes2600_get_tx_av_max_delta_time(void)
|
||||
{
|
||||
int max_avg = 0;
|
||||
int i = 0;
|
||||
|
||||
for (i = 0; i < 4; i++) {
|
||||
if (max_avg < bes2600_get_tx_ac_delta_time(i)) {
|
||||
max_avg = bes2600_get_tx_ac_delta_time(i);
|
||||
}
|
||||
//bes2600_clear_tx_ac_delta_time(i);
|
||||
}
|
||||
return max_avg;
|
||||
}
|
||||
|
||||
static bool bes2600_station_is_ap_ht40(struct bes2600_common *hw_priv)
|
||||
{
|
||||
|
||||
if (hw_priv->hw) {
|
||||
struct ieee80211_conf *conf = &hw_priv->hw->conf;
|
||||
if (conf != NULL)
|
||||
if (conf->chandef.width == NL80211_CHAN_WIDTH_40)
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void bes2600_dynamic_opt_rxtx(struct bes2600_common *hw_priv, struct bes2600_vif *priv, int rssi)
|
||||
{
|
||||
u32 succPro = 0, tx_cnt, tx_retry, rx_cnt, tx_fail;
|
||||
static u32 l_tx_cnt = 0, l_tx_fail = 0, l_tx_retry = 0, l_rx_cnt = 0;
|
||||
static u32 tx_bps = 0, rx_bps = 0;
|
||||
u32 total_kbps = 0;
|
||||
static int level;
|
||||
|
||||
/* calculate real time throughput */
|
||||
if (hw_priv == NULL || priv == NULL) {
|
||||
return;
|
||||
}
|
||||
tx_bps = abs (priv->dot11TransmittedDataBytes - tx_bps);
|
||||
rx_bps = abs (priv->dot11ReceivedDataBytes - rx_bps);
|
||||
total_kbps = (tx_bps / 128 + rx_bps / 128);
|
||||
|
||||
total_kbps *= 1000;
|
||||
total_kbps /= TXRX_OPT_PEROID;
|
||||
|
||||
/* if tx/rx < 100k/s, close*/
|
||||
if (total_kbps < 100) {
|
||||
level = 0;
|
||||
last_rts_set_time = -1;
|
||||
TXRX_RTS_PROT_CLOSE(cur_rts_thres);
|
||||
goto txrx_opt_clear;
|
||||
}
|
||||
|
||||
/* calculate tx_cnt, tx_retry, rx_cnt */
|
||||
tx_cnt = (priv->dot11TransmittedFrameCount - l_tx_cnt);
|
||||
tx_fail = (priv->dot11FailedCount - l_tx_fail);
|
||||
tx_retry = (priv->dot11RetryCount - l_tx_retry);
|
||||
rx_cnt = (priv->dot11ReceivedFragmentCount - l_rx_cnt);
|
||||
( (tx_cnt + tx_retry) > 0 ) ? (succPro = tx_cnt * 100 / (tx_cnt + tx_retry)) : (succPro = 0);
|
||||
|
||||
bes_devel( "%s, tx_cnt:%d prob:%d\n", __func__, tx_cnt, succPro);
|
||||
|
||||
/* set rts/cts protection dynamically */
|
||||
if (tx_cnt > 50 && succPro != 0) {
|
||||
if (succPro > TXRX_RTS_PROT_TRIG_THRESH &&
|
||||
TXRX_RTS_PROT_OPENED(cur_rts_thres) &&
|
||||
time_after(jiffies, last_rts_set_time + TXRX_RTS_PROT_DURATION * HZ)) {
|
||||
TXRX_RTS_PROT_CLOSE(cur_rts_thres);
|
||||
} else if (succPro <= TXRX_RTS_PROT_TRIG_THRESH){
|
||||
TXRX_RTS_PROT_OPEN(cur_rts_thres);
|
||||
last_rts_set_time = jiffies;
|
||||
}
|
||||
}
|
||||
|
||||
/* dynamic set edca param */
|
||||
if (succPro != 0) {
|
||||
if (bes2600_station_is_ap_ht40(hw_priv)) {
|
||||
if (bes2600_get_tx_delta_time() > 8 || bes2600_get_tx_av_max_delta_time() > 8) {
|
||||
if (level < 4)
|
||||
level++;
|
||||
} else {
|
||||
if (level > 0)
|
||||
level--;
|
||||
}
|
||||
/* high throughput force level = 0 */
|
||||
if (total_kbps > TXRX_HIGH_TP_THRESHOLD_5G && level > 0 && priv->hw_value > 19) {
|
||||
level = 0;
|
||||
}
|
||||
|
||||
} else {//shiled room 13, office 8
|
||||
if (bes2600_get_tx_delta_time() > (TXRX_HIGH_TP_DELTA_TIME_5G + total_kbps / 8000)
|
||||
|| bes2600_get_tx_av_max_delta_time() > (TXRX_HIGH_TP_DELTA_TIME_5G + total_kbps / 8000)) {
|
||||
if (level < 4)
|
||||
level++;
|
||||
} else {
|
||||
if (level > 0)
|
||||
level--;
|
||||
}
|
||||
/* high throughput force level = 0 */
|
||||
if (total_kbps > TXRX_HIGH_TP_THRESHOLD_2G4 && level > 0) {
|
||||
level = 0;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
/* dynamic set power table */
|
||||
if (rssi <= BES2600_TX_RSSI_LOW)
|
||||
cur_pwr_tbl = 2; // use high power table
|
||||
else if(rssi >= BES2600_TX_RSSI_HIGH)
|
||||
cur_pwr_tbl = 1; // use standard power table
|
||||
|
||||
#if TXRX_OPT_CLOSE_EDCA
|
||||
level = 0;
|
||||
#endif
|
||||
if (level > TXRX_OPT_EDCA_MAX_LEVEL)
|
||||
level = TXRX_OPT_EDCA_MAX_LEVEL;
|
||||
|
||||
bes_devel( "txrx_opt: tx(cnt=%d retry=%d psr=%d tx_fail=%d (wsm level=%d) tx=%dk/s)\n\r",
|
||||
tx_cnt, tx_retry, succPro, tx_fail, level, tx_bps / 128);
|
||||
bes_devel( "txrx_opt: rx(cnt=%d rx=%dk/s) total=%dk/s\n\r", rx_cnt, rx_bps / 128, total_kbps);
|
||||
bes_devel( "txrx_opt: tx_delta_time=%d [%d %d %d %d] hw_value=%d ht=%d maxtxcnt=%d\n\r",
|
||||
bes2600_get_tx_delta_time(), bes2600_get_tx_ac_delta_time(0), bes2600_get_tx_ac_delta_time(1),
|
||||
bes2600_get_tx_ac_delta_time(2), bes2600_get_tx_ac_delta_time(3), priv->hw_value,
|
||||
bes2600_station_is_ap_ht40(hw_priv), hw_priv->long_frame_max_tx_count);
|
||||
|
||||
|
||||
/* dynamic set cca */
|
||||
bes2600_set_cca_method(hw_priv, priv, 0);
|
||||
/* dynamic set agc */
|
||||
bes2600_set_dynamic_agc(hw_priv, priv, 0);
|
||||
bes2600_update_pwr_table(hw_priv, priv, cur_pwr_tbl);
|
||||
txrx_opt_clear:
|
||||
bes2600_set_high_edca_params(hw_priv, priv, level);
|
||||
bes2600_set_rts_threshold(hw_priv->hw, -1, cur_rts_thres);
|
||||
bes2600_clear_tx_delta_time();
|
||||
bes2600_clear_tx_ac_delta_time(0);
|
||||
bes2600_clear_tx_ac_delta_time(1);
|
||||
bes2600_clear_tx_ac_delta_time(2);
|
||||
bes2600_clear_tx_ac_delta_time(3);
|
||||
tx_bps = priv->dot11TransmittedDataBytes;
|
||||
rx_bps = priv->dot11ReceivedDataBytes;
|
||||
l_tx_cnt = priv->dot11TransmittedFrameCount;
|
||||
l_tx_fail = priv->dot11FailedCount;
|
||||
l_tx_retry = priv->dot11RetryCount;
|
||||
l_rx_cnt = priv->dot11ReceivedFragmentCount;
|
||||
return ;
|
||||
}
|
||||
|
||||
static struct bes2600_common *txrx_hw_priv = NULL;
|
||||
|
||||
static bool bes2600_is_sta_connected(void)
|
||||
{
|
||||
if (txrx_hw_priv == NULL)
|
||||
return false;
|
||||
else
|
||||
return true;
|
||||
}
|
||||
|
||||
void bes2600_txrx_opt_timer_restore(void)
|
||||
{
|
||||
if (bes2600_is_sta_connected()) {
|
||||
mod_timer(&txrx_hw_priv->txrx_opt_timer, jiffies + msecs_to_jiffies(TXRX_OPT_PEROID));
|
||||
}
|
||||
}
|
||||
|
||||
static void txrx_opt_timer_callback(struct timer_list* data)
|
||||
{
|
||||
bes_devel( "####Timer callback function Called time = %lu\n", jiffies);
|
||||
queue_work(txrx_hw_priv->workqueue, &txrx_hw_priv->dynamic_opt_txrx_work);
|
||||
}
|
||||
|
||||
static void txrx_opt_timer_start(struct bes2600_common *hw_priv)
|
||||
{
|
||||
mod_timer(&hw_priv->txrx_opt_timer, jiffies + msecs_to_jiffies(TXRX_OPT_PEROID));
|
||||
}
|
||||
|
||||
static void txrx_opt_timer_stop(struct bes2600_common *hw_priv)
|
||||
{
|
||||
timer_delete_sync(&hw_priv->txrx_opt_timer);
|
||||
}
|
||||
|
||||
static void bes2600_set_txrx_opt_default_param(struct bes2600_common * hw_priv)
|
||||
{
|
||||
MIB_TXRX_OPT_PARAM g_txrx_param = {2, (PROCTECT_MODE_RTS_CTS | PROCTECT_MODE_RTS_CTS_RETRY), 3000};
|
||||
struct bes2600_vif *priv = __cw12xx_hwpriv_to_vifpriv(hw_priv, 0);
|
||||
struct ieee80211_sta *sta = NULL;
|
||||
int shortgi = 0;
|
||||
|
||||
if (priv == NULL)
|
||||
return;
|
||||
|
||||
/* reset states */
|
||||
cur_pwr_tbl = 1;
|
||||
TXRX_RTS_PROT_CLOSE(cur_rts_thres);
|
||||
last_rts_set_time = -1;
|
||||
memcpy(&hw_priv->txrx_opt_param, &g_txrx_param, sizeof(MIB_TXRX_OPT_PARAM));
|
||||
|
||||
/* reset device states */
|
||||
bes2600_set_txrx_opt_param(hw_priv, priv, &hw_priv->txrx_opt_param);
|
||||
bes2600_set_rts_threshold(hw_priv->hw, -1, cur_rts_thres); // close rts/cts
|
||||
bes2600_update_pwr_table(hw_priv, priv, cur_pwr_tbl); // use standard pwr table
|
||||
|
||||
if (priv->join_status == BES2600_JOIN_STATUS_STA) {
|
||||
rcu_read_lock();
|
||||
|
||||
sta = ieee80211_find_sta(priv->vif, priv->vif->bss_conf.bssid);
|
||||
if (sta) {
|
||||
if (sta->deflink.ht_cap.ht_supported &&
|
||||
((priv->vif->bss_conf.chanreq.oper.width == NL80211_CHAN_WIDTH_20 &&
|
||||
sta->deflink.ht_cap.cap & IEEE80211_HT_CAP_SGI_20) ||
|
||||
(priv->vif->bss_conf.chanreq.oper.width == NL80211_CHAN_WIDTH_40 &&
|
||||
sta->deflink.ht_cap.cap & IEEE80211_HT_CAP_SGI_40))) {
|
||||
shortgi = 1;
|
||||
}
|
||||
}
|
||||
|
||||
rcu_read_unlock();
|
||||
|
||||
bes_devel("short gi tx status: %d\n", shortgi);
|
||||
bes2600_enable_tx_shortgi(hw_priv, priv, shortgi);
|
||||
}
|
||||
}
|
||||
|
||||
static int bes2600_set_txrx_opt_unjoin_param(struct bes2600_common * hw_priv)
|
||||
{
|
||||
MIB_TXRX_OPT_PARAM g_txrx_param = {1, 0, 2002};
|
||||
struct bes2600_vif *priv = __cw12xx_hwpriv_to_vifpriv(hw_priv, 0);
|
||||
if (priv == NULL)
|
||||
return 0;
|
||||
|
||||
/* reset states */
|
||||
cur_pwr_tbl = 1;
|
||||
bes2600_update_pwr_table(hw_priv, priv, cur_pwr_tbl);
|
||||
memcpy(&hw_priv->txrx_opt_param, &g_txrx_param, sizeof(MIB_TXRX_OPT_PARAM));
|
||||
bes2600_set_txrx_opt_param(hw_priv, priv, &hw_priv->txrx_opt_param);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void bes2600_txrx_opt_multivif_connected_handler(struct bes2600_common *hw_priv, bool multivif_connected)
|
||||
{
|
||||
struct bes2600_vif *priv = __cw12xx_hwpriv_to_vifpriv(hw_priv, 0);
|
||||
|
||||
if (multivif_connected) {
|
||||
bes2600_set_txrx_opt_default_param(hw_priv);
|
||||
} else {
|
||||
bes_devel("%s, rssi:%d\n", __func__, priv->signal);
|
||||
bes2600_dynamic_opt_rxtx(hw_priv, priv, priv->signal);
|
||||
mod_timer(&hw_priv->txrx_opt_timer, jiffies + msecs_to_jiffies(TXRX_OPT_PEROID));
|
||||
}
|
||||
}
|
||||
|
||||
int txrx_opt_timer_init(struct bes2600_vif *priv)
|
||||
{
|
||||
struct bes2600_common *hw_priv = cw12xx_vifpriv_to_hwpriv(priv);
|
||||
bes_devel( "txrx_opt_timer_init:%p", txrx_hw_priv);
|
||||
if (priv->if_id != 0)
|
||||
return 0;
|
||||
|
||||
if (!txrx_hw_priv) {
|
||||
txrx_hw_priv = hw_priv;
|
||||
bes_devel( "####Timer init hw_priv = %p\n", txrx_hw_priv);
|
||||
timer_setup(&hw_priv->txrx_opt_timer, txrx_opt_timer_callback, 0);
|
||||
bes2600_set_txrx_opt_default_param(hw_priv);
|
||||
}
|
||||
|
||||
mod_timer(&hw_priv->txrx_opt_timer, jiffies + msecs_to_jiffies(TXRX_OPT_PEROID));
|
||||
bes2600_pwr_register_en_lp_cb(hw_priv, txrx_opt_timer_stop);
|
||||
bes2600_pwr_register_exit_lp_cb(hw_priv, txrx_opt_timer_start);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void txrx_opt_timer_exit(struct bes2600_vif *priv)
|
||||
{
|
||||
struct bes2600_common *hw_priv = cw12xx_vifpriv_to_hwpriv(priv);
|
||||
bes_devel( "txrx_opt_timer_exit");
|
||||
|
||||
if (priv->if_id == 0) {
|
||||
timer_delete_sync(&hw_priv->txrx_opt_timer);
|
||||
cancel_work_sync(&hw_priv->dynamic_opt_txrx_work);
|
||||
bes2600_pwr_unregister_en_lp_cb(hw_priv, txrx_opt_timer_stop);
|
||||
bes2600_pwr_unregister_exit_lp_cb(hw_priv, txrx_opt_timer_start);
|
||||
txrx_hw_priv = NULL;
|
||||
bes2600_set_txrx_opt_unjoin_param(hw_priv);
|
||||
} else if (priv->if_id == 1) {
|
||||
bes2600_txrx_opt_timer_restore();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,38 +0,0 @@
|
||||
|
||||
/***************************************************************************
|
||||
*
|
||||
* Copyright 2015-2022 BES.
|
||||
* All rights reserved. All unpublished rights reserved.
|
||||
*
|
||||
* No part of this work may be used or reproduced in any form or by any
|
||||
* means, or stored in a database or retrieval system, without prior written
|
||||
* permission of BES.
|
||||
*
|
||||
* Use of this work is governed by a license granted by BES.
|
||||
* This work contains confidential and proprietary information of
|
||||
* BES. which is protected by copyright, trade secret,
|
||||
* trademark and other intellectual property rights.
|
||||
*
|
||||
****************************************************************************/
|
||||
#ifndef bes2600_TXRX_OPT_H
|
||||
#define bes2600_TXRX_OPT_H
|
||||
#include <linux/list.h>
|
||||
/* open it for enhance wifi throughput */
|
||||
#define BES2600_TX_RX_OPT 1
|
||||
|
||||
/* Threshold for powrt table switch */
|
||||
#define BES2600_TX_RSSI_LOW -65
|
||||
#define BES2600_TX_RSSI_HIGH -60
|
||||
|
||||
void bes2600_add_tx_ac_delta_time(int ac, uint32_t del_time);
|
||||
void bes2600_add_tx_delta_time(uint32_t tx_time);
|
||||
void bes2600_rx_status(struct bes2600_vif *priv, struct sk_buff *skb);
|
||||
void bes2600_tx_status(struct bes2600_vif *priv, struct sk_buff *skb);
|
||||
void bes2600_dynamic_opt_rxtx(struct bes2600_common *hw_priv,struct bes2600_vif *priv, int rssi);
|
||||
void bes2600_txrx_opt_multivif_connected_handler(struct bes2600_common *hw_priv, bool multivif_connected);
|
||||
void bes2600_txrx_opt_timer_restore(void);
|
||||
int txrx_opt_timer_init(struct bes2600_vif *priv);
|
||||
void txrx_opt_timer_exit(struct bes2600_vif *priv);
|
||||
|
||||
#endif
|
||||
|
||||
Reference in New Issue
Block a user