2019-05-30 07:57:47 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2015-08-11 05:24:17 +08:00
|
|
|
/*
|
|
|
|
* Bluetooth Software UART Qualcomm protocol
|
|
|
|
*
|
|
|
|
* HCI_IBS (HCI In-Band Sleep) is Qualcomm's power management
|
|
|
|
* protocol extension to H4.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007 Texas Instruments, Inc.
|
2018-08-03 20:16:32 +08:00
|
|
|
* Copyright (c) 2010, 2012, 2018 The Linux Foundation. All rights reserved.
|
2015-08-11 05:24:17 +08:00
|
|
|
*
|
|
|
|
* Acknowledgements:
|
|
|
|
* This file is based on hci_ll.c, which was...
|
|
|
|
* Written by Ohad Ben-Cohen <ohad@bencohen.org>
|
|
|
|
* which was in turn based on hci_h4.c, which was written
|
|
|
|
* by Maxim Krasnyansky and Marcel Holtmann.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/kernel.h>
|
2018-03-30 03:15:24 +08:00
|
|
|
#include <linux/clk.h>
|
2019-05-22 03:53:07 +08:00
|
|
|
#include <linux/completion.h>
|
2015-08-11 05:24:17 +08:00
|
|
|
#include <linux/debugfs.h>
|
2018-08-03 20:16:32 +08:00
|
|
|
#include <linux/delay.h>
|
2020-01-02 22:49:11 +08:00
|
|
|
#include <linux/devcoredump.h>
|
2018-08-03 20:16:32 +08:00
|
|
|
#include <linux/device.h>
|
2018-03-30 03:15:24 +08:00
|
|
|
#include <linux/gpio/consumer.h>
|
|
|
|
#include <linux/mod_devicetable.h>
|
|
|
|
#include <linux/module.h>
|
2018-08-03 20:16:32 +08:00
|
|
|
#include <linux/of_device.h>
|
2020-03-25 10:26:37 +08:00
|
|
|
#include <linux/acpi.h>
|
2018-08-03 20:16:32 +08:00
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/regulator/consumer.h>
|
2018-03-30 03:15:24 +08:00
|
|
|
#include <linux/serdev.h>
|
2020-02-15 00:17:15 +08:00
|
|
|
#include <linux/mutex.h>
|
2018-10-16 22:21:35 +08:00
|
|
|
#include <asm/unaligned.h>
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
#include <net/bluetooth/bluetooth.h>
|
|
|
|
#include <net/bluetooth/hci_core.h>
|
|
|
|
|
|
|
|
#include "hci_uart.h"
|
|
|
|
#include "btqca.h"
|
|
|
|
|
|
|
|
/* HCI_IBS protocol messages */
|
|
|
|
#define HCI_IBS_SLEEP_IND 0xFE
|
|
|
|
#define HCI_IBS_WAKE_IND 0xFD
|
|
|
|
#define HCI_IBS_WAKE_ACK 0xFC
|
2015-08-31 05:05:32 +08:00
|
|
|
#define HCI_MAX_IBS_SIZE 10
|
2015-08-11 05:24:17 +08:00
|
|
|
|
2015-08-31 05:05:32 +08:00
|
|
|
#define IBS_WAKE_RETRANS_TIMEOUT_MS 100
|
2020-06-12 20:21:31 +08:00
|
|
|
#define IBS_BTSOC_TX_IDLE_TIMEOUT_MS 200
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
#define IBS_HOST_TX_IDLE_TIMEOUT_MS 2000
|
2019-02-28 07:52:23 +08:00
|
|
|
#define CMD_TRANS_TIMEOUT_MS 100
|
2020-01-02 22:49:11 +08:00
|
|
|
#define MEMDUMP_TIMEOUT_MS 8000
|
2020-12-31 01:17:08 +08:00
|
|
|
#define IBS_DISABLE_SSR_TIMEOUT_MS \
|
|
|
|
(MEMDUMP_TIMEOUT_MS + FW_DOWNLOAD_TIMEOUT_MS)
|
2020-10-06 23:20:21 +08:00
|
|
|
#define FW_DOWNLOAD_TIMEOUT_MS 3000
|
2015-08-11 05:24:17 +08:00
|
|
|
|
2018-03-30 03:15:24 +08:00
|
|
|
/* susclk rate */
|
|
|
|
#define SUSCLK_RATE_32KHZ 32768
|
|
|
|
|
2018-10-16 22:21:35 +08:00
|
|
|
/* Controller debug log header */
|
|
|
|
#define QCA_DEBUG_HANDLE 0x2EDC
|
|
|
|
|
2020-01-15 16:55:51 +08:00
|
|
|
/* max retry count when init fails */
|
|
|
|
#define MAX_INIT_RETRIES 3
|
|
|
|
|
2020-01-02 22:49:11 +08:00
|
|
|
/* Controller dump header */
|
|
|
|
#define QCA_SSR_DUMP_HANDLE 0x0108
|
|
|
|
#define QCA_DUMP_PACKET_SIZE 255
|
|
|
|
#define QCA_LAST_SEQUENCE_NUM 0xFFFF
|
|
|
|
#define QCA_CRASHBYTE_PACKET_LEN 1096
|
|
|
|
#define QCA_MEMDUMP_BYTE 0xFB
|
|
|
|
|
2019-04-30 07:21:30 +08:00
|
|
|
enum qca_flags {
|
2020-10-06 23:20:21 +08:00
|
|
|
QCA_IBS_DISABLED,
|
2019-05-22 03:53:07 +08:00
|
|
|
QCA_DROP_VENDOR_EVENT,
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
QCA_SUSPENDING,
|
2020-02-15 00:17:15 +08:00
|
|
|
QCA_MEMDUMP_COLLECTION,
|
2020-07-11 19:31:12 +08:00
|
|
|
QCA_HW_ERROR_EVENT,
|
2020-10-06 23:20:21 +08:00
|
|
|
QCA_SSR_TRIGGERED,
|
2021-02-05 23:37:16 +08:00
|
|
|
QCA_BT_OFF,
|
|
|
|
QCA_ROM_FW
|
2019-04-30 07:21:30 +08:00
|
|
|
};
|
|
|
|
|
2020-05-15 04:14:04 +08:00
|
|
|
enum qca_capabilities {
|
|
|
|
QCA_CAP_WIDEBAND_SPEECH = BIT(0),
|
2020-10-01 04:01:38 +08:00
|
|
|
QCA_CAP_VALID_LE_STATES = BIT(1),
|
2020-05-15 04:14:04 +08:00
|
|
|
};
|
2020-01-02 22:49:11 +08:00
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
/* HCI_IBS transmit side sleep protocol states */
|
|
|
|
enum tx_ibs_states {
|
|
|
|
HCI_IBS_TX_ASLEEP,
|
|
|
|
HCI_IBS_TX_WAKING,
|
|
|
|
HCI_IBS_TX_AWAKE,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* HCI_IBS receive side sleep protocol states */
|
|
|
|
enum rx_states {
|
|
|
|
HCI_IBS_RX_ASLEEP,
|
|
|
|
HCI_IBS_RX_AWAKE,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* HCI_IBS transmit and receive side clock state vote */
|
|
|
|
enum hci_ibs_clock_state_vote {
|
|
|
|
HCI_IBS_VOTE_STATS_UPDATE,
|
|
|
|
HCI_IBS_TX_VOTE_CLOCK_ON,
|
|
|
|
HCI_IBS_TX_VOTE_CLOCK_OFF,
|
|
|
|
HCI_IBS_RX_VOTE_CLOCK_ON,
|
|
|
|
HCI_IBS_RX_VOTE_CLOCK_OFF,
|
|
|
|
};
|
|
|
|
|
2020-01-02 22:49:11 +08:00
|
|
|
/* Controller memory dump states */
|
|
|
|
enum qca_memdump_states {
|
|
|
|
QCA_MEMDUMP_IDLE,
|
|
|
|
QCA_MEMDUMP_COLLECTING,
|
|
|
|
QCA_MEMDUMP_COLLECTED,
|
|
|
|
QCA_MEMDUMP_TIMEOUT,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct qca_memdump_data {
|
|
|
|
char *memdump_buf_head;
|
|
|
|
char *memdump_buf_tail;
|
|
|
|
u32 current_seq_no;
|
|
|
|
u32 received_dump;
|
2020-05-29 22:38:31 +08:00
|
|
|
u32 ram_dump_size;
|
2020-01-02 22:49:11 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct qca_memdump_event_hdr {
|
|
|
|
__u8 evt;
|
|
|
|
__u8 plen;
|
|
|
|
__u16 opcode;
|
|
|
|
__u16 seq_no;
|
|
|
|
__u8 reserved;
|
|
|
|
} __packed;
|
|
|
|
|
|
|
|
|
|
|
|
struct qca_dump_size {
|
|
|
|
u32 dump_size;
|
|
|
|
} __packed;
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
struct qca_data {
|
|
|
|
struct hci_uart *hu;
|
|
|
|
struct sk_buff *rx_skb;
|
|
|
|
struct sk_buff_head txq;
|
|
|
|
struct sk_buff_head tx_wait_q; /* HCI_IBS wait queue */
|
2020-01-02 22:49:11 +08:00
|
|
|
struct sk_buff_head rx_memdump_q; /* Memdump wait queue */
|
2015-08-11 05:24:17 +08:00
|
|
|
spinlock_t hci_ibs_lock; /* HCI_IBS state lock */
|
|
|
|
u8 tx_ibs_state; /* HCI_IBS transmit side power state*/
|
|
|
|
u8 rx_ibs_state; /* HCI_IBS receive side power state */
|
2015-09-27 06:04:07 +08:00
|
|
|
bool tx_vote; /* Clock must be on for TX */
|
|
|
|
bool rx_vote; /* Clock must be on for RX */
|
2015-08-11 05:24:17 +08:00
|
|
|
struct timer_list tx_idle_timer;
|
|
|
|
u32 tx_idle_delay;
|
|
|
|
struct timer_list wake_retrans_timer;
|
|
|
|
u32 wake_retrans;
|
|
|
|
struct workqueue_struct *workqueue;
|
|
|
|
struct work_struct ws_awake_rx;
|
|
|
|
struct work_struct ws_awake_device;
|
|
|
|
struct work_struct ws_rx_vote_off;
|
|
|
|
struct work_struct ws_tx_vote_off;
|
2020-01-02 22:49:11 +08:00
|
|
|
struct work_struct ctrl_memdump_evt;
|
2020-02-15 00:17:15 +08:00
|
|
|
struct delayed_work ctrl_memdump_timeout;
|
2020-01-02 22:49:11 +08:00
|
|
|
struct qca_memdump_data *qca_memdump;
|
2015-08-11 05:24:17 +08:00
|
|
|
unsigned long flags;
|
2019-05-22 03:53:07 +08:00
|
|
|
struct completion drop_ev_comp;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
wait_queue_head_t suspend_wait_q;
|
2020-01-02 22:49:11 +08:00
|
|
|
enum qca_memdump_states memdump_state;
|
2020-02-15 00:17:15 +08:00
|
|
|
struct mutex hci_memdump_lock;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
/* For debugging purpose */
|
|
|
|
u64 ibs_sent_wacks;
|
|
|
|
u64 ibs_sent_slps;
|
|
|
|
u64 ibs_sent_wakes;
|
|
|
|
u64 ibs_recv_wacks;
|
|
|
|
u64 ibs_recv_slps;
|
|
|
|
u64 ibs_recv_wakes;
|
|
|
|
u64 vote_last_jif;
|
|
|
|
u32 vote_on_ms;
|
|
|
|
u32 vote_off_ms;
|
|
|
|
u64 tx_votes_on;
|
|
|
|
u64 rx_votes_on;
|
|
|
|
u64 tx_votes_off;
|
|
|
|
u64 rx_votes_off;
|
|
|
|
u64 votes_on;
|
|
|
|
u64 votes_off;
|
|
|
|
};
|
|
|
|
|
2018-08-03 20:16:29 +08:00
|
|
|
enum qca_speed_type {
|
|
|
|
QCA_INIT_SPEED = 1,
|
|
|
|
QCA_OPER_SPEED
|
|
|
|
};
|
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
/*
|
|
|
|
* Voltage regulator information required for configuring the
|
|
|
|
* QCA Bluetooth chipset
|
|
|
|
*/
|
|
|
|
struct qca_vreg {
|
|
|
|
const char *name;
|
|
|
|
unsigned int load_uA;
|
|
|
|
};
|
|
|
|
|
2020-05-15 04:14:04 +08:00
|
|
|
struct qca_device_data {
|
2018-08-03 20:16:32 +08:00
|
|
|
enum qca_btsoc_type soc_type;
|
|
|
|
struct qca_vreg *vregs;
|
|
|
|
size_t num_vregs;
|
2020-05-15 04:14:04 +08:00
|
|
|
uint32_t capabilities;
|
2018-08-03 20:16:32 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Platform data for the QCA Bluetooth power driver.
|
|
|
|
*/
|
|
|
|
struct qca_power {
|
|
|
|
struct device *dev;
|
|
|
|
struct regulator_bulk_data *vreg_bulk;
|
2019-10-18 13:24:03 +08:00
|
|
|
int num_vregs;
|
2018-08-03 20:16:32 +08:00
|
|
|
bool vregs_on;
|
|
|
|
};
|
|
|
|
|
2018-03-30 03:15:24 +08:00
|
|
|
struct qca_serdev {
|
|
|
|
struct hci_uart serdev_hu;
|
|
|
|
struct gpio_desc *bt_en;
|
2021-05-19 00:34:42 +08:00
|
|
|
struct gpio_desc *sw_ctrl;
|
2018-03-30 03:15:24 +08:00
|
|
|
struct clk *susclk;
|
2018-08-03 20:16:32 +08:00
|
|
|
enum qca_btsoc_type btsoc_type;
|
|
|
|
struct qca_power *bt_power;
|
|
|
|
u32 init_speed;
|
|
|
|
u32 oper_speed;
|
2019-06-06 17:40:30 +08:00
|
|
|
const char *firmware_name;
|
2018-03-30 03:15:24 +08:00
|
|
|
};
|
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
static int qca_regulator_enable(struct qca_serdev *qcadev);
|
|
|
|
static void qca_regulator_disable(struct qca_serdev *qcadev);
|
2018-08-22 20:20:05 +08:00
|
|
|
static void qca_power_shutdown(struct hci_uart *hu);
|
2018-09-24 22:44:45 +08:00
|
|
|
static int qca_power_off(struct hci_dev *hdev);
|
2020-01-02 22:49:11 +08:00
|
|
|
static void qca_controller_memdump(struct work_struct *work);
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-03-12 02:38:31 +08:00
|
|
|
static enum qca_btsoc_type qca_soc_type(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
enum qca_btsoc_type soc_type;
|
|
|
|
|
|
|
|
if (hu->serdev) {
|
|
|
|
struct qca_serdev *qsd = serdev_device_get_drvdata(hu->serdev);
|
|
|
|
|
|
|
|
soc_type = qsd->btsoc_type;
|
|
|
|
} else {
|
|
|
|
soc_type = QCA_ROME;
|
|
|
|
}
|
|
|
|
|
|
|
|
return soc_type;
|
|
|
|
}
|
|
|
|
|
2019-06-06 17:40:30 +08:00
|
|
|
static const char *qca_get_firmware_name(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
if (hu->serdev) {
|
|
|
|
struct qca_serdev *qsd = serdev_device_get_drvdata(hu->serdev);
|
|
|
|
|
|
|
|
return qsd->firmware_name;
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
static void __serial_clock_on(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
/* TODO: Some chipset requires to enable UART clock on client
|
|
|
|
* side to save power consumption or manual work is required.
|
|
|
|
* Please put your code to control UART clock here if needed
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __serial_clock_off(struct tty_struct *tty)
|
|
|
|
{
|
|
|
|
/* TODO: Some chipset requires to disable UART clock on client
|
|
|
|
* side to save power consumption or manual work is required.
|
|
|
|
* Please put your code to control UART clock off here if needed
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
/* serial_clock_vote needs to be called with the ibs lock held */
|
|
|
|
static void serial_clock_vote(unsigned long vote, struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
unsigned int diff;
|
|
|
|
|
|
|
|
bool old_vote = (qca->tx_vote | qca->rx_vote);
|
|
|
|
bool new_vote;
|
|
|
|
|
|
|
|
switch (vote) {
|
|
|
|
case HCI_IBS_VOTE_STATS_UPDATE:
|
|
|
|
diff = jiffies_to_msecs(jiffies - qca->vote_last_jif);
|
|
|
|
|
|
|
|
if (old_vote)
|
|
|
|
qca->vote_off_ms += diff;
|
|
|
|
else
|
|
|
|
qca->vote_on_ms += diff;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_VOTE_CLOCK_ON:
|
|
|
|
qca->tx_vote = true;
|
|
|
|
qca->tx_votes_on++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_RX_VOTE_CLOCK_ON:
|
|
|
|
qca->rx_vote = true;
|
|
|
|
qca->rx_votes_on++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_VOTE_CLOCK_OFF:
|
|
|
|
qca->tx_vote = false;
|
|
|
|
qca->tx_votes_off++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_RX_VOTE_CLOCK_OFF:
|
|
|
|
qca->rx_vote = false;
|
|
|
|
qca->rx_votes_off++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BT_ERR("Voting irregularity");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-07 00:53:06 +08:00
|
|
|
new_vote = qca->rx_vote | qca->tx_vote;
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
if (new_vote != old_vote) {
|
|
|
|
if (new_vote)
|
|
|
|
__serial_clock_on(hu->tty);
|
|
|
|
else
|
|
|
|
__serial_clock_off(hu->tty);
|
|
|
|
|
2015-09-15 20:19:45 +08:00
|
|
|
BT_DBG("Vote serial clock %s(%s)", new_vote ? "true" : "false",
|
|
|
|
vote ? "true" : "false");
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
diff = jiffies_to_msecs(jiffies - qca->vote_last_jif);
|
|
|
|
|
|
|
|
if (new_vote) {
|
|
|
|
qca->votes_on++;
|
|
|
|
qca->vote_off_ms += diff;
|
|
|
|
} else {
|
|
|
|
qca->votes_off++;
|
|
|
|
qca->vote_on_ms += diff;
|
|
|
|
}
|
|
|
|
qca->vote_last_jif = jiffies;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Builds and sends an HCI_IBS command packet.
|
|
|
|
* These are very simple packets with only 1 cmd byte.
|
|
|
|
*/
|
|
|
|
static int send_hci_ibs_cmd(u8 cmd, struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
struct sk_buff *skb = NULL;
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
BT_DBG("hu %p send hci ibs cmd 0x%x", hu, cmd);
|
|
|
|
|
|
|
|
skb = bt_skb_alloc(1, GFP_ATOMIC);
|
|
|
|
if (!skb) {
|
|
|
|
BT_ERR("Failed to allocate memory for HCI_IBS packet");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assign HCI_IBS type */
|
networking: add and use skb_put_u8()
Joe and Bjørn suggested that it'd be nicer to not have the
cast in the fairly common case of doing
*(u8 *)skb_put(skb, 1) = c;
Add skb_put_u8() for this case, and use it across the code,
using the following spatch:
@@
expression SKB, C, S;
typedef u8;
identifier fn = {skb_put};
fresh identifier fn2 = fn ## "_u8";
@@
- *(u8 *)fn(SKB, S) = C;
+ fn2(SKB, C);
Note that due to the "S", the spatch isn't perfect, it should
have checked that S is 1, but there's also places that use a
sizeof expression like sizeof(var) or sizeof(u8) etc. Turns
out that nobody ever did something like
*(u8 *)skb_put(skb, 2) = c;
which would be wrong anyway since the second byte wouldn't be
initialized.
Suggested-by: Joe Perches <joe@perches.com>
Suggested-by: Bjørn Mork <bjorn@mork.no>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2017-06-16 20:29:24 +08:00
|
|
|
skb_put_u8(skb, cmd);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
skb_queue_tail(&qca->txq, skb);
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_wq_awake_device(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = container_of(work, struct qca_data,
|
|
|
|
ws_awake_device);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
|
|
|
unsigned long retrans_delay;
|
2019-09-04 12:34:16 +08:00
|
|
|
unsigned long flags;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
BT_DBG("hu %p wq awake device", hu);
|
|
|
|
|
|
|
|
/* Vote for serial clock */
|
|
|
|
serial_clock_vote(HCI_IBS_TX_VOTE_CLOCK_ON, hu);
|
|
|
|
|
2019-09-04 12:34:16 +08:00
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
/* Send wake indication to device */
|
|
|
|
if (send_hci_ibs_cmd(HCI_IBS_WAKE_IND, hu) < 0)
|
|
|
|
BT_ERR("Failed to send WAKE to device");
|
|
|
|
|
|
|
|
qca->ibs_sent_wakes++;
|
|
|
|
|
|
|
|
/* Start retransmit timer */
|
|
|
|
retrans_delay = msecs_to_jiffies(qca->wake_retrans);
|
|
|
|
mod_timer(&qca->wake_retrans_timer, jiffies + retrans_delay);
|
|
|
|
|
2019-09-04 12:34:16 +08:00
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
/* Actually send the packets */
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_wq_awake_rx(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = container_of(work, struct qca_data,
|
|
|
|
ws_awake_rx);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
2019-09-04 12:34:16 +08:00
|
|
|
unsigned long flags;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
BT_DBG("hu %p wq awake rx", hu);
|
|
|
|
|
|
|
|
serial_clock_vote(HCI_IBS_RX_VOTE_CLOCK_ON, hu);
|
|
|
|
|
2019-09-04 12:34:16 +08:00
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
qca->rx_ibs_state = HCI_IBS_RX_AWAKE;
|
|
|
|
|
|
|
|
/* Always acknowledge device wake up,
|
|
|
|
* sending IBS message doesn't count as TX ON.
|
|
|
|
*/
|
|
|
|
if (send_hci_ibs_cmd(HCI_IBS_WAKE_ACK, hu) < 0)
|
|
|
|
BT_ERR("Failed to acknowledge device wake up");
|
|
|
|
|
|
|
|
qca->ibs_sent_wacks++;
|
|
|
|
|
2019-09-04 12:34:16 +08:00
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
/* Actually send the packets */
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_wq_serial_rx_clock_vote_off(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = container_of(work, struct qca_data,
|
|
|
|
ws_rx_vote_off);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
|
|
|
|
|
|
|
BT_DBG("hu %p rx clock vote off", hu);
|
|
|
|
|
|
|
|
serial_clock_vote(HCI_IBS_RX_VOTE_CLOCK_OFF, hu);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_wq_serial_tx_clock_vote_off(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = container_of(work, struct qca_data,
|
|
|
|
ws_tx_vote_off);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
|
|
|
|
|
|
|
BT_DBG("hu %p tx clock vote off", hu);
|
|
|
|
|
|
|
|
/* Run HCI tx handling unlocked */
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
|
|
|
|
/* Now that message queued to tty driver, vote for tty clocks off.
|
|
|
|
* It is up to the tty driver to pend the clocks off until tx done.
|
|
|
|
*/
|
|
|
|
serial_clock_vote(HCI_IBS_TX_VOTE_CLOCK_OFF, hu);
|
|
|
|
}
|
|
|
|
|
2017-10-05 08:54:29 +08:00
|
|
|
static void hci_ibs_tx_idle_timeout(struct timer_list *t)
|
2015-08-11 05:24:17 +08:00
|
|
|
{
|
2017-10-05 08:54:29 +08:00
|
|
|
struct qca_data *qca = from_timer(qca, t, tx_idle_timer);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
2015-08-11 05:24:17 +08:00
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
BT_DBG("hu %p idle timeout in %d state", hu, qca->tx_ibs_state);
|
|
|
|
|
|
|
|
spin_lock_irqsave_nested(&qca->hci_ibs_lock,
|
|
|
|
flags, SINGLE_DEPTH_NESTING);
|
|
|
|
|
|
|
|
switch (qca->tx_ibs_state) {
|
|
|
|
case HCI_IBS_TX_AWAKE:
|
|
|
|
/* TX_IDLE, go to SLEEP */
|
|
|
|
if (send_hci_ibs_cmd(HCI_IBS_SLEEP_IND, hu) < 0) {
|
|
|
|
BT_ERR("Failed to send SLEEP to device");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
qca->tx_ibs_state = HCI_IBS_TX_ASLEEP;
|
|
|
|
qca->ibs_sent_slps++;
|
|
|
|
queue_work(qca->workqueue, &qca->ws_tx_vote_off);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_ASLEEP:
|
|
|
|
case HCI_IBS_TX_WAKING:
|
|
|
|
default:
|
2017-02-18 03:58:10 +08:00
|
|
|
BT_ERR("Spurious timeout tx state %d", qca->tx_ibs_state);
|
2015-08-11 05:24:17 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
}
|
|
|
|
|
2017-10-05 08:54:29 +08:00
|
|
|
static void hci_ibs_wake_retrans_timeout(struct timer_list *t)
|
2015-08-11 05:24:17 +08:00
|
|
|
{
|
2017-10-05 08:54:29 +08:00
|
|
|
struct qca_data *qca = from_timer(qca, t, wake_retrans_timer);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
2015-08-11 05:24:17 +08:00
|
|
|
unsigned long flags, retrans_delay;
|
2015-09-28 16:03:24 +08:00
|
|
|
bool retransmit = false;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
BT_DBG("hu %p wake retransmit timeout in %d state",
|
|
|
|
hu, qca->tx_ibs_state);
|
|
|
|
|
|
|
|
spin_lock_irqsave_nested(&qca->hci_ibs_lock,
|
|
|
|
flags, SINGLE_DEPTH_NESTING);
|
|
|
|
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
/* Don't retransmit the HCI_IBS_WAKE_IND when suspending. */
|
|
|
|
if (test_bit(QCA_SUSPENDING, &qca->flags)) {
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
switch (qca->tx_ibs_state) {
|
|
|
|
case HCI_IBS_TX_WAKING:
|
|
|
|
/* No WAKE_ACK, retransmit WAKE */
|
2015-09-28 16:03:24 +08:00
|
|
|
retransmit = true;
|
2015-08-11 05:24:17 +08:00
|
|
|
if (send_hci_ibs_cmd(HCI_IBS_WAKE_IND, hu) < 0) {
|
|
|
|
BT_ERR("Failed to acknowledge device wake up");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
qca->ibs_sent_wakes++;
|
|
|
|
retrans_delay = msecs_to_jiffies(qca->wake_retrans);
|
|
|
|
mod_timer(&qca->wake_retrans_timer, jiffies + retrans_delay);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_ASLEEP:
|
|
|
|
case HCI_IBS_TX_AWAKE:
|
|
|
|
default:
|
2017-02-18 03:58:10 +08:00
|
|
|
BT_ERR("Spurious timeout tx state %d", qca->tx_ibs_state);
|
2015-08-11 05:24:17 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
if (retransmit)
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
}
|
|
|
|
|
2020-02-15 00:17:15 +08:00
|
|
|
|
|
|
|
static void qca_controller_memdump_timeout(struct work_struct *work)
|
2020-01-02 22:49:11 +08:00
|
|
|
{
|
2020-02-15 00:17:15 +08:00
|
|
|
struct qca_data *qca = container_of(work, struct qca_data,
|
|
|
|
ctrl_memdump_timeout.work);
|
2020-01-02 22:49:11 +08:00
|
|
|
struct hci_uart *hu = qca->hu;
|
2020-02-15 00:17:15 +08:00
|
|
|
|
|
|
|
mutex_lock(&qca->hci_memdump_lock);
|
|
|
|
if (test_bit(QCA_MEMDUMP_COLLECTION, &qca->flags)) {
|
|
|
|
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
|
|
|
if (!test_bit(QCA_HW_ERROR_EVENT, &qca->flags)) {
|
|
|
|
/* Inject hw error event to reset the device
|
|
|
|
* and driver.
|
|
|
|
*/
|
|
|
|
hci_reset_dev(hu->hdev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
2020-01-02 22:49:11 +08:00
|
|
|
}
|
|
|
|
|
2020-02-15 00:17:15 +08:00
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
/* Initialize protocol */
|
|
|
|
static int qca_open(struct hci_uart *hu)
|
|
|
|
{
|
2018-03-30 03:15:24 +08:00
|
|
|
struct qca_serdev *qcadev;
|
2015-08-11 05:24:17 +08:00
|
|
|
struct qca_data *qca;
|
|
|
|
|
|
|
|
BT_DBG("hu %p qca_open", hu);
|
|
|
|
|
2019-07-30 17:33:45 +08:00
|
|
|
if (!hci_uart_has_flow_control(hu))
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
2018-07-23 11:56:51 +08:00
|
|
|
qca = kzalloc(sizeof(struct qca_data), GFP_KERNEL);
|
2015-08-11 05:24:17 +08:00
|
|
|
if (!qca)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
skb_queue_head_init(&qca->txq);
|
|
|
|
skb_queue_head_init(&qca->tx_wait_q);
|
2020-01-02 22:49:11 +08:00
|
|
|
skb_queue_head_init(&qca->rx_memdump_q);
|
2015-08-11 05:24:17 +08:00
|
|
|
spin_lock_init(&qca->hci_ibs_lock);
|
2020-02-15 00:17:15 +08:00
|
|
|
mutex_init(&qca->hci_memdump_lock);
|
2016-08-31 01:12:53 +08:00
|
|
|
qca->workqueue = alloc_ordered_workqueue("qca_wq", 0);
|
2015-08-11 05:24:17 +08:00
|
|
|
if (!qca->workqueue) {
|
|
|
|
BT_ERR("QCA Workqueue not initialized properly");
|
|
|
|
kfree(qca);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
INIT_WORK(&qca->ws_awake_rx, qca_wq_awake_rx);
|
|
|
|
INIT_WORK(&qca->ws_awake_device, qca_wq_awake_device);
|
|
|
|
INIT_WORK(&qca->ws_rx_vote_off, qca_wq_serial_rx_clock_vote_off);
|
|
|
|
INIT_WORK(&qca->ws_tx_vote_off, qca_wq_serial_tx_clock_vote_off);
|
2020-01-02 22:49:11 +08:00
|
|
|
INIT_WORK(&qca->ctrl_memdump_evt, qca_controller_memdump);
|
2020-02-15 00:17:15 +08:00
|
|
|
INIT_DELAYED_WORK(&qca->ctrl_memdump_timeout,
|
|
|
|
qca_controller_memdump_timeout);
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
init_waitqueue_head(&qca->suspend_wait_q);
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
qca->hu = hu;
|
2019-05-22 03:53:07 +08:00
|
|
|
init_completion(&qca->drop_ev_comp);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
/* Assume we start with both sides asleep -- extra wakes OK */
|
|
|
|
qca->tx_ibs_state = HCI_IBS_TX_ASLEEP;
|
|
|
|
qca->rx_ibs_state = HCI_IBS_RX_ASLEEP;
|
|
|
|
|
|
|
|
qca->vote_last_jif = jiffies;
|
|
|
|
|
|
|
|
hu->priv = qca;
|
|
|
|
|
2018-03-30 03:15:24 +08:00
|
|
|
if (hu->serdev) {
|
|
|
|
qcadev = serdev_device_get_drvdata(hu->serdev);
|
2020-04-23 09:34:30 +08:00
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(qcadev->btsoc_type) ||
|
|
|
|
qca_is_wcn6750(qcadev->btsoc_type))
|
2018-08-03 20:16:32 +08:00
|
|
|
hu->init_speed = qcadev->init_speed;
|
2020-04-23 09:34:30 +08:00
|
|
|
|
|
|
|
if (qcadev->oper_speed)
|
2018-08-03 20:16:32 +08:00
|
|
|
hu->oper_speed = qcadev->oper_speed;
|
2018-03-30 03:15:24 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
timer_setup(&qca->wake_retrans_timer, hci_ibs_wake_retrans_timeout, 0);
|
|
|
|
qca->wake_retrans = IBS_WAKE_RETRANS_TIMEOUT_MS;
|
|
|
|
|
|
|
|
timer_setup(&qca->tx_idle_timer, hci_ibs_tx_idle_timeout, 0);
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
qca->tx_idle_delay = IBS_HOST_TX_IDLE_TIMEOUT_MS;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
BT_DBG("HCI_UART_QCA open, tx_idle_delay=%u, wake_retrans=%u",
|
|
|
|
qca->tx_idle_delay, qca->wake_retrans);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_debugfs_init(struct hci_dev *hdev)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
struct dentry *ibs_dir;
|
|
|
|
umode_t mode;
|
|
|
|
|
|
|
|
if (!hdev->debugfs)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ibs_dir = debugfs_create_dir("ibs", hdev->debugfs);
|
|
|
|
|
|
|
|
/* read only */
|
2020-11-11 12:55:38 +08:00
|
|
|
mode = 0444;
|
2015-08-11 05:24:17 +08:00
|
|
|
debugfs_create_u8("tx_ibs_state", mode, ibs_dir, &qca->tx_ibs_state);
|
|
|
|
debugfs_create_u8("rx_ibs_state", mode, ibs_dir, &qca->rx_ibs_state);
|
|
|
|
debugfs_create_u64("ibs_sent_sleeps", mode, ibs_dir,
|
|
|
|
&qca->ibs_sent_slps);
|
|
|
|
debugfs_create_u64("ibs_sent_wakes", mode, ibs_dir,
|
|
|
|
&qca->ibs_sent_wakes);
|
|
|
|
debugfs_create_u64("ibs_sent_wake_acks", mode, ibs_dir,
|
|
|
|
&qca->ibs_sent_wacks);
|
|
|
|
debugfs_create_u64("ibs_recv_sleeps", mode, ibs_dir,
|
|
|
|
&qca->ibs_recv_slps);
|
|
|
|
debugfs_create_u64("ibs_recv_wakes", mode, ibs_dir,
|
|
|
|
&qca->ibs_recv_wakes);
|
|
|
|
debugfs_create_u64("ibs_recv_wake_acks", mode, ibs_dir,
|
|
|
|
&qca->ibs_recv_wacks);
|
2015-08-14 13:09:42 +08:00
|
|
|
debugfs_create_bool("tx_vote", mode, ibs_dir, &qca->tx_vote);
|
2015-08-11 05:24:17 +08:00
|
|
|
debugfs_create_u64("tx_votes_on", mode, ibs_dir, &qca->tx_votes_on);
|
|
|
|
debugfs_create_u64("tx_votes_off", mode, ibs_dir, &qca->tx_votes_off);
|
2015-08-14 13:09:42 +08:00
|
|
|
debugfs_create_bool("rx_vote", mode, ibs_dir, &qca->rx_vote);
|
2015-08-11 05:24:17 +08:00
|
|
|
debugfs_create_u64("rx_votes_on", mode, ibs_dir, &qca->rx_votes_on);
|
|
|
|
debugfs_create_u64("rx_votes_off", mode, ibs_dir, &qca->rx_votes_off);
|
|
|
|
debugfs_create_u64("votes_on", mode, ibs_dir, &qca->votes_on);
|
|
|
|
debugfs_create_u64("votes_off", mode, ibs_dir, &qca->votes_off);
|
|
|
|
debugfs_create_u32("vote_on_ms", mode, ibs_dir, &qca->vote_on_ms);
|
|
|
|
debugfs_create_u32("vote_off_ms", mode, ibs_dir, &qca->vote_off_ms);
|
|
|
|
|
|
|
|
/* read/write */
|
2020-11-11 12:55:38 +08:00
|
|
|
mode = 0644;
|
2015-08-11 05:24:17 +08:00
|
|
|
debugfs_create_u32("wake_retrans", mode, ibs_dir, &qca->wake_retrans);
|
|
|
|
debugfs_create_u32("tx_idle_delay", mode, ibs_dir,
|
|
|
|
&qca->tx_idle_delay);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush protocol data */
|
|
|
|
static int qca_flush(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
BT_DBG("hu %p qca flush", hu);
|
|
|
|
|
|
|
|
skb_queue_purge(&qca->tx_wait_q);
|
|
|
|
skb_queue_purge(&qca->txq);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Close protocol */
|
|
|
|
static int qca_close(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
BT_DBG("hu %p qca close", hu);
|
|
|
|
|
|
|
|
serial_clock_vote(HCI_IBS_VOTE_STATS_UPDATE, hu);
|
|
|
|
|
|
|
|
skb_queue_purge(&qca->tx_wait_q);
|
|
|
|
skb_queue_purge(&qca->txq);
|
2020-01-02 22:49:11 +08:00
|
|
|
skb_queue_purge(&qca->rx_memdump_q);
|
2015-08-11 05:24:17 +08:00
|
|
|
destroy_workqueue(qca->workqueue);
|
2022-04-05 22:02:00 +08:00
|
|
|
del_timer_sync(&qca->tx_idle_timer);
|
|
|
|
del_timer_sync(&qca->wake_retrans_timer);
|
2015-08-11 05:24:17 +08:00
|
|
|
qca->hu = NULL;
|
|
|
|
|
|
|
|
kfree_skb(qca->rx_skb);
|
|
|
|
|
|
|
|
hu->priv = NULL;
|
|
|
|
|
|
|
|
kfree(qca);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Called upon a wake-up-indication from the device.
|
|
|
|
*/
|
|
|
|
static void device_want_to_wakeup(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
BT_DBG("hu %p want to wake up", hu);
|
|
|
|
|
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
qca->ibs_recv_wakes++;
|
|
|
|
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
/* Don't wake the rx up when suspending. */
|
|
|
|
if (test_bit(QCA_SUSPENDING, &qca->flags)) {
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
switch (qca->rx_ibs_state) {
|
|
|
|
case HCI_IBS_RX_ASLEEP:
|
|
|
|
/* Make sure clock is on - we may have turned clock off since
|
|
|
|
* receiving the wake up indicator awake rx clock.
|
|
|
|
*/
|
|
|
|
queue_work(qca->workqueue, &qca->ws_awake_rx);
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case HCI_IBS_RX_AWAKE:
|
|
|
|
/* Always acknowledge device wake up,
|
|
|
|
* sending IBS message doesn't count as TX ON.
|
|
|
|
*/
|
|
|
|
if (send_hci_ibs_cmd(HCI_IBS_WAKE_ACK, hu) < 0) {
|
|
|
|
BT_ERR("Failed to acknowledge device wake up");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
qca->ibs_sent_wacks++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* Any other state is illegal */
|
|
|
|
BT_ERR("Received HCI_IBS_WAKE_IND in rx state %d",
|
|
|
|
qca->rx_ibs_state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
/* Actually send the packets */
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Called upon a sleep-indication from the device.
|
|
|
|
*/
|
|
|
|
static void device_want_to_sleep(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
2019-08-14 15:42:39 +08:00
|
|
|
BT_DBG("hu %p want to sleep in %d state", hu, qca->rx_ibs_state);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
qca->ibs_recv_slps++;
|
|
|
|
|
|
|
|
switch (qca->rx_ibs_state) {
|
|
|
|
case HCI_IBS_RX_AWAKE:
|
|
|
|
/* Update state */
|
|
|
|
qca->rx_ibs_state = HCI_IBS_RX_ASLEEP;
|
|
|
|
/* Vote off rx clock under workqueue */
|
|
|
|
queue_work(qca->workqueue, &qca->ws_rx_vote_off);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_RX_ASLEEP:
|
2019-08-14 15:42:39 +08:00
|
|
|
break;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
default:
|
|
|
|
/* Any other state is illegal */
|
|
|
|
BT_ERR("Received HCI_IBS_SLEEP_IND in rx state %d",
|
|
|
|
qca->rx_ibs_state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
wake_up_interruptible(&qca->suspend_wait_q);
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Called upon wake-up-acknowledgement from the device
|
|
|
|
*/
|
|
|
|
static void device_woke_up(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
unsigned long flags, idle_delay;
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
struct sk_buff *skb = NULL;
|
|
|
|
|
|
|
|
BT_DBG("hu %p woke up", hu);
|
|
|
|
|
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
qca->ibs_recv_wacks++;
|
|
|
|
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
/* Don't react to the wake-up-acknowledgment when suspending. */
|
|
|
|
if (test_bit(QCA_SUSPENDING, &qca->flags)) {
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
switch (qca->tx_ibs_state) {
|
|
|
|
case HCI_IBS_TX_AWAKE:
|
|
|
|
/* Expect one if we send 2 WAKEs */
|
|
|
|
BT_DBG("Received HCI_IBS_WAKE_ACK in tx state %d",
|
|
|
|
qca->tx_ibs_state);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_WAKING:
|
|
|
|
/* Send pending packets */
|
|
|
|
while ((skb = skb_dequeue(&qca->tx_wait_q)))
|
|
|
|
skb_queue_tail(&qca->txq, skb);
|
|
|
|
|
|
|
|
/* Switch timers and change state to HCI_IBS_TX_AWAKE */
|
|
|
|
del_timer(&qca->wake_retrans_timer);
|
|
|
|
idle_delay = msecs_to_jiffies(qca->tx_idle_delay);
|
|
|
|
mod_timer(&qca->tx_idle_timer, jiffies + idle_delay);
|
|
|
|
qca->tx_ibs_state = HCI_IBS_TX_AWAKE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_ASLEEP:
|
|
|
|
default:
|
|
|
|
BT_ERR("Received HCI_IBS_WAKE_ACK in tx state %d",
|
|
|
|
qca->tx_ibs_state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
/* Actually send the packets */
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enqueue frame for transmittion (padding, crc, etc) may be called from
|
|
|
|
* two simultaneous tasklets.
|
|
|
|
*/
|
|
|
|
static int qca_enqueue(struct hci_uart *hu, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
unsigned long flags = 0, idle_delay;
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
BT_DBG("hu %p qca enq skb %p tx_ibs_state %d", hu, skb,
|
|
|
|
qca->tx_ibs_state);
|
|
|
|
|
2020-07-11 19:31:12 +08:00
|
|
|
if (test_bit(QCA_SSR_TRIGGERED, &qca->flags)) {
|
|
|
|
/* As SSR is in progress, ignore the packets */
|
|
|
|
bt_dev_dbg(hu->hdev, "SSR is in progress");
|
|
|
|
kfree_skb(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
/* Prepend skb with frame type */
|
2015-11-05 14:33:56 +08:00
|
|
|
memcpy(skb_push(skb, 1), &hci_skb_pkt_type(skb), 1);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
2019-02-04 23:06:43 +08:00
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
/* Don't go to sleep in middle of patch download or
|
|
|
|
* Out-Of-Band(GPIOs control) sleep is selected.
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
* Don't wake the device up when suspending.
|
2015-08-11 05:24:17 +08:00
|
|
|
*/
|
2020-10-06 23:20:21 +08:00
|
|
|
if (test_bit(QCA_IBS_DISABLED, &qca->flags) ||
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
test_bit(QCA_SUSPENDING, &qca->flags)) {
|
2015-08-11 05:24:17 +08:00
|
|
|
skb_queue_tail(&qca->txq, skb);
|
2019-02-04 23:06:43 +08:00
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Act according to current state */
|
|
|
|
switch (qca->tx_ibs_state) {
|
|
|
|
case HCI_IBS_TX_AWAKE:
|
|
|
|
BT_DBG("Device awake, sending normally");
|
|
|
|
skb_queue_tail(&qca->txq, skb);
|
|
|
|
idle_delay = msecs_to_jiffies(qca->tx_idle_delay);
|
|
|
|
mod_timer(&qca->tx_idle_timer, jiffies + idle_delay);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_ASLEEP:
|
|
|
|
BT_DBG("Device asleep, waking up and queueing packet");
|
|
|
|
/* Save packet for later */
|
|
|
|
skb_queue_tail(&qca->tx_wait_q, skb);
|
|
|
|
|
|
|
|
qca->tx_ibs_state = HCI_IBS_TX_WAKING;
|
|
|
|
/* Schedule a work queue to wake up device */
|
|
|
|
queue_work(qca->workqueue, &qca->ws_awake_device);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_WAKING:
|
|
|
|
BT_DBG("Device waking up, queueing packet");
|
|
|
|
/* Transient state; just keep packet for later */
|
|
|
|
skb_queue_tail(&qca->tx_wait_q, skb);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BT_ERR("Illegal tx state: %d (losing packet)",
|
|
|
|
qca->tx_ibs_state);
|
|
|
|
kfree_skb(skb);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qca_ibs_sleep_ind(struct hci_dev *hdev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
|
|
|
|
BT_DBG("hu %p recv hci ibs cmd 0x%x", hu, HCI_IBS_SLEEP_IND);
|
|
|
|
|
|
|
|
device_want_to_sleep(hu);
|
|
|
|
|
|
|
|
kfree_skb(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qca_ibs_wake_ind(struct hci_dev *hdev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
|
|
|
|
BT_DBG("hu %p recv hci ibs cmd 0x%x", hu, HCI_IBS_WAKE_IND);
|
|
|
|
|
|
|
|
device_want_to_wakeup(hu);
|
|
|
|
|
|
|
|
kfree_skb(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qca_ibs_wake_ack(struct hci_dev *hdev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
|
|
|
|
BT_DBG("hu %p recv hci ibs cmd 0x%x", hu, HCI_IBS_WAKE_ACK);
|
|
|
|
|
|
|
|
device_woke_up(hu);
|
|
|
|
|
|
|
|
kfree_skb(skb);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-16 22:21:35 +08:00
|
|
|
static int qca_recv_acl_data(struct hci_dev *hdev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
/* We receive debug logs from chip as an ACL packets.
|
|
|
|
* Instead of sending the data to ACL to decode the
|
|
|
|
* received data, we are pushing them to the above layers
|
|
|
|
* as a diagnostic packet.
|
|
|
|
*/
|
|
|
|
if (get_unaligned_le16(skb->data) == QCA_DEBUG_HANDLE)
|
|
|
|
return hci_recv_diag(hdev, skb);
|
|
|
|
|
|
|
|
return hci_recv_frame(hdev, skb);
|
|
|
|
}
|
|
|
|
|
2020-01-02 22:49:11 +08:00
|
|
|
static void qca_controller_memdump(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = container_of(work, struct qca_data,
|
|
|
|
ctrl_memdump_evt);
|
|
|
|
struct hci_uart *hu = qca->hu;
|
|
|
|
struct sk_buff *skb;
|
|
|
|
struct qca_memdump_event_hdr *cmd_hdr;
|
|
|
|
struct qca_memdump_data *qca_memdump = qca->qca_memdump;
|
|
|
|
struct qca_dump_size *dump;
|
|
|
|
char *memdump_buf;
|
|
|
|
char nullBuff[QCA_DUMP_PACKET_SIZE] = { 0 };
|
2020-01-08 09:54:31 +08:00
|
|
|
u16 seq_no;
|
2020-01-02 22:49:11 +08:00
|
|
|
u32 dump_size;
|
2020-05-29 22:38:31 +08:00
|
|
|
u32 rx_size;
|
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
2020-01-02 22:49:11 +08:00
|
|
|
|
|
|
|
while ((skb = skb_dequeue(&qca->rx_memdump_q))) {
|
|
|
|
|
2020-02-15 00:17:15 +08:00
|
|
|
mutex_lock(&qca->hci_memdump_lock);
|
2020-06-09 23:57:08 +08:00
|
|
|
/* Skip processing the received packets if timeout detected
|
|
|
|
* or memdump collection completed.
|
|
|
|
*/
|
|
|
|
if (qca->memdump_state == QCA_MEMDUMP_TIMEOUT ||
|
|
|
|
qca->memdump_state == QCA_MEMDUMP_COLLECTED) {
|
2020-02-15 00:17:15 +08:00
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-02 22:49:11 +08:00
|
|
|
if (!qca_memdump) {
|
|
|
|
qca_memdump = kzalloc(sizeof(struct qca_memdump_data),
|
|
|
|
GFP_ATOMIC);
|
2020-02-15 00:17:15 +08:00
|
|
|
if (!qca_memdump) {
|
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
2020-01-02 22:49:11 +08:00
|
|
|
return;
|
2020-02-15 00:17:15 +08:00
|
|
|
}
|
2020-01-02 22:49:11 +08:00
|
|
|
|
|
|
|
qca->qca_memdump = qca_memdump;
|
|
|
|
}
|
|
|
|
|
|
|
|
qca->memdump_state = QCA_MEMDUMP_COLLECTING;
|
|
|
|
cmd_hdr = (void *) skb->data;
|
|
|
|
seq_no = __le16_to_cpu(cmd_hdr->seq_no);
|
|
|
|
skb_pull(skb, sizeof(struct qca_memdump_event_hdr));
|
|
|
|
|
|
|
|
if (!seq_no) {
|
|
|
|
|
|
|
|
/* This is the first frame of memdump packet from
|
|
|
|
* the controller, Disable IBS to recevie dump
|
|
|
|
* with out any interruption, ideally time required for
|
|
|
|
* the controller to send the dump is 8 seconds. let us
|
|
|
|
* start timer to handle this asynchronous activity.
|
|
|
|
*/
|
2020-10-06 23:20:21 +08:00
|
|
|
set_bit(QCA_IBS_DISABLED, &qca->flags);
|
2020-01-02 22:49:11 +08:00
|
|
|
set_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
|
|
|
dump = (void *) skb->data;
|
|
|
|
dump_size = __le32_to_cpu(dump->dump_size);
|
|
|
|
if (!(dump_size)) {
|
|
|
|
bt_dev_err(hu->hdev, "Rx invalid memdump size");
|
2021-01-02 13:47:55 +08:00
|
|
|
kfree(qca_memdump);
|
2020-01-02 22:49:11 +08:00
|
|
|
kfree_skb(skb);
|
2021-01-02 13:47:55 +08:00
|
|
|
qca->qca_memdump = NULL;
|
2020-02-15 00:17:15 +08:00
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
2020-01-02 22:49:11 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bt_dev_info(hu->hdev, "QCA collecting dump of size:%u",
|
|
|
|
dump_size);
|
2020-02-15 00:17:15 +08:00
|
|
|
queue_delayed_work(qca->workqueue,
|
|
|
|
&qca->ctrl_memdump_timeout,
|
2020-05-29 22:38:31 +08:00
|
|
|
msecs_to_jiffies(MEMDUMP_TIMEOUT_MS)
|
|
|
|
);
|
2020-01-02 22:49:11 +08:00
|
|
|
|
|
|
|
skb_pull(skb, sizeof(dump_size));
|
|
|
|
memdump_buf = vmalloc(dump_size);
|
2020-05-29 22:38:31 +08:00
|
|
|
qca_memdump->ram_dump_size = dump_size;
|
2020-01-02 22:49:11 +08:00
|
|
|
qca_memdump->memdump_buf_head = memdump_buf;
|
|
|
|
qca_memdump->memdump_buf_tail = memdump_buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
memdump_buf = qca_memdump->memdump_buf_tail;
|
|
|
|
|
|
|
|
/* If sequence no 0 is missed then there is no point in
|
|
|
|
* accepting the other sequences.
|
|
|
|
*/
|
|
|
|
if (!memdump_buf) {
|
|
|
|
bt_dev_err(hu->hdev, "QCA: Discarding other packets");
|
|
|
|
kfree(qca_memdump);
|
|
|
|
kfree_skb(skb);
|
|
|
|
qca->qca_memdump = NULL;
|
2020-02-15 00:17:15 +08:00
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
2020-01-02 22:49:11 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* There could be chance of missing some packets from
|
|
|
|
* the controller. In such cases let us store the dummy
|
|
|
|
* packets in the buffer.
|
|
|
|
*/
|
2020-05-29 22:38:31 +08:00
|
|
|
/* For QCA6390, controller does not lost packets but
|
2021-03-22 09:00:51 +08:00
|
|
|
* sequence number field of packet sometimes has error
|
2020-05-29 22:38:31 +08:00
|
|
|
* bits, so skip this checking for missing packet.
|
|
|
|
*/
|
2020-01-02 22:49:11 +08:00
|
|
|
while ((seq_no > qca_memdump->current_seq_no + 1) &&
|
2020-05-29 22:38:31 +08:00
|
|
|
(soc_type != QCA_QCA6390) &&
|
|
|
|
seq_no != QCA_LAST_SEQUENCE_NUM) {
|
2020-01-02 22:49:11 +08:00
|
|
|
bt_dev_err(hu->hdev, "QCA controller missed packet:%d",
|
|
|
|
qca_memdump->current_seq_no);
|
2020-05-29 22:38:31 +08:00
|
|
|
rx_size = qca_memdump->received_dump;
|
|
|
|
rx_size += QCA_DUMP_PACKET_SIZE;
|
|
|
|
if (rx_size > qca_memdump->ram_dump_size) {
|
|
|
|
bt_dev_err(hu->hdev,
|
|
|
|
"QCA memdump received %d, no space for missed packet",
|
|
|
|
qca_memdump->received_dump);
|
|
|
|
break;
|
|
|
|
}
|
2020-01-02 22:49:11 +08:00
|
|
|
memcpy(memdump_buf, nullBuff, QCA_DUMP_PACKET_SIZE);
|
|
|
|
memdump_buf = memdump_buf + QCA_DUMP_PACKET_SIZE;
|
|
|
|
qca_memdump->received_dump += QCA_DUMP_PACKET_SIZE;
|
|
|
|
qca_memdump->current_seq_no++;
|
|
|
|
}
|
|
|
|
|
2020-05-29 22:38:31 +08:00
|
|
|
rx_size = qca_memdump->received_dump + skb->len;
|
|
|
|
if (rx_size <= qca_memdump->ram_dump_size) {
|
|
|
|
if ((seq_no != QCA_LAST_SEQUENCE_NUM) &&
|
|
|
|
(seq_no != qca_memdump->current_seq_no))
|
|
|
|
bt_dev_err(hu->hdev,
|
|
|
|
"QCA memdump unexpected packet %d",
|
|
|
|
seq_no);
|
|
|
|
bt_dev_dbg(hu->hdev,
|
|
|
|
"QCA memdump packet %d with length %d",
|
|
|
|
seq_no, skb->len);
|
|
|
|
memcpy(memdump_buf, (unsigned char *)skb->data,
|
|
|
|
skb->len);
|
|
|
|
memdump_buf = memdump_buf + skb->len;
|
|
|
|
qca_memdump->memdump_buf_tail = memdump_buf;
|
|
|
|
qca_memdump->current_seq_no = seq_no + 1;
|
|
|
|
qca_memdump->received_dump += skb->len;
|
|
|
|
} else {
|
|
|
|
bt_dev_err(hu->hdev,
|
|
|
|
"QCA memdump received %d, no space for packet %d",
|
|
|
|
qca_memdump->received_dump, seq_no);
|
|
|
|
}
|
2020-01-02 22:49:11 +08:00
|
|
|
qca->qca_memdump = qca_memdump;
|
|
|
|
kfree_skb(skb);
|
|
|
|
if (seq_no == QCA_LAST_SEQUENCE_NUM) {
|
2020-05-29 22:38:31 +08:00
|
|
|
bt_dev_info(hu->hdev,
|
|
|
|
"QCA memdump Done, received %d, total %d",
|
|
|
|
qca_memdump->received_dump,
|
|
|
|
qca_memdump->ram_dump_size);
|
2020-01-02 22:49:11 +08:00
|
|
|
memdump_buf = qca_memdump->memdump_buf_head;
|
|
|
|
dev_coredumpv(&hu->serdev->dev, memdump_buf,
|
|
|
|
qca_memdump->received_dump, GFP_KERNEL);
|
2020-02-15 00:17:15 +08:00
|
|
|
cancel_delayed_work(&qca->ctrl_memdump_timeout);
|
2020-01-02 22:49:11 +08:00
|
|
|
kfree(qca->qca_memdump);
|
|
|
|
qca->qca_memdump = NULL;
|
|
|
|
qca->memdump_state = QCA_MEMDUMP_COLLECTED;
|
2020-02-15 00:17:15 +08:00
|
|
|
clear_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
2020-01-02 22:49:11 +08:00
|
|
|
}
|
2020-02-15 00:17:15 +08:00
|
|
|
|
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
2020-01-02 22:49:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-02-15 00:17:15 +08:00
|
|
|
static int qca_controller_memdump_event(struct hci_dev *hdev,
|
|
|
|
struct sk_buff *skb)
|
2020-01-02 22:49:11 +08:00
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
2020-07-11 19:31:12 +08:00
|
|
|
set_bit(QCA_SSR_TRIGGERED, &qca->flags);
|
2020-01-02 22:49:11 +08:00
|
|
|
skb_queue_tail(&qca->rx_memdump_q, skb);
|
|
|
|
queue_work(qca->workqueue, &qca->ctrl_memdump_evt);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-05-22 03:53:07 +08:00
|
|
|
static int qca_recv_event(struct hci_dev *hdev, struct sk_buff *skb)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
if (test_bit(QCA_DROP_VENDOR_EVENT, &qca->flags)) {
|
|
|
|
struct hci_event_hdr *hdr = (void *)skb->data;
|
|
|
|
|
|
|
|
/* For the WCN3990 the vendor command for a baudrate change
|
|
|
|
* isn't sent as synchronous HCI command, because the
|
|
|
|
* controller sends the corresponding vendor event with the
|
|
|
|
* new baudrate. The event is received and properly decoded
|
|
|
|
* after changing the baudrate of the host port. It needs to
|
|
|
|
* be dropped, otherwise it can be misinterpreted as
|
|
|
|
* response to a later firmware download command (also a
|
|
|
|
* vendor command).
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (hdr->evt == HCI_EV_VENDOR)
|
|
|
|
complete(&qca->drop_ev_comp);
|
|
|
|
|
2019-07-09 09:35:30 +08:00
|
|
|
kfree_skb(skb);
|
2019-05-22 03:53:07 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2020-01-02 22:49:11 +08:00
|
|
|
/* We receive chip memory dump as an event packet, With a dedicated
|
|
|
|
* handler followed by a hardware error event. When this event is
|
|
|
|
* received we store dump into a file before closing hci. This
|
|
|
|
* dump will help in triaging the issues.
|
|
|
|
*/
|
|
|
|
if ((skb->data[0] == HCI_VENDOR_PKT) &&
|
|
|
|
(get_unaligned_be16(skb->data + 2) == QCA_SSR_DUMP_HANDLE))
|
|
|
|
return qca_controller_memdump_event(hdev, skb);
|
2019-05-22 03:53:07 +08:00
|
|
|
|
|
|
|
return hci_recv_frame(hdev, skb);
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
#define QCA_IBS_SLEEP_IND_EVENT \
|
|
|
|
.type = HCI_IBS_SLEEP_IND, \
|
|
|
|
.hlen = 0, \
|
|
|
|
.loff = 0, \
|
|
|
|
.lsize = 0, \
|
|
|
|
.maxlen = HCI_MAX_IBS_SIZE
|
|
|
|
|
|
|
|
#define QCA_IBS_WAKE_IND_EVENT \
|
|
|
|
.type = HCI_IBS_WAKE_IND, \
|
|
|
|
.hlen = 0, \
|
|
|
|
.loff = 0, \
|
|
|
|
.lsize = 0, \
|
|
|
|
.maxlen = HCI_MAX_IBS_SIZE
|
|
|
|
|
|
|
|
#define QCA_IBS_WAKE_ACK_EVENT \
|
|
|
|
.type = HCI_IBS_WAKE_ACK, \
|
|
|
|
.hlen = 0, \
|
|
|
|
.loff = 0, \
|
|
|
|
.lsize = 0, \
|
|
|
|
.maxlen = HCI_MAX_IBS_SIZE
|
|
|
|
|
|
|
|
static const struct h4_recv_pkt qca_recv_pkts[] = {
|
2018-10-16 22:21:35 +08:00
|
|
|
{ H4_RECV_ACL, .recv = qca_recv_acl_data },
|
2015-08-11 05:24:17 +08:00
|
|
|
{ H4_RECV_SCO, .recv = hci_recv_frame },
|
2019-05-22 03:53:07 +08:00
|
|
|
{ H4_RECV_EVENT, .recv = qca_recv_event },
|
2015-08-11 05:24:17 +08:00
|
|
|
{ QCA_IBS_WAKE_IND_EVENT, .recv = qca_ibs_wake_ind },
|
|
|
|
{ QCA_IBS_WAKE_ACK_EVENT, .recv = qca_ibs_wake_ack },
|
|
|
|
{ QCA_IBS_SLEEP_IND_EVENT, .recv = qca_ibs_sleep_ind },
|
|
|
|
};
|
|
|
|
|
|
|
|
static int qca_recv(struct hci_uart *hu, const void *data, int count)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
if (!test_bit(HCI_UART_REGISTERED, &hu->flags))
|
|
|
|
return -EUNATCH;
|
|
|
|
|
|
|
|
qca->rx_skb = h4_recv_buf(hu->hdev, qca->rx_skb, data, count,
|
|
|
|
qca_recv_pkts, ARRAY_SIZE(qca_recv_pkts));
|
|
|
|
if (IS_ERR(qca->rx_skb)) {
|
|
|
|
int err = PTR_ERR(qca->rx_skb);
|
2017-10-30 17:42:59 +08:00
|
|
|
bt_dev_err(hu->hdev, "Frame reassembly failed (%d)", err);
|
2015-08-11 05:24:17 +08:00
|
|
|
qca->rx_skb = NULL;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct sk_buff *qca_dequeue(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
return skb_dequeue(&qca->txq);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t qca_get_baudrate_value(int speed)
|
|
|
|
{
|
2015-09-15 20:19:45 +08:00
|
|
|
switch (speed) {
|
2015-08-11 05:24:17 +08:00
|
|
|
case 9600:
|
|
|
|
return QCA_BAUDRATE_9600;
|
|
|
|
case 19200:
|
|
|
|
return QCA_BAUDRATE_19200;
|
|
|
|
case 38400:
|
|
|
|
return QCA_BAUDRATE_38400;
|
|
|
|
case 57600:
|
|
|
|
return QCA_BAUDRATE_57600;
|
|
|
|
case 115200:
|
|
|
|
return QCA_BAUDRATE_115200;
|
|
|
|
case 230400:
|
|
|
|
return QCA_BAUDRATE_230400;
|
|
|
|
case 460800:
|
|
|
|
return QCA_BAUDRATE_460800;
|
|
|
|
case 500000:
|
|
|
|
return QCA_BAUDRATE_500000;
|
|
|
|
case 921600:
|
|
|
|
return QCA_BAUDRATE_921600;
|
|
|
|
case 1000000:
|
|
|
|
return QCA_BAUDRATE_1000000;
|
|
|
|
case 2000000:
|
|
|
|
return QCA_BAUDRATE_2000000;
|
|
|
|
case 3000000:
|
|
|
|
return QCA_BAUDRATE_3000000;
|
2018-08-03 20:16:30 +08:00
|
|
|
case 3200000:
|
|
|
|
return QCA_BAUDRATE_3200000;
|
2015-08-11 05:24:17 +08:00
|
|
|
case 3500000:
|
|
|
|
return QCA_BAUDRATE_3500000;
|
|
|
|
default:
|
|
|
|
return QCA_BAUDRATE_115200;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qca_set_baudrate(struct hci_dev *hdev, uint8_t baudrate)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
struct sk_buff *skb;
|
|
|
|
u8 cmd[] = { 0x01, 0x48, 0xFC, 0x01, 0x00 };
|
|
|
|
|
2018-08-03 20:16:30 +08:00
|
|
|
if (baudrate > QCA_BAUDRATE_3200000)
|
2015-08-11 05:24:17 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
cmd[4] = baudrate;
|
|
|
|
|
2018-07-23 11:56:51 +08:00
|
|
|
skb = bt_skb_alloc(sizeof(cmd), GFP_KERNEL);
|
2015-08-11 05:24:17 +08:00
|
|
|
if (!skb) {
|
2017-10-30 17:42:59 +08:00
|
|
|
bt_dev_err(hdev, "Failed to allocate baudrate packet");
|
2015-08-11 05:24:17 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assign commands to change baudrate and packet type. */
|
networking: introduce and use skb_put_data()
A common pattern with skb_put() is to just want to memcpy()
some data into the new space, introduce skb_put_data() for
this.
An spatch similar to the one for skb_put_zero() converts many
of the places using it:
@@
identifier p, p2;
expression len, skb, data;
type t, t2;
@@
(
-p = skb_put(skb, len);
+p = skb_put_data(skb, data, len);
|
-p = (t)skb_put(skb, len);
+p = skb_put_data(skb, data, len);
)
(
p2 = (t2)p;
-memcpy(p2, data, len);
|
-memcpy(p, data, len);
)
@@
type t, t2;
identifier p, p2;
expression skb, data;
@@
t *p;
...
(
-p = skb_put(skb, sizeof(t));
+p = skb_put_data(skb, data, sizeof(t));
|
-p = (t *)skb_put(skb, sizeof(t));
+p = skb_put_data(skb, data, sizeof(t));
)
(
p2 = (t2)p;
-memcpy(p2, data, sizeof(*p));
|
-memcpy(p, data, sizeof(*p));
)
@@
expression skb, len, data;
@@
-memcpy(skb_put(skb, len), data, len);
+skb_put_data(skb, data, len);
(again, manually post-processed to retain some comments)
Reviewed-by: Stephen Hemminger <stephen@networkplumber.org>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2017-06-16 20:29:20 +08:00
|
|
|
skb_put_data(skb, cmd, sizeof(cmd));
|
2015-11-05 14:33:56 +08:00
|
|
|
hci_skb_pkt_type(skb) = HCI_COMMAND_PKT;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
skb_queue_tail(&qca->txq, skb);
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
|
2019-02-28 07:52:23 +08:00
|
|
|
/* Wait for the baudrate change request to be sent */
|
|
|
|
|
|
|
|
while (!skb_queue_empty(&qca->txq))
|
|
|
|
usleep_range(100, 200);
|
|
|
|
|
2019-04-24 02:16:52 +08:00
|
|
|
if (hu->serdev)
|
|
|
|
serdev_device_wait_until_sent(hu->serdev,
|
2019-02-28 07:52:23 +08:00
|
|
|
msecs_to_jiffies(CMD_TRANS_TIMEOUT_MS));
|
|
|
|
|
|
|
|
/* Give the controller time to process the request */
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(qca_soc_type(hu)) ||
|
|
|
|
qca_is_wcn6750(qca_soc_type(hu)))
|
2020-11-11 12:55:38 +08:00
|
|
|
usleep_range(1000, 10000);
|
2019-02-28 07:52:23 +08:00
|
|
|
else
|
|
|
|
msleep(300);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-03-30 03:15:24 +08:00
|
|
|
static inline void host_set_baudrate(struct hci_uart *hu, unsigned int speed)
|
|
|
|
{
|
|
|
|
if (hu->serdev)
|
|
|
|
serdev_device_set_baudrate(hu->serdev, speed);
|
|
|
|
else
|
|
|
|
hci_uart_set_baudrate(hu, speed);
|
|
|
|
}
|
|
|
|
|
2019-02-27 03:46:45 +08:00
|
|
|
static int qca_send_power_pulse(struct hci_uart *hu, bool on)
|
2018-08-03 20:16:32 +08:00
|
|
|
{
|
2019-02-04 23:06:41 +08:00
|
|
|
int ret;
|
2019-02-28 07:52:23 +08:00
|
|
|
int timeout = msecs_to_jiffies(CMD_TRANS_TIMEOUT_MS);
|
2019-02-27 03:46:45 +08:00
|
|
|
u8 cmd = on ? QCA_WCN3990_POWERON_PULSE : QCA_WCN3990_POWEROFF_PULSE;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
|
|
|
/* These power pulses are single byte command which are sent
|
|
|
|
* at required baudrate to wcn3990. On wcn3990, we have an external
|
|
|
|
* circuit at Tx pin which decodes the pulse sent at specific baudrate.
|
|
|
|
* For example, wcn3990 supports RF COEX antenna for both Wi-Fi/BT
|
|
|
|
* and also we use the same power inputs to turn on and off for
|
|
|
|
* Wi-Fi/BT. Powering up the power sources will not enable BT, until
|
|
|
|
* we send a power on pulse at 115200 bps. This algorithm will help to
|
|
|
|
* save power. Disabling hardware flow control is mandatory while
|
|
|
|
* sending power pulses to SoC.
|
|
|
|
*/
|
2019-02-04 23:06:41 +08:00
|
|
|
bt_dev_dbg(hu->hdev, "sending power pulse %02x to controller", cmd);
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-02-04 23:06:41 +08:00
|
|
|
serdev_device_write_flush(hu->serdev);
|
2018-08-03 20:16:32 +08:00
|
|
|
hci_uart_set_flow_control(hu, true);
|
2019-02-04 23:06:41 +08:00
|
|
|
ret = serdev_device_write_buf(hu->serdev, &cmd, sizeof(cmd));
|
|
|
|
if (ret < 0) {
|
|
|
|
bt_dev_err(hu->hdev, "failed to send power pulse %02x", cmd);
|
|
|
|
return ret;
|
|
|
|
}
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-02-04 23:06:41 +08:00
|
|
|
serdev_device_wait_until_sent(hu->serdev, timeout);
|
2018-08-03 20:16:32 +08:00
|
|
|
hci_uart_set_flow_control(hu, false);
|
|
|
|
|
2019-02-27 03:46:47 +08:00
|
|
|
/* Give to controller time to boot/shutdown */
|
2019-02-27 03:46:46 +08:00
|
|
|
if (on)
|
|
|
|
msleep(100);
|
2019-02-27 03:46:47 +08:00
|
|
|
else
|
2020-11-11 12:55:38 +08:00
|
|
|
usleep_range(1000, 10000);
|
2019-02-27 03:46:46 +08:00
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-08-03 20:16:29 +08:00
|
|
|
static unsigned int qca_get_speed(struct hci_uart *hu,
|
|
|
|
enum qca_speed_type speed_type)
|
|
|
|
{
|
|
|
|
unsigned int speed = 0;
|
|
|
|
|
|
|
|
if (speed_type == QCA_INIT_SPEED) {
|
|
|
|
if (hu->init_speed)
|
|
|
|
speed = hu->init_speed;
|
|
|
|
else if (hu->proto->init_speed)
|
|
|
|
speed = hu->proto->init_speed;
|
|
|
|
} else {
|
|
|
|
if (hu->oper_speed)
|
|
|
|
speed = hu->oper_speed;
|
|
|
|
else if (hu->proto->oper_speed)
|
|
|
|
speed = hu->proto->oper_speed;
|
|
|
|
}
|
|
|
|
|
|
|
|
return speed;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qca_check_speeds(struct hci_uart *hu)
|
|
|
|
{
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(qca_soc_type(hu)) ||
|
|
|
|
qca_is_wcn6750(qca_soc_type(hu))) {
|
2018-08-03 20:16:32 +08:00
|
|
|
if (!qca_get_speed(hu, QCA_INIT_SPEED) &&
|
|
|
|
!qca_get_speed(hu, QCA_OPER_SPEED))
|
|
|
|
return -EINVAL;
|
|
|
|
} else {
|
|
|
|
if (!qca_get_speed(hu, QCA_INIT_SPEED) ||
|
|
|
|
!qca_get_speed(hu, QCA_OPER_SPEED))
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2018-08-03 20:16:29 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qca_set_speed(struct hci_uart *hu, enum qca_speed_type speed_type)
|
|
|
|
{
|
|
|
|
unsigned int speed, qca_baudrate;
|
2019-05-22 03:53:07 +08:00
|
|
|
struct qca_data *qca = hu->priv;
|
2019-02-04 23:06:42 +08:00
|
|
|
int ret = 0;
|
2018-08-03 20:16:29 +08:00
|
|
|
|
|
|
|
if (speed_type == QCA_INIT_SPEED) {
|
|
|
|
speed = qca_get_speed(hu, QCA_INIT_SPEED);
|
|
|
|
if (speed)
|
|
|
|
host_set_baudrate(hu, speed);
|
|
|
|
} else {
|
2019-03-12 02:38:31 +08:00
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
|
|
|
|
2018-08-03 20:16:29 +08:00
|
|
|
speed = qca_get_speed(hu, QCA_OPER_SPEED);
|
|
|
|
if (!speed)
|
|
|
|
return 0;
|
|
|
|
|
2019-02-04 23:06:42 +08:00
|
|
|
/* Disable flow control for wcn3990 to deassert RTS while
|
|
|
|
* changing the baudrate of chip and host.
|
|
|
|
*/
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(soc_type) ||
|
|
|
|
qca_is_wcn6750(soc_type))
|
2019-02-04 23:06:42 +08:00
|
|
|
hci_uart_set_flow_control(hu, true);
|
|
|
|
|
2019-05-22 03:53:07 +08:00
|
|
|
if (soc_type == QCA_WCN3990) {
|
|
|
|
reinit_completion(&qca->drop_ev_comp);
|
|
|
|
set_bit(QCA_DROP_VENDOR_EVENT, &qca->flags);
|
|
|
|
}
|
|
|
|
|
2018-08-03 20:16:29 +08:00
|
|
|
qca_baudrate = qca_get_baudrate_value(speed);
|
2018-08-03 20:16:32 +08:00
|
|
|
bt_dev_dbg(hu->hdev, "Set UART speed to %d", speed);
|
2018-08-03 20:16:29 +08:00
|
|
|
ret = qca_set_baudrate(hu->hdev, qca_baudrate);
|
|
|
|
if (ret)
|
2019-02-04 23:06:42 +08:00
|
|
|
goto error;
|
2018-08-03 20:16:29 +08:00
|
|
|
|
|
|
|
host_set_baudrate(hu, speed);
|
2019-02-04 23:06:42 +08:00
|
|
|
|
|
|
|
error:
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(soc_type) ||
|
|
|
|
qca_is_wcn6750(soc_type))
|
2019-02-04 23:06:42 +08:00
|
|
|
hci_uart_set_flow_control(hu, false);
|
2019-05-22 03:53:07 +08:00
|
|
|
|
|
|
|
if (soc_type == QCA_WCN3990) {
|
|
|
|
/* Wait for the controller to send the vendor event
|
|
|
|
* for the baudrate change command.
|
|
|
|
*/
|
|
|
|
if (!wait_for_completion_timeout(&qca->drop_ev_comp,
|
|
|
|
msecs_to_jiffies(100))) {
|
|
|
|
bt_dev_err(hu->hdev,
|
|
|
|
"Failed to change controller baudrate\n");
|
|
|
|
ret = -ETIMEDOUT;
|
|
|
|
}
|
|
|
|
|
|
|
|
clear_bit(QCA_DROP_VENDOR_EVENT, &qca->flags);
|
|
|
|
}
|
2018-08-03 20:16:29 +08:00
|
|
|
}
|
|
|
|
|
2019-02-04 23:06:42 +08:00
|
|
|
return ret;
|
2018-08-03 20:16:29 +08:00
|
|
|
}
|
|
|
|
|
2020-01-02 22:49:11 +08:00
|
|
|
static int qca_send_crashbuffer(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
struct sk_buff *skb;
|
|
|
|
|
|
|
|
skb = bt_skb_alloc(QCA_CRASHBYTE_PACKET_LEN, GFP_KERNEL);
|
|
|
|
if (!skb) {
|
|
|
|
bt_dev_err(hu->hdev, "Failed to allocate memory for skb packet");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We forcefully crash the controller, by sending 0xfb byte for
|
|
|
|
* 1024 times. We also might have chance of losing data, To be
|
|
|
|
* on safer side we send 1096 bytes to the SoC.
|
|
|
|
*/
|
|
|
|
memset(skb_put(skb, QCA_CRASHBYTE_PACKET_LEN), QCA_MEMDUMP_BYTE,
|
|
|
|
QCA_CRASHBYTE_PACKET_LEN);
|
|
|
|
hci_skb_pkt_type(skb) = HCI_COMMAND_PKT;
|
|
|
|
bt_dev_info(hu->hdev, "crash the soc to collect controller dump");
|
|
|
|
skb_queue_tail(&qca->txq, skb);
|
|
|
|
hci_uart_tx_wakeup(hu);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_wait_for_dump_collection(struct hci_dev *hdev)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
wait_on_bit_timeout(&qca->flags, QCA_MEMDUMP_COLLECTION,
|
|
|
|
TASK_UNINTERRUPTIBLE, MEMDUMP_TIMEOUT_MS);
|
|
|
|
|
|
|
|
clear_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_hw_error(struct hci_dev *hdev, u8 code)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
2020-07-11 19:31:12 +08:00
|
|
|
set_bit(QCA_SSR_TRIGGERED, &qca->flags);
|
2020-02-15 00:17:15 +08:00
|
|
|
set_bit(QCA_HW_ERROR_EVENT, &qca->flags);
|
2020-01-02 22:49:11 +08:00
|
|
|
bt_dev_info(hdev, "mem_dump_status: %d", qca->memdump_state);
|
|
|
|
|
|
|
|
if (qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
|
|
|
/* If hardware error event received for other than QCA
|
|
|
|
* soc memory dump event, then we need to crash the SOC
|
|
|
|
* and wait here for 8 seconds to get the dump packets.
|
|
|
|
* This will block main thread to be on hold until we
|
|
|
|
* collect dump.
|
|
|
|
*/
|
|
|
|
set_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
|
|
|
qca_send_crashbuffer(hu);
|
|
|
|
qca_wait_for_dump_collection(hdev);
|
|
|
|
} else if (qca->memdump_state == QCA_MEMDUMP_COLLECTING) {
|
|
|
|
/* Let us wait here until memory dump collected or
|
|
|
|
* memory dump timer expired.
|
|
|
|
*/
|
|
|
|
bt_dev_info(hdev, "waiting for dump to complete");
|
|
|
|
qca_wait_for_dump_collection(hdev);
|
|
|
|
}
|
2020-02-15 00:17:15 +08:00
|
|
|
|
2020-06-09 23:57:08 +08:00
|
|
|
mutex_lock(&qca->hci_memdump_lock);
|
2020-02-15 00:17:15 +08:00
|
|
|
if (qca->memdump_state != QCA_MEMDUMP_COLLECTED) {
|
|
|
|
bt_dev_err(hu->hdev, "clearing allocated memory due to memdump timeout");
|
2020-06-09 23:57:08 +08:00
|
|
|
if (qca->qca_memdump) {
|
|
|
|
vfree(qca->qca_memdump->memdump_buf_head);
|
|
|
|
kfree(qca->qca_memdump);
|
|
|
|
qca->qca_memdump = NULL;
|
|
|
|
}
|
2020-02-15 00:17:15 +08:00
|
|
|
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
|
|
|
cancel_delayed_work(&qca->ctrl_memdump_timeout);
|
2020-06-09 23:57:08 +08:00
|
|
|
}
|
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
|
|
|
|
|
|
|
if (qca->memdump_state == QCA_MEMDUMP_TIMEOUT ||
|
|
|
|
qca->memdump_state == QCA_MEMDUMP_COLLECTED) {
|
2020-02-15 00:17:15 +08:00
|
|
|
cancel_work_sync(&qca->ctrl_memdump_evt);
|
2020-06-09 23:57:08 +08:00
|
|
|
skb_queue_purge(&qca->rx_memdump_q);
|
2020-02-15 00:17:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
clear_bit(QCA_HW_ERROR_EVENT, &qca->flags);
|
2020-01-02 22:49:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_cmd_timeout(struct hci_dev *hdev)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
2020-07-11 19:31:12 +08:00
|
|
|
set_bit(QCA_SSR_TRIGGERED, &qca->flags);
|
|
|
|
if (qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
|
|
|
set_bit(QCA_MEMDUMP_COLLECTION, &qca->flags);
|
2020-01-02 22:49:11 +08:00
|
|
|
qca_send_crashbuffer(hu);
|
2020-07-11 19:31:12 +08:00
|
|
|
qca_wait_for_dump_collection(hdev);
|
|
|
|
} else if (qca->memdump_state == QCA_MEMDUMP_COLLECTING) {
|
|
|
|
/* Let us wait here until memory dump collected or
|
|
|
|
* memory dump timer expired.
|
|
|
|
*/
|
|
|
|
bt_dev_info(hdev, "waiting for dump to complete");
|
|
|
|
qca_wait_for_dump_collection(hdev);
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_lock(&qca->hci_memdump_lock);
|
|
|
|
if (qca->memdump_state != QCA_MEMDUMP_COLLECTED) {
|
|
|
|
qca->memdump_state = QCA_MEMDUMP_TIMEOUT;
|
|
|
|
if (!test_bit(QCA_HW_ERROR_EVENT, &qca->flags)) {
|
|
|
|
/* Inject hw error event to reset the device
|
|
|
|
* and driver.
|
|
|
|
*/
|
|
|
|
hci_reset_dev(hu->hdev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock(&qca->hci_memdump_lock);
|
2020-01-02 22:49:11 +08:00
|
|
|
}
|
|
|
|
|
2021-10-02 04:22:31 +08:00
|
|
|
static bool qca_wakeup(struct hci_dev *hdev)
|
2021-03-11 15:03:40 +08:00
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
bool wakeup;
|
|
|
|
|
|
|
|
/* UART driver handles the interrupt from BT SoC.So we need to use
|
|
|
|
* device handle of UART driver to get the status of device may wakeup.
|
|
|
|
*/
|
|
|
|
wakeup = device_may_wakeup(hu->serdev->ctrl->dev.parent);
|
|
|
|
bt_dev_dbg(hu->hdev, "wakeup status : %d", wakeup);
|
|
|
|
|
|
|
|
return !wakeup;
|
|
|
|
}
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
static int qca_regulator_init(struct hci_uart *hu)
|
2018-08-03 20:16:32 +08:00
|
|
|
{
|
2021-05-19 00:34:42 +08:00
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
2018-09-24 22:44:45 +08:00
|
|
|
struct qca_serdev *qcadev;
|
2018-08-03 20:16:32 +08:00
|
|
|
int ret;
|
2021-05-19 00:34:42 +08:00
|
|
|
bool sw_ctrl_state;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2018-09-24 22:44:45 +08:00
|
|
|
/* Check for vregs status, may be hci down has turned
|
|
|
|
* off the voltage regulator.
|
|
|
|
*/
|
|
|
|
qcadev = serdev_device_get_drvdata(hu->serdev);
|
|
|
|
if (!qcadev->bt_power->vregs_on) {
|
|
|
|
serdev_device_close(hu->serdev);
|
2019-10-18 13:24:04 +08:00
|
|
|
ret = qca_regulator_enable(qcadev);
|
2018-09-24 22:44:45 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = serdev_device_open(hu->serdev);
|
|
|
|
if (ret) {
|
|
|
|
bt_dev_err(hu->hdev, "failed to open port");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(soc_type)) {
|
|
|
|
/* Forcefully enable wcn399x to enter in to boot mode. */
|
|
|
|
host_set_baudrate(hu, 2400);
|
|
|
|
ret = qca_send_power_pulse(hu, false);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For wcn6750 need to enable gpio bt_en */
|
|
|
|
if (qcadev->bt_en) {
|
|
|
|
gpiod_set_value_cansleep(qcadev->bt_en, 0);
|
|
|
|
msleep(50);
|
|
|
|
gpiod_set_value_cansleep(qcadev->bt_en, 1);
|
|
|
|
msleep(50);
|
|
|
|
if (qcadev->sw_ctrl) {
|
|
|
|
sw_ctrl_state = gpiod_get_value_cansleep(qcadev->sw_ctrl);
|
|
|
|
bt_dev_dbg(hu->hdev, "SW_CTRL is %d", sw_ctrl_state);
|
|
|
|
}
|
|
|
|
}
|
2018-08-03 20:16:32 +08:00
|
|
|
|
|
|
|
qca_set_speed(hu, QCA_INIT_SPEED);
|
2021-05-19 00:34:42 +08:00
|
|
|
|
|
|
|
if (qca_is_wcn399x(soc_type)) {
|
|
|
|
ret = qca_send_power_pulse(hu, true);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
2018-08-03 20:16:32 +08:00
|
|
|
|
|
|
|
/* Now the device is in ready state to communicate with host.
|
|
|
|
* To sync host with device we need to reopen port.
|
|
|
|
* Without this, we will have RTS and CTS synchronization
|
|
|
|
* issues.
|
|
|
|
*/
|
|
|
|
serdev_device_close(hu->serdev);
|
|
|
|
ret = serdev_device_open(hu->serdev);
|
|
|
|
if (ret) {
|
|
|
|
bt_dev_err(hu->hdev, "failed to open port");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
hci_uart_set_flow_control(hu, false);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-01-13 12:30:20 +08:00
|
|
|
static int qca_power_on(struct hci_dev *hdev)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
|
|
|
struct qca_serdev *qcadev;
|
2020-10-06 23:20:21 +08:00
|
|
|
struct qca_data *qca = hu->priv;
|
2020-01-13 12:30:20 +08:00
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* Non-serdev device usually is powered by external power
|
|
|
|
* and don't need additional action in driver for power on
|
|
|
|
*/
|
|
|
|
if (!hu->serdev)
|
|
|
|
return 0;
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(soc_type) ||
|
|
|
|
qca_is_wcn6750(soc_type)) {
|
|
|
|
ret = qca_regulator_init(hu);
|
2020-01-13 12:30:20 +08:00
|
|
|
} else {
|
|
|
|
qcadev = serdev_device_get_drvdata(hu->serdev);
|
2020-03-04 21:16:45 +08:00
|
|
|
if (qcadev->bt_en) {
|
2020-03-04 09:54:29 +08:00
|
|
|
gpiod_set_value_cansleep(qcadev->bt_en, 1);
|
|
|
|
/* Controller needs time to bootup. */
|
|
|
|
msleep(150);
|
|
|
|
}
|
2020-01-13 12:30:20 +08:00
|
|
|
}
|
|
|
|
|
2020-10-06 23:20:21 +08:00
|
|
|
clear_bit(QCA_BT_OFF, &qca->flags);
|
2020-01-13 12:30:20 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
static int qca_setup(struct hci_uart *hu)
|
|
|
|
{
|
|
|
|
struct hci_dev *hdev = hu->hdev;
|
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
unsigned int speed, qca_baudrate = QCA_BAUDRATE_115200;
|
2020-01-15 16:55:51 +08:00
|
|
|
unsigned int retries = 0;
|
2019-03-12 02:38:31 +08:00
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
2019-06-06 17:40:30 +08:00
|
|
|
const char *firmware_name = qca_get_firmware_name(hu);
|
2015-08-11 05:24:17 +08:00
|
|
|
int ret;
|
2020-11-19 20:54:09 +08:00
|
|
|
struct qca_btsoc_version ver;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
2018-08-03 20:16:29 +08:00
|
|
|
ret = qca_check_speeds(hu);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2021-02-05 23:37:16 +08:00
|
|
|
clear_bit(QCA_ROM_FW, &qca->flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
/* Patch downloading has to be done without IBS mode */
|
2020-10-06 23:20:21 +08:00
|
|
|
set_bit(QCA_IBS_DISABLED, &qca->flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
|
2019-08-21 14:23:39 +08:00
|
|
|
/* Enable controller to do both LE scan and BR/EDR inquiry
|
|
|
|
* simultaneously.
|
|
|
|
*/
|
|
|
|
set_bit(HCI_QUIRK_SIMULTANEOUS_DISCOVERY, &hdev->quirks);
|
|
|
|
|
2020-01-13 12:30:20 +08:00
|
|
|
bt_dev_info(hdev, "setting up %s",
|
2021-05-19 00:34:42 +08:00
|
|
|
qca_is_wcn399x(soc_type) ? "wcn399x" :
|
|
|
|
(soc_type == QCA_WCN6750) ? "wcn6750" : "ROME/QCA6390");
|
2018-09-24 22:44:45 +08:00
|
|
|
|
2020-07-28 23:53:00 +08:00
|
|
|
qca->memdump_state = QCA_MEMDUMP_IDLE;
|
|
|
|
|
2020-01-15 16:55:51 +08:00
|
|
|
retry:
|
2020-01-13 12:30:20 +08:00
|
|
|
ret = qca_power_on(hdev);
|
|
|
|
if (ret)
|
2020-11-11 13:14:13 +08:00
|
|
|
goto out;
|
2020-01-13 12:30:20 +08:00
|
|
|
|
2020-07-11 19:31:12 +08:00
|
|
|
clear_bit(QCA_SSR_TRIGGERED, &qca->flags);
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if (qca_is_wcn399x(soc_type) ||
|
|
|
|
qca_is_wcn6750(soc_type)) {
|
2019-02-20 04:05:59 +08:00
|
|
|
set_bit(HCI_QUIRK_USE_BDADDR_PROPERTY, &hdev->quirks);
|
2021-09-26 15:07:47 +08:00
|
|
|
hci_set_aosp_capable(hdev);
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2020-11-19 20:54:09 +08:00
|
|
|
ret = qca_read_soc_version(hdev, &ver, soc_type);
|
2018-08-03 20:16:32 +08:00
|
|
|
if (ret)
|
2020-11-11 13:14:13 +08:00
|
|
|
goto out;
|
2018-08-03 20:16:32 +08:00
|
|
|
} else {
|
|
|
|
qca_set_speed(hu, QCA_INIT_SPEED);
|
|
|
|
}
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
/* Setup user speed if needed */
|
2018-08-03 20:16:29 +08:00
|
|
|
speed = qca_get_speed(hu, QCA_OPER_SPEED);
|
2015-08-11 05:24:17 +08:00
|
|
|
if (speed) {
|
2018-08-03 20:16:29 +08:00
|
|
|
ret = qca_set_speed(hu, QCA_OPER_SPEED);
|
|
|
|
if (ret)
|
2020-11-11 13:14:13 +08:00
|
|
|
goto out;
|
2018-08-03 20:16:29 +08:00
|
|
|
|
|
|
|
qca_baudrate = qca_get_baudrate_value(speed);
|
2015-08-11 05:24:17 +08:00
|
|
|
}
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if (!(qca_is_wcn399x(soc_type) ||
|
|
|
|
qca_is_wcn6750(soc_type))) {
|
2018-08-03 20:16:32 +08:00
|
|
|
/* Get QCA version information */
|
2020-11-19 20:54:09 +08:00
|
|
|
ret = qca_read_soc_version(hdev, &ver, soc_type);
|
2018-08-03 20:16:32 +08:00
|
|
|
if (ret)
|
2020-11-11 13:14:13 +08:00
|
|
|
goto out;
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
2018-08-03 20:16:28 +08:00
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
/* Setup patch / NVM configurations */
|
2020-11-19 20:54:09 +08:00
|
|
|
ret = qca_uart_setup(hdev, qca_baudrate, soc_type, ver,
|
2019-06-06 17:40:30 +08:00
|
|
|
firmware_name);
|
2015-08-11 05:24:17 +08:00
|
|
|
if (!ret) {
|
2020-10-06 23:20:21 +08:00
|
|
|
clear_bit(QCA_IBS_DISABLED, &qca->flags);
|
2015-08-11 05:24:17 +08:00
|
|
|
qca_debugfs_init(hdev);
|
2020-01-02 22:49:11 +08:00
|
|
|
hu->hdev->hw_error = qca_hw_error;
|
|
|
|
hu->hdev->cmd_timeout = qca_cmd_timeout;
|
2021-10-02 04:22:31 +08:00
|
|
|
hu->hdev->wakeup = qca_wakeup;
|
2017-11-06 19:16:56 +08:00
|
|
|
} else if (ret == -ENOENT) {
|
|
|
|
/* No patch/nvm-config found, run with original fw/config */
|
2021-02-05 23:37:16 +08:00
|
|
|
set_bit(QCA_ROM_FW, &qca->flags);
|
2017-11-06 19:16:56 +08:00
|
|
|
ret = 0;
|
2018-04-16 14:40:24 +08:00
|
|
|
} else if (ret == -EAGAIN) {
|
|
|
|
/*
|
|
|
|
* Userspace firmware loader will return -EAGAIN in case no
|
|
|
|
* patch/nvm-config is found, so run with original fw/config.
|
|
|
|
*/
|
2021-02-05 23:37:16 +08:00
|
|
|
set_bit(QCA_ROM_FW, &qca->flags);
|
2018-04-16 14:40:24 +08:00
|
|
|
ret = 0;
|
2020-11-11 13:14:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (ret && retries < MAX_INIT_RETRIES) {
|
|
|
|
bt_dev_warn(hdev, "Retry BT power ON:%d", retries);
|
|
|
|
qca_power_shutdown(hu);
|
|
|
|
if (hu->serdev) {
|
|
|
|
serdev_device_close(hu->serdev);
|
|
|
|
ret = serdev_device_open(hu->serdev);
|
|
|
|
if (ret) {
|
|
|
|
bt_dev_err(hdev, "failed to open port");
|
|
|
|
return ret;
|
2020-01-15 16:55:51 +08:00
|
|
|
}
|
|
|
|
}
|
2020-11-11 13:14:13 +08:00
|
|
|
retries++;
|
|
|
|
goto retry;
|
2015-08-11 05:24:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Setup bdaddr */
|
2020-03-25 10:26:37 +08:00
|
|
|
if (soc_type == QCA_ROME)
|
2019-01-16 20:31:15 +08:00
|
|
|
hu->hdev->set_bdaddr = qca_set_bdaddr_rome;
|
2020-03-25 10:26:37 +08:00
|
|
|
else
|
|
|
|
hu->hdev->set_bdaddr = qca_set_bdaddr;
|
2015-08-11 05:24:17 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-08-15 13:51:49 +08:00
|
|
|
static const struct hci_uart_proto qca_proto = {
|
2015-08-11 05:24:17 +08:00
|
|
|
.id = HCI_UART_QCA,
|
|
|
|
.name = "QCA",
|
2015-10-21 03:30:45 +08:00
|
|
|
.manufacturer = 29,
|
2015-08-11 05:24:17 +08:00
|
|
|
.init_speed = 115200,
|
|
|
|
.oper_speed = 3000000,
|
|
|
|
.open = qca_open,
|
|
|
|
.close = qca_close,
|
|
|
|
.flush = qca_flush,
|
|
|
|
.setup = qca_setup,
|
|
|
|
.recv = qca_recv,
|
|
|
|
.enqueue = qca_enqueue,
|
|
|
|
.dequeue = qca_dequeue,
|
|
|
|
};
|
|
|
|
|
2020-05-15 04:14:04 +08:00
|
|
|
static const struct qca_device_data qca_soc_data_wcn3990 = {
|
2018-08-03 20:16:32 +08:00
|
|
|
.soc_type = QCA_WCN3990,
|
|
|
|
.vregs = (struct qca_vreg []) {
|
2019-10-18 13:24:02 +08:00
|
|
|
{ "vddio", 15000 },
|
|
|
|
{ "vddxo", 80000 },
|
|
|
|
{ "vddrf", 300000 },
|
|
|
|
{ "vddch0", 450000 },
|
2018-08-03 20:16:32 +08:00
|
|
|
},
|
|
|
|
.num_vregs = 4,
|
|
|
|
};
|
|
|
|
|
2020-05-15 04:14:04 +08:00
|
|
|
static const struct qca_device_data qca_soc_data_wcn3991 = {
|
2019-11-06 17:48:32 +08:00
|
|
|
.soc_type = QCA_WCN3991,
|
|
|
|
.vregs = (struct qca_vreg []) {
|
|
|
|
{ "vddio", 15000 },
|
|
|
|
{ "vddxo", 80000 },
|
|
|
|
{ "vddrf", 300000 },
|
|
|
|
{ "vddch0", 450000 },
|
|
|
|
},
|
|
|
|
.num_vregs = 4,
|
2020-10-01 04:01:38 +08:00
|
|
|
.capabilities = QCA_CAP_WIDEBAND_SPEECH | QCA_CAP_VALID_LE_STATES,
|
2019-11-06 17:48:32 +08:00
|
|
|
};
|
|
|
|
|
2020-05-15 04:14:04 +08:00
|
|
|
static const struct qca_device_data qca_soc_data_wcn3998 = {
|
2019-04-26 21:56:01 +08:00
|
|
|
.soc_type = QCA_WCN3998,
|
|
|
|
.vregs = (struct qca_vreg []) {
|
2019-10-18 13:24:02 +08:00
|
|
|
{ "vddio", 10000 },
|
|
|
|
{ "vddxo", 80000 },
|
|
|
|
{ "vddrf", 300000 },
|
|
|
|
{ "vddch0", 450000 },
|
2019-04-26 21:56:01 +08:00
|
|
|
},
|
|
|
|
.num_vregs = 4,
|
|
|
|
};
|
|
|
|
|
2020-05-15 04:14:04 +08:00
|
|
|
static const struct qca_device_data qca_soc_data_qca6390 = {
|
2020-03-25 10:26:37 +08:00
|
|
|
.soc_type = QCA_QCA6390,
|
|
|
|
.num_vregs = 0,
|
|
|
|
};
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
static const struct qca_device_data qca_soc_data_wcn6750 = {
|
|
|
|
.soc_type = QCA_WCN6750,
|
|
|
|
.vregs = (struct qca_vreg []) {
|
|
|
|
{ "vddio", 5000 },
|
|
|
|
{ "vddaon", 26000 },
|
|
|
|
{ "vddbtcxmx", 126000 },
|
|
|
|
{ "vddrfacmn", 12500 },
|
|
|
|
{ "vddrfa0p8", 102000 },
|
|
|
|
{ "vddrfa1p7", 302000 },
|
|
|
|
{ "vddrfa1p2", 257000 },
|
|
|
|
{ "vddrfa2p2", 1700000 },
|
|
|
|
{ "vddasd", 200 },
|
|
|
|
},
|
|
|
|
.num_vregs = 9,
|
|
|
|
.capabilities = QCA_CAP_WIDEBAND_SPEECH | QCA_CAP_VALID_LE_STATES,
|
|
|
|
};
|
|
|
|
|
2018-08-22 20:20:05 +08:00
|
|
|
static void qca_power_shutdown(struct hci_uart *hu)
|
2018-08-03 20:16:32 +08:00
|
|
|
{
|
2019-10-18 13:24:04 +08:00
|
|
|
struct qca_serdev *qcadev;
|
2019-02-04 23:06:43 +08:00
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
unsigned long flags;
|
2020-01-15 16:55:50 +08:00
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
2021-05-19 00:34:42 +08:00
|
|
|
bool sw_ctrl_state;
|
2019-02-04 23:06:43 +08:00
|
|
|
|
|
|
|
/* From this point we go into power off state. But serial port is
|
|
|
|
* still open, stop queueing the IBS data and flush all the buffered
|
|
|
|
* data in skb's.
|
|
|
|
*/
|
|
|
|
spin_lock_irqsave(&qca->hci_ibs_lock, flags);
|
2020-10-06 23:20:21 +08:00
|
|
|
set_bit(QCA_IBS_DISABLED, &qca->flags);
|
2019-02-04 23:06:43 +08:00
|
|
|
qca_flush(hu);
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
|
2020-01-15 16:55:50 +08:00
|
|
|
/* Non-serdev device usually is powered by external power
|
|
|
|
* and don't need additional action in driver for power down
|
|
|
|
*/
|
|
|
|
if (!hu->serdev)
|
|
|
|
return;
|
|
|
|
|
2021-05-03 18:06:05 +08:00
|
|
|
qcadev = serdev_device_get_drvdata(hu->serdev);
|
|
|
|
|
2020-01-15 16:55:50 +08:00
|
|
|
if (qca_is_wcn399x(soc_type)) {
|
|
|
|
host_set_baudrate(hu, 2400);
|
|
|
|
qca_send_power_pulse(hu, false);
|
|
|
|
qca_regulator_disable(qcadev);
|
2021-05-19 00:34:42 +08:00
|
|
|
} else if (soc_type == QCA_WCN6750) {
|
|
|
|
gpiod_set_value_cansleep(qcadev->bt_en, 0);
|
|
|
|
msleep(100);
|
|
|
|
qca_regulator_disable(qcadev);
|
|
|
|
if (qcadev->sw_ctrl) {
|
|
|
|
sw_ctrl_state = gpiod_get_value_cansleep(qcadev->sw_ctrl);
|
|
|
|
bt_dev_dbg(hu->hdev, "SW_CTRL is %d", sw_ctrl_state);
|
|
|
|
}
|
2020-03-04 21:16:45 +08:00
|
|
|
} else if (qcadev->bt_en) {
|
2020-01-15 16:55:50 +08:00
|
|
|
gpiod_set_value_cansleep(qcadev->bt_en, 0);
|
|
|
|
}
|
2020-10-06 23:20:21 +08:00
|
|
|
|
|
|
|
set_bit(QCA_BT_OFF, &qca->flags);
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
|
|
|
|
2018-09-24 22:44:45 +08:00
|
|
|
static int qca_power_off(struct hci_dev *hdev)
|
|
|
|
{
|
|
|
|
struct hci_uart *hu = hci_get_drvdata(hdev);
|
2020-01-02 22:49:11 +08:00
|
|
|
struct qca_data *qca = hu->priv;
|
2020-02-29 20:21:18 +08:00
|
|
|
enum qca_btsoc_type soc_type = qca_soc_type(hu);
|
2018-09-24 22:44:45 +08:00
|
|
|
|
2020-07-28 23:53:00 +08:00
|
|
|
hu->hdev->hw_error = NULL;
|
|
|
|
hu->hdev->cmd_timeout = NULL;
|
|
|
|
|
2021-12-22 15:29:05 +08:00
|
|
|
del_timer_sync(&qca->wake_retrans_timer);
|
|
|
|
del_timer_sync(&qca->tx_idle_timer);
|
|
|
|
|
2020-01-02 22:49:11 +08:00
|
|
|
/* Stop sending shutdown command if soc crashes. */
|
2020-03-25 10:26:37 +08:00
|
|
|
if (soc_type != QCA_ROME
|
2020-02-29 20:21:18 +08:00
|
|
|
&& qca->memdump_state == QCA_MEMDUMP_IDLE) {
|
2020-01-02 22:49:11 +08:00
|
|
|
qca_send_pre_shutdown_cmd(hdev);
|
|
|
|
usleep_range(8000, 10000);
|
|
|
|
}
|
2019-08-30 20:28:36 +08:00
|
|
|
|
2018-09-24 22:44:45 +08:00
|
|
|
qca_power_shutdown(hu);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
static int qca_regulator_enable(struct qca_serdev *qcadev)
|
2018-08-03 20:16:32 +08:00
|
|
|
{
|
2019-10-18 13:24:04 +08:00
|
|
|
struct qca_power *power = qcadev->bt_power;
|
|
|
|
int ret;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
/* Already enabled */
|
|
|
|
if (power->vregs_on)
|
|
|
|
return 0;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
BT_DBG("enabling %d regulators)", power->num_vregs);
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
ret = regulator_bulk_enable(power->num_vregs, power->vreg_bulk);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
power->vregs_on = true;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2020-02-03 18:40:40 +08:00
|
|
|
ret = clk_prepare_enable(qcadev->susclk);
|
2020-02-05 18:51:43 +08:00
|
|
|
if (ret)
|
2020-02-03 18:40:40 +08:00
|
|
|
qca_regulator_disable(qcadev);
|
|
|
|
|
2020-02-05 18:51:43 +08:00
|
|
|
return ret;
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
|
|
|
|
2019-10-18 13:24:04 +08:00
|
|
|
static void qca_regulator_disable(struct qca_serdev *qcadev)
|
|
|
|
{
|
|
|
|
struct qca_power *power;
|
|
|
|
|
|
|
|
if (!qcadev)
|
|
|
|
return;
|
|
|
|
|
|
|
|
power = qcadev->bt_power;
|
|
|
|
|
|
|
|
/* Already disabled? */
|
|
|
|
if (!power->vregs_on)
|
|
|
|
return;
|
|
|
|
|
|
|
|
regulator_bulk_disable(power->num_vregs, power->vreg_bulk);
|
|
|
|
power->vregs_on = false;
|
2020-02-03 18:40:40 +08:00
|
|
|
|
2020-02-05 18:51:43 +08:00
|
|
|
clk_disable_unprepare(qcadev->susclk);
|
2019-10-18 13:24:04 +08:00
|
|
|
}
|
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
static int qca_init_regulators(struct qca_power *qca,
|
|
|
|
const struct qca_vreg *vregs, size_t num_vregs)
|
|
|
|
{
|
2019-10-18 13:24:01 +08:00
|
|
|
struct regulator_bulk_data *bulk;
|
|
|
|
int ret;
|
2018-08-03 20:16:32 +08:00
|
|
|
int i;
|
|
|
|
|
2019-10-18 13:24:01 +08:00
|
|
|
bulk = devm_kcalloc(qca->dev, num_vregs, sizeof(*bulk), GFP_KERNEL);
|
|
|
|
if (!bulk)
|
2018-08-03 20:16:32 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for (i = 0; i < num_vregs; i++)
|
2019-10-18 13:24:01 +08:00
|
|
|
bulk[i].supply = vregs[i].name;
|
|
|
|
|
|
|
|
ret = devm_regulator_bulk_get(qca->dev, num_vregs, bulk);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
2019-10-18 13:24:01 +08:00
|
|
|
for (i = 0; i < num_vregs; i++) {
|
|
|
|
ret = regulator_set_load(bulk[i].consumer, vregs[i].load_uA);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
qca->vreg_bulk = bulk;
|
2019-10-18 13:24:03 +08:00
|
|
|
qca->num_vregs = num_vregs;
|
2019-10-18 13:24:01 +08:00
|
|
|
|
|
|
|
return 0;
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
|
|
|
|
2018-03-30 03:15:24 +08:00
|
|
|
static int qca_serdev_probe(struct serdev_device *serdev)
|
|
|
|
{
|
|
|
|
struct qca_serdev *qcadev;
|
2020-01-16 11:22:54 +08:00
|
|
|
struct hci_dev *hdev;
|
2020-05-15 04:14:04 +08:00
|
|
|
const struct qca_device_data *data;
|
2018-03-30 03:15:24 +08:00
|
|
|
int err;
|
2020-03-04 09:54:29 +08:00
|
|
|
bool power_ctrl_enabled = true;
|
2018-03-30 03:15:24 +08:00
|
|
|
|
|
|
|
qcadev = devm_kzalloc(&serdev->dev, sizeof(*qcadev), GFP_KERNEL);
|
|
|
|
if (!qcadev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
qcadev->serdev_hu.serdev = serdev;
|
2019-12-13 16:50:45 +08:00
|
|
|
data = device_get_match_data(&serdev->dev);
|
2018-03-30 03:15:24 +08:00
|
|
|
serdev_device_set_drvdata(serdev, qcadev);
|
2019-06-06 17:40:30 +08:00
|
|
|
device_property_read_string(&serdev->dev, "firmware-name",
|
|
|
|
&qcadev->firmware_name);
|
2020-04-23 09:34:30 +08:00
|
|
|
device_property_read_u32(&serdev->dev, "max-speed",
|
|
|
|
&qcadev->oper_speed);
|
|
|
|
if (!qcadev->oper_speed)
|
|
|
|
BT_DBG("UART will pick default operating speed");
|
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if (data &&
|
|
|
|
(qca_is_wcn399x(data->soc_type) ||
|
|
|
|
qca_is_wcn6750(data->soc_type))) {
|
2019-04-26 21:56:01 +08:00
|
|
|
qcadev->btsoc_type = data->soc_type;
|
2018-08-03 20:16:32 +08:00
|
|
|
qcadev->bt_power = devm_kzalloc(&serdev->dev,
|
|
|
|
sizeof(struct qca_power),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!qcadev->bt_power)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
qcadev->bt_power->dev = &serdev->dev;
|
|
|
|
err = qca_init_regulators(qcadev->bt_power, data->vregs,
|
|
|
|
data->num_vregs);
|
|
|
|
if (err) {
|
|
|
|
BT_ERR("Failed to init regulators:%d", err);
|
2020-01-16 11:22:54 +08:00
|
|
|
return err;
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
2018-03-30 03:15:24 +08:00
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
qcadev->bt_power->vregs_on = false;
|
2018-03-30 03:15:24 +08:00
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
qcadev->bt_en = devm_gpiod_get_optional(&serdev->dev, "enable",
|
|
|
|
GPIOD_OUT_LOW);
|
2021-12-24 16:02:49 +08:00
|
|
|
if (IS_ERR_OR_NULL(qcadev->bt_en) && data->soc_type == QCA_WCN6750) {
|
2021-05-19 00:34:42 +08:00
|
|
|
dev_err(&serdev->dev, "failed to acquire BT_EN gpio\n");
|
|
|
|
power_ctrl_enabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
qcadev->sw_ctrl = devm_gpiod_get_optional(&serdev->dev, "swctrl",
|
|
|
|
GPIOD_IN);
|
2021-12-24 16:02:49 +08:00
|
|
|
if (IS_ERR_OR_NULL(qcadev->sw_ctrl) && data->soc_type == QCA_WCN6750)
|
2021-05-19 00:34:42 +08:00
|
|
|
dev_warn(&serdev->dev, "failed to acquire SW_CTRL gpio\n");
|
|
|
|
|
2020-02-03 18:40:40 +08:00
|
|
|
qcadev->susclk = devm_clk_get_optional(&serdev->dev, NULL);
|
|
|
|
if (IS_ERR(qcadev->susclk)) {
|
|
|
|
dev_err(&serdev->dev, "failed to acquire clk\n");
|
|
|
|
return PTR_ERR(qcadev->susclk);
|
|
|
|
}
|
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
err = hci_uart_register_device(&qcadev->serdev_hu, &qca_proto);
|
|
|
|
if (err) {
|
|
|
|
BT_ERR("wcn3990 serdev registration failed");
|
2020-01-16 11:22:54 +08:00
|
|
|
return err;
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
|
|
|
} else {
|
2020-03-25 10:26:37 +08:00
|
|
|
if (data)
|
|
|
|
qcadev->btsoc_type = data->soc_type;
|
|
|
|
else
|
|
|
|
qcadev->btsoc_type = QCA_ROME;
|
|
|
|
|
2020-03-04 21:16:45 +08:00
|
|
|
qcadev->bt_en = devm_gpiod_get_optional(&serdev->dev, "enable",
|
2018-08-03 20:16:32 +08:00
|
|
|
GPIOD_OUT_LOW);
|
2021-12-24 16:02:49 +08:00
|
|
|
if (IS_ERR_OR_NULL(qcadev->bt_en)) {
|
2020-03-04 09:54:29 +08:00
|
|
|
dev_warn(&serdev->dev, "failed to acquire enable gpio\n");
|
|
|
|
power_ctrl_enabled = false;
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
2018-03-30 03:15:24 +08:00
|
|
|
|
2020-03-04 21:16:45 +08:00
|
|
|
qcadev->susclk = devm_clk_get_optional(&serdev->dev, NULL);
|
2020-05-29 17:59:48 +08:00
|
|
|
if (IS_ERR(qcadev->susclk)) {
|
2020-03-04 09:54:29 +08:00
|
|
|
dev_warn(&serdev->dev, "failed to acquire clk\n");
|
2020-05-29 17:59:48 +08:00
|
|
|
return PTR_ERR(qcadev->susclk);
|
2020-03-04 09:54:29 +08:00
|
|
|
}
|
2020-05-29 17:59:48 +08:00
|
|
|
err = clk_set_rate(qcadev->susclk, SUSCLK_RATE_32KHZ);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = clk_prepare_enable(qcadev->susclk);
|
|
|
|
if (err)
|
|
|
|
return err;
|
2018-08-03 20:16:32 +08:00
|
|
|
|
|
|
|
err = hci_uart_register_device(&qcadev->serdev_hu, &qca_proto);
|
2020-01-16 11:22:54 +08:00
|
|
|
if (err) {
|
|
|
|
BT_ERR("Rome serdev registration failed");
|
2020-09-03 16:27:19 +08:00
|
|
|
clk_disable_unprepare(qcadev->susclk);
|
2020-01-16 11:22:54 +08:00
|
|
|
return err;
|
|
|
|
}
|
2018-08-03 20:16:32 +08:00
|
|
|
}
|
|
|
|
|
2020-05-21 00:32:28 +08:00
|
|
|
hdev = qcadev->serdev_hu.hdev;
|
|
|
|
|
2020-03-04 09:54:29 +08:00
|
|
|
if (power_ctrl_enabled) {
|
|
|
|
set_bit(HCI_QUIRK_NON_PERSISTENT_SETUP, &hdev->quirks);
|
|
|
|
hdev->shutdown = qca_power_off;
|
|
|
|
}
|
2018-03-30 03:15:24 +08:00
|
|
|
|
2020-10-01 04:01:38 +08:00
|
|
|
if (data) {
|
|
|
|
/* Wideband speech support must be set per driver since it can't
|
|
|
|
* be queried via hci. Same with the valid le states quirk.
|
|
|
|
*/
|
|
|
|
if (data->capabilities & QCA_CAP_WIDEBAND_SPEECH)
|
|
|
|
set_bit(HCI_QUIRK_WIDEBAND_SPEECH_SUPPORTED,
|
|
|
|
&hdev->quirks);
|
|
|
|
|
|
|
|
if (data->capabilities & QCA_CAP_VALID_LE_STATES)
|
|
|
|
set_bit(HCI_QUIRK_VALID_LE_STATES, &hdev->quirks);
|
|
|
|
}
|
2020-05-15 04:14:04 +08:00
|
|
|
|
2020-01-16 11:22:54 +08:00
|
|
|
return 0;
|
2018-03-30 03:15:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void qca_serdev_remove(struct serdev_device *serdev)
|
|
|
|
{
|
|
|
|
struct qca_serdev *qcadev = serdev_device_get_drvdata(serdev);
|
2020-09-10 18:30:43 +08:00
|
|
|
struct qca_power *power = qcadev->bt_power;
|
2018-03-30 03:15:24 +08:00
|
|
|
|
2021-05-19 00:34:42 +08:00
|
|
|
if ((qca_is_wcn399x(qcadev->btsoc_type) ||
|
|
|
|
qca_is_wcn6750(qcadev->btsoc_type)) &&
|
|
|
|
power->vregs_on)
|
2018-08-22 20:20:05 +08:00
|
|
|
qca_power_shutdown(&qcadev->serdev_hu);
|
2020-03-04 21:16:45 +08:00
|
|
|
else if (qcadev->susclk)
|
2018-08-03 20:16:32 +08:00
|
|
|
clk_disable_unprepare(qcadev->susclk);
|
2018-03-30 03:15:24 +08:00
|
|
|
|
2018-08-03 20:16:32 +08:00
|
|
|
hci_uart_unregister_device(&qcadev->serdev_hu);
|
2018-03-30 03:15:24 +08:00
|
|
|
}
|
|
|
|
|
2020-05-29 21:56:57 +08:00
|
|
|
static void qca_serdev_shutdown(struct device *dev)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int timeout = msecs_to_jiffies(CMD_TRANS_TIMEOUT_MS);
|
|
|
|
struct serdev_device *serdev = to_serdev_device(dev);
|
|
|
|
struct qca_serdev *qcadev = serdev_device_get_drvdata(serdev);
|
|
|
|
const u8 ibs_wake_cmd[] = { 0xFD };
|
|
|
|
const u8 edl_reset_soc_cmd[] = { 0x01, 0x00, 0xFC, 0x01, 0x05 };
|
|
|
|
|
|
|
|
if (qcadev->btsoc_type == QCA_QCA6390) {
|
|
|
|
serdev_device_write_flush(serdev);
|
|
|
|
ret = serdev_device_write_buf(serdev, ibs_wake_cmd,
|
|
|
|
sizeof(ibs_wake_cmd));
|
|
|
|
if (ret < 0) {
|
|
|
|
BT_ERR("QCA send IBS_WAKE_IND error: %d", ret);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
serdev_device_wait_until_sent(serdev, timeout);
|
|
|
|
usleep_range(8000, 10000);
|
|
|
|
|
|
|
|
serdev_device_write_flush(serdev);
|
|
|
|
ret = serdev_device_write_buf(serdev, edl_reset_soc_cmd,
|
|
|
|
sizeof(edl_reset_soc_cmd));
|
|
|
|
if (ret < 0) {
|
|
|
|
BT_ERR("QCA send EDL_RESET_REQ error: %d", ret);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
serdev_device_wait_until_sent(serdev, timeout);
|
|
|
|
usleep_range(8000, 10000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
static int __maybe_unused qca_suspend(struct device *dev)
|
|
|
|
{
|
2020-05-29 04:31:07 +08:00
|
|
|
struct serdev_device *serdev = to_serdev_device(dev);
|
|
|
|
struct qca_serdev *qcadev = serdev_device_get_drvdata(serdev);
|
|
|
|
struct hci_uart *hu = &qcadev->serdev_hu;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
unsigned long flags;
|
2020-06-06 02:46:10 +08:00
|
|
|
bool tx_pending = false;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
int ret = 0;
|
|
|
|
u8 cmd;
|
2020-10-06 23:20:21 +08:00
|
|
|
u32 wait_timeout = 0;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
|
|
|
|
set_bit(QCA_SUSPENDING, &qca->flags);
|
|
|
|
|
2021-02-05 23:37:16 +08:00
|
|
|
/* if BT SoC is running with default firmware then it does not
|
|
|
|
* support in-band sleep
|
|
|
|
*/
|
|
|
|
if (test_bit(QCA_ROM_FW, &qca->flags))
|
|
|
|
return 0;
|
|
|
|
|
2020-12-31 01:17:08 +08:00
|
|
|
/* During SSR after memory dump collection, controller will be
|
|
|
|
* powered off and then powered on.If controller is powered off
|
|
|
|
* during SSR then we should wait until SSR is completed.
|
|
|
|
*/
|
|
|
|
if (test_bit(QCA_BT_OFF, &qca->flags) &&
|
|
|
|
!test_bit(QCA_SSR_TRIGGERED, &qca->flags))
|
2020-10-06 23:20:21 +08:00
|
|
|
return 0;
|
|
|
|
|
2021-02-02 22:57:42 +08:00
|
|
|
if (test_bit(QCA_IBS_DISABLED, &qca->flags) ||
|
|
|
|
test_bit(QCA_SSR_TRIGGERED, &qca->flags)) {
|
2020-10-06 23:20:21 +08:00
|
|
|
wait_timeout = test_bit(QCA_SSR_TRIGGERED, &qca->flags) ?
|
|
|
|
IBS_DISABLE_SSR_TIMEOUT_MS :
|
|
|
|
FW_DOWNLOAD_TIMEOUT_MS;
|
|
|
|
|
|
|
|
/* QCA_IBS_DISABLED flag is set to true, During FW download
|
|
|
|
* and during memory dump collection. It is reset to false,
|
2020-12-31 01:17:08 +08:00
|
|
|
* After FW download complete.
|
2020-10-06 23:20:21 +08:00
|
|
|
*/
|
|
|
|
wait_on_bit_timeout(&qca->flags, QCA_IBS_DISABLED,
|
|
|
|
TASK_UNINTERRUPTIBLE, msecs_to_jiffies(wait_timeout));
|
|
|
|
|
|
|
|
if (test_bit(QCA_IBS_DISABLED, &qca->flags)) {
|
|
|
|
bt_dev_err(hu->hdev, "SSR or FW download time out");
|
|
|
|
ret = -ETIMEDOUT;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
cancel_work_sync(&qca->ws_awake_device);
|
|
|
|
cancel_work_sync(&qca->ws_awake_rx);
|
|
|
|
|
|
|
|
spin_lock_irqsave_nested(&qca->hci_ibs_lock,
|
|
|
|
flags, SINGLE_DEPTH_NESTING);
|
|
|
|
|
|
|
|
switch (qca->tx_ibs_state) {
|
|
|
|
case HCI_IBS_TX_WAKING:
|
|
|
|
del_timer(&qca->wake_retrans_timer);
|
2020-07-09 04:26:50 +08:00
|
|
|
fallthrough;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
case HCI_IBS_TX_AWAKE:
|
|
|
|
del_timer(&qca->tx_idle_timer);
|
|
|
|
|
|
|
|
serdev_device_write_flush(hu->serdev);
|
|
|
|
cmd = HCI_IBS_SLEEP_IND;
|
|
|
|
ret = serdev_device_write_buf(hu->serdev, &cmd, sizeof(cmd));
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
BT_ERR("Failed to send SLEEP to device");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
qca->tx_ibs_state = HCI_IBS_TX_ASLEEP;
|
|
|
|
qca->ibs_sent_slps++;
|
2020-06-06 02:46:10 +08:00
|
|
|
tx_pending = true;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case HCI_IBS_TX_ASLEEP:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BT_ERR("Spurious tx state %d", qca->tx_ibs_state);
|
|
|
|
ret = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&qca->hci_ibs_lock, flags);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
goto error;
|
|
|
|
|
2020-06-06 02:46:10 +08:00
|
|
|
if (tx_pending) {
|
|
|
|
serdev_device_wait_until_sent(hu->serdev,
|
|
|
|
msecs_to_jiffies(CMD_TRANS_TIMEOUT_MS));
|
2020-06-12 20:15:17 +08:00
|
|
|
serial_clock_vote(HCI_IBS_TX_VOTE_CLOCK_OFF, hu);
|
2020-06-06 02:46:10 +08:00
|
|
|
}
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
|
|
|
|
/* Wait for HCI_IBS_SLEEP_IND sent by device to indicate its Tx is going
|
|
|
|
* to sleep, so that the packet does not wake the system later.
|
|
|
|
*/
|
|
|
|
ret = wait_event_interruptible_timeout(qca->suspend_wait_q,
|
|
|
|
qca->rx_ibs_state == HCI_IBS_RX_ASLEEP,
|
|
|
|
msecs_to_jiffies(IBS_BTSOC_TX_IDLE_TIMEOUT_MS));
|
2020-06-06 02:46:11 +08:00
|
|
|
if (ret == 0) {
|
|
|
|
ret = -ETIMEDOUT;
|
|
|
|
goto error;
|
2020-06-06 02:46:09 +08:00
|
|
|
}
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
|
2020-06-06 02:46:11 +08:00
|
|
|
return 0;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
|
|
|
|
error:
|
|
|
|
clear_bit(QCA_SUSPENDING, &qca->flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __maybe_unused qca_resume(struct device *dev)
|
|
|
|
{
|
2020-05-29 04:31:07 +08:00
|
|
|
struct serdev_device *serdev = to_serdev_device(dev);
|
|
|
|
struct qca_serdev *qcadev = serdev_device_get_drvdata(serdev);
|
|
|
|
struct hci_uart *hu = &qcadev->serdev_hu;
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
struct qca_data *qca = hu->priv;
|
|
|
|
|
|
|
|
clear_bit(QCA_SUSPENDING, &qca->flags);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SIMPLE_DEV_PM_OPS(qca_pm_ops, qca_suspend, qca_resume);
|
|
|
|
|
2020-03-25 10:26:37 +08:00
|
|
|
#ifdef CONFIG_OF
|
2018-03-30 03:15:24 +08:00
|
|
|
static const struct of_device_id qca_bluetooth_of_match[] = {
|
|
|
|
{ .compatible = "qcom,qca6174-bt" },
|
2020-03-25 10:26:37 +08:00
|
|
|
{ .compatible = "qcom,qca6390-bt", .data = &qca_soc_data_qca6390},
|
2020-04-23 09:34:29 +08:00
|
|
|
{ .compatible = "qcom,qca9377-bt" },
|
2019-04-26 21:56:01 +08:00
|
|
|
{ .compatible = "qcom,wcn3990-bt", .data = &qca_soc_data_wcn3990},
|
2019-11-06 17:48:32 +08:00
|
|
|
{ .compatible = "qcom,wcn3991-bt", .data = &qca_soc_data_wcn3991},
|
2019-04-26 21:56:01 +08:00
|
|
|
{ .compatible = "qcom,wcn3998-bt", .data = &qca_soc_data_wcn3998},
|
2021-05-19 00:34:42 +08:00
|
|
|
{ .compatible = "qcom,wcn6750-bt", .data = &qca_soc_data_wcn6750},
|
2018-03-30 03:15:24 +08:00
|
|
|
{ /* sentinel */ }
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(of, qca_bluetooth_of_match);
|
2020-03-25 10:26:37 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_ACPI
|
|
|
|
static const struct acpi_device_id qca_bluetooth_acpi_match[] = {
|
|
|
|
{ "QCOM6390", (kernel_ulong_t)&qca_soc_data_qca6390 },
|
|
|
|
{ "DLA16390", (kernel_ulong_t)&qca_soc_data_qca6390 },
|
|
|
|
{ "DLB16390", (kernel_ulong_t)&qca_soc_data_qca6390 },
|
|
|
|
{ "DLB26390", (kernel_ulong_t)&qca_soc_data_qca6390 },
|
|
|
|
{ },
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(acpi, qca_bluetooth_acpi_match);
|
|
|
|
#endif
|
|
|
|
|
2018-03-30 03:15:24 +08:00
|
|
|
|
|
|
|
static struct serdev_device_driver qca_serdev_driver = {
|
|
|
|
.probe = qca_serdev_probe,
|
|
|
|
.remove = qca_serdev_remove,
|
|
|
|
.driver = {
|
|
|
|
.name = "hci_uart_qca",
|
2020-03-25 10:26:37 +08:00
|
|
|
.of_match_table = of_match_ptr(qca_bluetooth_of_match),
|
|
|
|
.acpi_match_table = ACPI_PTR(qca_bluetooth_acpi_match),
|
2020-05-29 21:56:57 +08:00
|
|
|
.shutdown = qca_serdev_shutdown,
|
Bluetooth: hci_qca: add PM support
Add PM suspend/resume callbacks for hci_qca driver.
BT host will make sure both Rx and Tx go into sleep state in
qca_suspend. Without this, Tx may still remain in awake state, which
prevents BTSOC from entering deep sleep. For example, BlueZ will send
Set Event Mask to device when suspending and this will wake the device
Rx up. However, the Tx idle timeout on the host side is 2000 ms. If the
host is suspended before its Tx idle times out, it won't send
HCI_IBS_SLEEP_IND to the device and the device Rx will remain awake.
We implement this by canceling relevant work in workqueue, sending
HCI_IBS_SLEEP_IND to the device and then waiting HCI_IBS_SLEEP_IND sent
by the device.
In order to prevent the device from being awaken again after qca_suspend
is called, we introduce QCA_SUSPEND flag. QCA_SUSPEND is set in the
beginning of qca_suspend to indicate system is suspending and that we'd
like to ignore any further wake events.
With QCA_SUSPEND and spinlock, we can avoid race condition, e.g. if
qca_enqueue acquires qca->hci_ibs_lock before qca_suspend calls
cancel_work_sync and then qca_enqueue adds a new qca->ws_awake_device
work after the previous one is cancelled.
If BTSOC wants to wake the whole system up after qca_suspend is called,
it will keep sending HCI_IBS_WAKE_IND and uart driver will take care of
waking the system. For example, uart driver will reconfigure its Rx pin
to a normal GPIO pin and enable irq wake on that pin when suspending.
Once host detects Rx falling, the system will begin resuming. Then, the
BT host clears QCA_SUSPEND flag in qca_resume and begins dealing with
normal HCI packets. By doing so, only a few HCI_IBS_WAKE_IND packets are
lost and there is no data packet loss.
Signed-off-by: Claire Chang <tientzu@chromium.org>
Reviewed-by: Balakrishna Godavarthi <bgodavar@codeaurora.org>
Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
2019-10-31 18:46:14 +08:00
|
|
|
.pm = &qca_pm_ops,
|
2018-03-30 03:15:24 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
int __init qca_init(void)
|
|
|
|
{
|
2018-03-30 03:15:24 +08:00
|
|
|
serdev_device_driver_register(&qca_serdev_driver);
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
return hci_uart_register_proto(&qca_proto);
|
|
|
|
}
|
|
|
|
|
|
|
|
int __exit qca_deinit(void)
|
|
|
|
{
|
2018-03-30 03:15:24 +08:00
|
|
|
serdev_device_driver_unregister(&qca_serdev_driver);
|
|
|
|
|
2015-08-11 05:24:17 +08:00
|
|
|
return hci_uart_unregister_proto(&qca_proto);
|
|
|
|
}
|