2019-03-08 09:15:44 +08:00
// SPDX-License-Identifier: GPL-2.0
// Copyright (c) 2019 MediaTek Inc.
/*
* Bluetooth support for MediaTek SDIO devices
*
* This file is written based on btsdio . c and btmtkuart . c .
*
* Author : Sean Wang < sean . wang @ mediatek . com >
*
*/
# include <asm/unaligned.h>
# include <linux/atomic.h>
2022-02-09 02:17:40 +08:00
# include <linux/gpio/consumer.h>
2019-03-08 09:15:44 +08:00
# include <linux/init.h>
# include <linux/iopoll.h>
# include <linux/kernel.h>
# include <linux/module.h>
2022-02-09 02:17:40 +08:00
# include <linux/of.h>
2019-04-18 17:08:02 +08:00
# include <linux/pm_runtime.h>
2019-03-08 09:15:44 +08:00
# include <linux/skbuff.h>
# include <linux/mmc/host.h>
# include <linux/mmc/sdio_ids.h>
# include <linux/mmc/sdio_func.h>
# include <net/bluetooth/bluetooth.h>
# include <net/bluetooth/hci_core.h>
# include "h4_recv.h"
2021-10-19 05:30:13 +08:00
# include "btmtk.h"
2019-03-08 09:15:44 +08:00
# define VERSION "0.1"
2022-01-12 15:39:16 +08:00
# define MTKBTSDIO_AUTOSUSPEND_DELAY 1000
2019-04-18 17:08:02 +08:00
2022-01-12 15:39:16 +08:00
static bool enable_autosuspend = true ;
2019-04-18 17:08:02 +08:00
2019-03-08 09:15:44 +08:00
struct btmtksdio_data {
const char * fwname ;
2021-10-19 05:30:21 +08:00
u16 chipid ;
2022-01-12 15:39:14 +08:00
bool lp_mbox_supported ;
2019-03-08 09:15:44 +08:00
} ;
static const struct btmtksdio_data mt7663_data = {
. fwname = FIRMWARE_MT7663 ,
2021-10-19 05:30:21 +08:00
. chipid = 0x7663 ,
2022-01-12 15:39:14 +08:00
. lp_mbox_supported = false ,
2019-03-08 09:15:44 +08:00
} ;
static const struct btmtksdio_data mt7668_data = {
. fwname = FIRMWARE_MT7668 ,
2021-10-19 05:30:21 +08:00
. chipid = 0x7668 ,
2022-01-12 15:39:14 +08:00
. lp_mbox_supported = false ,
2021-10-19 05:30:21 +08:00
} ;
static const struct btmtksdio_data mt7921_data = {
. fwname = FIRMWARE_MT7961 ,
. chipid = 0x7921 ,
2022-01-12 15:39:14 +08:00
. lp_mbox_supported = true ,
2019-03-08 09:15:44 +08:00
} ;
static const struct sdio_device_id btmtksdio_table [ ] = {
2020-05-22 16:44:06 +02:00
{ SDIO_DEVICE ( SDIO_VENDOR_ID_MEDIATEK , SDIO_DEVICE_ID_MEDIATEK_MT7663 ) ,
2019-03-08 09:15:44 +08:00
. driver_data = ( kernel_ulong_t ) & mt7663_data } ,
2020-05-22 16:44:06 +02:00
{ SDIO_DEVICE ( SDIO_VENDOR_ID_MEDIATEK , SDIO_DEVICE_ID_MEDIATEK_MT7668 ) ,
2019-03-08 09:15:44 +08:00
. driver_data = ( kernel_ulong_t ) & mt7668_data } ,
2021-10-19 05:30:21 +08:00
{ SDIO_DEVICE ( SDIO_VENDOR_ID_MEDIATEK , SDIO_DEVICE_ID_MEDIATEK_MT7961 ) ,
. driver_data = ( kernel_ulong_t ) & mt7921_data } ,
2019-03-08 09:15:44 +08:00
{ } /* Terminating entry */
} ;
2019-11-07 10:46:10 +01:00
MODULE_DEVICE_TABLE ( sdio , btmtksdio_table ) ;
2019-03-08 09:15:44 +08:00
# define MTK_REG_CHLPCR 0x4 /* W1S */
# define C_INT_EN_SET BIT(0)
# define C_INT_EN_CLR BIT(1)
2019-04-18 17:08:00 +08:00
# define C_FW_OWN_REQ_SET BIT(8) /* For write */
# define C_COM_DRV_OWN BIT(8) /* For read */
2019-03-08 09:15:44 +08:00
# define C_FW_OWN_REQ_CLR BIT(9)
# define MTK_REG_CSDIOCSR 0x8
# define SDIO_RE_INIT_EN BIT(0)
# define SDIO_INT_CTL BIT(2)
# define MTK_REG_CHCR 0xc
# define C_INT_CLR_CTRL BIT(1)
2022-02-09 02:17:40 +08:00
# define BT_RST_DONE BIT(8)
2019-03-08 09:15:44 +08:00
/* CHISR have the same bits field definition with CHIER */
# define MTK_REG_CHISR 0x10
# define MTK_REG_CHIER 0x14
# define FW_OWN_BACK_INT BIT(0)
# define RX_DONE_INT BIT(1)
# define TX_EMPTY BIT(2)
# define TX_FIFO_OVERFLOW BIT(8)
2022-01-12 15:39:11 +08:00
# define FW_MAILBOX_INT BIT(15)
2022-01-12 15:39:17 +08:00
# define INT_MASK GENMASK(15, 0)
2019-03-08 09:15:44 +08:00
# define RX_PKT_LEN GENMASK(31, 16)
2022-01-12 15:39:14 +08:00
# define MTK_REG_CSICR 0xc0
# define CSICR_CLR_MBOX_ACK BIT(0)
2022-01-12 15:39:11 +08:00
# define MTK_REG_PH2DSM0R 0xc4
# define PH2DSM0R_DRIVER_OWN BIT(0)
2022-01-12 15:39:14 +08:00
# define MTK_REG_PD2HRM0R 0xdc
# define PD2HRM0R_DRV_OWN BIT(0)
2022-01-12 15:39:11 +08:00
2019-03-08 09:15:44 +08:00
# define MTK_REG_CTDR 0x18
# define MTK_REG_CRDR 0x1c
2021-10-19 05:30:18 +08:00
# define MTK_REG_CRPLR 0x24
2019-03-08 09:15:44 +08:00
# define MTK_SDIO_BLOCK_SIZE 256
# define BTMTKSDIO_TX_WAIT_VND_EVT 1
2021-11-20 06:25:43 +08:00
# define BTMTKSDIO_HW_TX_READY 2
2021-12-02 02:02:46 +08:00
# define BTMTKSDIO_FUNC_ENABLED 3
2022-01-12 15:39:14 +08:00
# define BTMTKSDIO_PATCH_ENABLED 4
2022-02-09 02:17:40 +08:00
# define BTMTKSDIO_HW_RESET_ACTIVE 5
2019-03-08 09:15:44 +08:00
struct mtkbtsdio_hdr {
__le16 len ;
__le16 reserved ;
u8 bt_type ;
} __packed ;
struct btmtksdio_dev {
struct hci_dev * hdev ;
struct sdio_func * func ;
2019-04-18 17:08:02 +08:00
struct device * dev ;
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
struct work_struct txrx_work ;
2019-03-08 09:15:44 +08:00
unsigned long tx_state ;
struct sk_buff_head txq ;
struct sk_buff * evt_skb ;
const struct btmtksdio_data * data ;
2022-02-09 02:17:40 +08:00
struct gpio_desc * reset ;
2019-03-08 09:15:44 +08:00
} ;
static int mtk_hci_wmt_sync ( struct hci_dev * hdev ,
struct btmtk_hci_wmt_params * wmt_params )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
struct btmtk_hci_wmt_evt_funcc * wmt_evt_funcc ;
2021-10-19 05:30:21 +08:00
struct btmtk_hci_wmt_evt_reg * wmt_evt_reg ;
2019-03-08 09:15:44 +08:00
u32 hlen , status = BTMTK_WMT_INVALID ;
struct btmtk_hci_wmt_evt * wmt_evt ;
2021-10-19 05:30:13 +08:00
struct btmtk_hci_wmt_cmd * wc ;
struct btmtk_wmt_hdr * hdr ;
2019-03-08 09:15:44 +08:00
int err ;
2021-10-19 05:30:13 +08:00
/* Send the WMT command and wait until the WMT event returns */
2019-03-08 09:15:44 +08:00
hlen = sizeof ( * hdr ) + wmt_params - > dlen ;
if ( hlen > 255 )
return - EINVAL ;
2021-10-19 05:30:13 +08:00
wc = kzalloc ( hlen , GFP_KERNEL ) ;
if ( ! wc )
return - ENOMEM ;
hdr = & wc - > hdr ;
2019-03-08 09:15:44 +08:00
hdr - > dir = 1 ;
hdr - > op = wmt_params - > op ;
hdr - > dlen = cpu_to_le16 ( wmt_params - > dlen + 1 ) ;
hdr - > flag = wmt_params - > flag ;
2021-10-19 05:30:13 +08:00
memcpy ( wc - > data , wmt_params - > data , wmt_params - > dlen ) ;
2019-03-08 09:15:44 +08:00
set_bit ( BTMTKSDIO_TX_WAIT_VND_EVT , & bdev - > tx_state ) ;
2021-10-19 05:30:13 +08:00
err = __hci_cmd_send ( hdev , 0xfc6f , hlen , wc ) ;
2019-03-08 09:15:44 +08:00
if ( err < 0 ) {
clear_bit ( BTMTKSDIO_TX_WAIT_VND_EVT , & bdev - > tx_state ) ;
2021-10-19 05:30:13 +08:00
goto err_free_wc ;
2019-03-08 09:15:44 +08:00
}
/* The vendor specific WMT commands are all answered by a vendor
* specific event and will not have the Command Status or Command
* Complete as with usual HCI command flow control .
*
* After sending the command , wait for BTMTKSDIO_TX_WAIT_VND_EVT
* state to be cleared . The driver specific event receive routine
* will clear that state and with that indicate completion of the
* WMT command .
*/
err = wait_on_bit_timeout ( & bdev - > tx_state , BTMTKSDIO_TX_WAIT_VND_EVT ,
TASK_INTERRUPTIBLE , HCI_INIT_TIMEOUT ) ;
if ( err = = - EINTR ) {
bt_dev_err ( hdev , " Execution of wmt command interrupted " ) ;
clear_bit ( BTMTKSDIO_TX_WAIT_VND_EVT , & bdev - > tx_state ) ;
2021-10-19 05:30:13 +08:00
goto err_free_wc ;
2019-03-08 09:15:44 +08:00
}
if ( err ) {
bt_dev_err ( hdev , " Execution of wmt command timed out " ) ;
clear_bit ( BTMTKSDIO_TX_WAIT_VND_EVT , & bdev - > tx_state ) ;
2021-10-19 05:30:13 +08:00
err = - ETIMEDOUT ;
goto err_free_wc ;
2019-03-08 09:15:44 +08:00
}
/* Parse and handle the return WMT event */
wmt_evt = ( struct btmtk_hci_wmt_evt * ) bdev - > evt_skb - > data ;
if ( wmt_evt - > whdr . op ! = hdr - > op ) {
bt_dev_err ( hdev , " Wrong op received %d expected %d " ,
wmt_evt - > whdr . op , hdr - > op ) ;
err = - EIO ;
goto err_free_skb ;
}
switch ( wmt_evt - > whdr . op ) {
2021-10-19 05:30:13 +08:00
case BTMTK_WMT_SEMAPHORE :
2019-03-08 09:15:44 +08:00
if ( wmt_evt - > whdr . flag = = 2 )
status = BTMTK_WMT_PATCH_UNDONE ;
else
status = BTMTK_WMT_PATCH_DONE ;
break ;
2021-10-19 05:30:13 +08:00
case BTMTK_WMT_FUNC_CTRL :
2019-03-08 09:15:44 +08:00
wmt_evt_funcc = ( struct btmtk_hci_wmt_evt_funcc * ) wmt_evt ;
if ( be16_to_cpu ( wmt_evt_funcc - > status ) = = 0x404 )
status = BTMTK_WMT_ON_DONE ;
else if ( be16_to_cpu ( wmt_evt_funcc - > status ) = = 0x420 )
status = BTMTK_WMT_ON_PROGRESS ;
else
status = BTMTK_WMT_ON_UNDONE ;
break ;
2021-10-19 05:30:21 +08:00
case BTMTK_WMT_PATCH_DWNLD :
if ( wmt_evt - > whdr . flag = = 2 )
status = BTMTK_WMT_PATCH_DONE ;
else if ( wmt_evt - > whdr . flag = = 1 )
status = BTMTK_WMT_PATCH_PROGRESS ;
else
status = BTMTK_WMT_PATCH_UNDONE ;
break ;
case BTMTK_WMT_REGISTER :
wmt_evt_reg = ( struct btmtk_hci_wmt_evt_reg * ) wmt_evt ;
if ( le16_to_cpu ( wmt_evt - > whdr . dlen ) = = 12 )
status = le32_to_cpu ( wmt_evt_reg - > val ) ;
break ;
2019-03-08 09:15:44 +08:00
}
if ( wmt_params - > status )
* wmt_params - > status = status ;
err_free_skb :
kfree_skb ( bdev - > evt_skb ) ;
bdev - > evt_skb = NULL ;
2021-10-19 05:30:13 +08:00
err_free_wc :
kfree ( wc ) ;
2019-03-08 09:15:44 +08:00
return err ;
}
static int btmtksdio_tx_packet ( struct btmtksdio_dev * bdev ,
struct sk_buff * skb )
{
struct mtkbtsdio_hdr * sdio_hdr ;
int err ;
/* Make sure that there are enough rooms for SDIO header */
if ( unlikely ( skb_headroom ( skb ) < sizeof ( * sdio_hdr ) ) ) {
err = pskb_expand_head ( skb , sizeof ( * sdio_hdr ) , 0 ,
GFP_ATOMIC ) ;
if ( err < 0 )
return err ;
}
/* Prepend MediaTek SDIO Specific Header */
skb_push ( skb , sizeof ( * sdio_hdr ) ) ;
sdio_hdr = ( void * ) skb - > data ;
sdio_hdr - > len = cpu_to_le16 ( skb - > len ) ;
sdio_hdr - > reserved = cpu_to_le16 ( 0 ) ;
sdio_hdr - > bt_type = hci_skb_pkt_type ( skb ) ;
2021-11-20 06:25:43 +08:00
clear_bit ( BTMTKSDIO_HW_TX_READY , & bdev - > tx_state ) ;
2019-03-08 09:15:44 +08:00
err = sdio_writesb ( bdev - > func , MTK_REG_CTDR , skb - > data ,
round_up ( skb - > len , MTK_SDIO_BLOCK_SIZE ) ) ;
if ( err < 0 )
goto err_skb_pull ;
bdev - > hdev - > stat . byte_tx + = skb - > len ;
kfree_skb ( skb ) ;
return 0 ;
err_skb_pull :
skb_pull ( skb , sizeof ( * sdio_hdr ) ) ;
return err ;
}
static u32 btmtksdio_drv_own_query ( struct btmtksdio_dev * bdev )
{
return sdio_readl ( bdev - > func , MTK_REG_CHLPCR , NULL ) ;
}
2022-01-12 15:39:14 +08:00
static u32 btmtksdio_drv_own_query_79xx ( struct btmtksdio_dev * bdev )
{
return sdio_readl ( bdev - > func , MTK_REG_PD2HRM0R , NULL ) ;
}
2022-02-09 02:17:40 +08:00
static u32 btmtksdio_chcr_query ( struct btmtksdio_dev * bdev )
{
return sdio_readl ( bdev - > func , MTK_REG_CHCR , NULL ) ;
}
2022-01-12 15:39:12 +08:00
static int btmtksdio_fw_pmctrl ( struct btmtksdio_dev * bdev )
{
u32 status ;
int err ;
sdio_claim_host ( bdev - > func ) ;
2022-01-12 15:39:14 +08:00
if ( bdev - > data - > lp_mbox_supported & &
test_bit ( BTMTKSDIO_PATCH_ENABLED , & bdev - > tx_state ) ) {
sdio_writel ( bdev - > func , CSICR_CLR_MBOX_ACK , MTK_REG_CSICR ,
& err ) ;
err = readx_poll_timeout ( btmtksdio_drv_own_query_79xx , bdev ,
status , ! ( status & PD2HRM0R_DRV_OWN ) ,
2000 , 1000000 ) ;
if ( err < 0 ) {
bt_dev_err ( bdev - > hdev , " mailbox ACK not cleared " ) ;
goto out ;
}
}
2022-01-12 15:39:12 +08:00
/* Return ownership to the device */
sdio_writel ( bdev - > func , C_FW_OWN_REQ_SET , MTK_REG_CHLPCR , & err ) ;
if ( err < 0 )
goto out ;
err = readx_poll_timeout ( btmtksdio_drv_own_query , bdev , status ,
! ( status & C_COM_DRV_OWN ) , 2000 , 1000000 ) ;
out :
sdio_release_host ( bdev - > func ) ;
if ( err < 0 )
bt_dev_err ( bdev - > hdev , " Cannot return ownership to device " ) ;
return err ;
}
static int btmtksdio_drv_pmctrl ( struct btmtksdio_dev * bdev )
{
u32 status ;
int err ;
sdio_claim_host ( bdev - > func ) ;
/* Get ownership from the device */
sdio_writel ( bdev - > func , C_FW_OWN_REQ_CLR , MTK_REG_CHLPCR , & err ) ;
if ( err < 0 )
goto out ;
err = readx_poll_timeout ( btmtksdio_drv_own_query , bdev , status ,
status & C_COM_DRV_OWN , 2000 , 1000000 ) ;
2022-01-12 15:39:14 +08:00
if ( ! err & & bdev - > data - > lp_mbox_supported & &
test_bit ( BTMTKSDIO_PATCH_ENABLED , & bdev - > tx_state ) )
err = readx_poll_timeout ( btmtksdio_drv_own_query_79xx , bdev ,
status , status & PD2HRM0R_DRV_OWN ,
2000 , 1000000 ) ;
2022-01-12 15:39:12 +08:00
out :
sdio_release_host ( bdev - > func ) ;
if ( err < 0 )
bt_dev_err ( bdev - > hdev , " Cannot get ownership from device " ) ;
return err ;
}
2019-03-08 09:15:44 +08:00
static int btmtksdio_recv_event ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
struct hci_event_hdr * hdr = ( void * ) skb - > data ;
2022-05-12 06:22:15 +08:00
u8 evt = hdr - > evt ;
2019-03-08 09:15:44 +08:00
int err ;
/* When someone waits for the WMT event, the skb is being cloned
* and being processed the events from there then .
*/
if ( test_bit ( BTMTKSDIO_TX_WAIT_VND_EVT , & bdev - > tx_state ) ) {
bdev - > evt_skb = skb_clone ( skb , GFP_KERNEL ) ;
if ( ! bdev - > evt_skb ) {
err = - ENOMEM ;
goto err_out ;
}
}
err = hci_recv_frame ( hdev , skb ) ;
if ( err < 0 )
goto err_free_skb ;
2022-05-12 06:22:15 +08:00
if ( evt = = HCI_EV_WMT ) {
2019-03-08 09:15:44 +08:00
if ( test_and_clear_bit ( BTMTKSDIO_TX_WAIT_VND_EVT ,
& bdev - > tx_state ) ) {
/* Barrier to sync with other CPUs */
smp_mb__after_atomic ( ) ;
wake_up_bit ( & bdev - > tx_state , BTMTKSDIO_TX_WAIT_VND_EVT ) ;
}
}
return 0 ;
err_free_skb :
kfree_skb ( bdev - > evt_skb ) ;
bdev - > evt_skb = NULL ;
err_out :
return err ;
}
2021-11-20 06:25:46 +08:00
static int btmtksdio_recv_acl ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
u16 handle = le16_to_cpu ( hci_acl_hdr ( skb ) - > handle ) ;
switch ( handle ) {
case 0xfc6f :
/* Firmware dump from device: when the firmware hangs, the
* device can no longer suspend and thus disable auto - suspend .
*/
pm_runtime_forbid ( bdev - > dev ) ;
fallthrough ;
case 0x05ff :
case 0x05fe :
/* Firmware debug logging */
return hci_recv_diag ( hdev , skb ) ;
}
return hci_recv_frame ( hdev , skb ) ;
}
2019-03-08 09:15:44 +08:00
static const struct h4_recv_pkt mtk_recv_pkts [ ] = {
2021-11-20 06:25:46 +08:00
{ H4_RECV_ACL , . recv = btmtksdio_recv_acl } ,
2019-03-08 09:15:44 +08:00
{ H4_RECV_SCO , . recv = hci_recv_frame } ,
{ H4_RECV_EVENT , . recv = btmtksdio_recv_event } ,
} ;
static int btmtksdio_rx_packet ( struct btmtksdio_dev * bdev , u16 rx_size )
{
const struct h4_recv_pkt * pkts = mtk_recv_pkts ;
int pkts_count = ARRAY_SIZE ( mtk_recv_pkts ) ;
struct mtkbtsdio_hdr * sdio_hdr ;
int err , i , pad_size ;
struct sk_buff * skb ;
u16 dlen ;
if ( rx_size < sizeof ( * sdio_hdr ) )
return - EILSEQ ;
/* A SDIO packet is exactly containing a Bluetooth packet */
skb = bt_skb_alloc ( rx_size , GFP_KERNEL ) ;
if ( ! skb )
return - ENOMEM ;
skb_put ( skb , rx_size ) ;
err = sdio_readsb ( bdev - > func , skb - > data , MTK_REG_CRDR , rx_size ) ;
if ( err < 0 )
goto err_kfree_skb ;
sdio_hdr = ( void * ) skb - > data ;
/* We assume the default error as -EILSEQ simply to make the error path
* be cleaner .
*/
err = - EILSEQ ;
if ( rx_size ! = le16_to_cpu ( sdio_hdr - > len ) ) {
bt_dev_err ( bdev - > hdev , " Rx size in sdio header is mismatched " ) ;
goto err_kfree_skb ;
}
hci_skb_pkt_type ( skb ) = sdio_hdr - > bt_type ;
/* Remove MediaTek SDIO header */
skb_pull ( skb , sizeof ( * sdio_hdr ) ) ;
/* We have to dig into the packet to get payload size and then know how
* many padding bytes at the tail , these padding bytes should be removed
* before the packet is indicated to the core layer .
*/
for ( i = 0 ; i < pkts_count ; i + + ) {
if ( sdio_hdr - > bt_type = = ( & pkts [ i ] ) - > type )
break ;
}
if ( i > = pkts_count ) {
bt_dev_err ( bdev - > hdev , " Invalid bt type 0x%02x " ,
sdio_hdr - > bt_type ) ;
goto err_kfree_skb ;
}
/* Remaining bytes cannot hold a header*/
if ( skb - > len < ( & pkts [ i ] ) - > hlen ) {
bt_dev_err ( bdev - > hdev , " The size of bt header is mismatched " ) ;
goto err_kfree_skb ;
}
switch ( ( & pkts [ i ] ) - > lsize ) {
2020-12-15 15:17:30 +05:30
case 1 :
dlen = skb - > data [ ( & pkts [ i ] ) - > loff ] ;
break ;
case 2 :
dlen = get_unaligned_le16 ( skb - > data +
2019-03-08 09:15:44 +08:00
( & pkts [ i ] ) - > loff ) ;
2020-12-15 15:17:30 +05:30
break ;
default :
goto err_kfree_skb ;
2019-03-08 09:15:44 +08:00
}
pad_size = skb - > len - ( & pkts [ i ] ) - > hlen - dlen ;
/* Remaining bytes cannot hold a payload */
if ( pad_size < 0 ) {
bt_dev_err ( bdev - > hdev , " The size of bt payload is mismatched " ) ;
goto err_kfree_skb ;
}
/* Remove padding bytes */
skb_trim ( skb , skb - > len - pad_size ) ;
/* Complete frame */
( & pkts [ i ] ) - > recv ( bdev - > hdev , skb ) ;
2019-04-18 17:08:01 +08:00
bdev - > hdev - > stat . byte_rx + = rx_size ;
2019-03-08 09:15:44 +08:00
return 0 ;
err_kfree_skb :
kfree_skb ( skb ) ;
return err ;
}
2021-10-19 05:30:16 +08:00
static void btmtksdio_txrx_work ( struct work_struct * work )
2019-03-08 09:15:44 +08:00
{
2021-10-19 05:30:16 +08:00
struct btmtksdio_dev * bdev = container_of ( work , struct btmtksdio_dev ,
txrx_work ) ;
unsigned long txrx_timeout ;
2021-10-19 05:30:18 +08:00
u32 int_status , rx_size ;
2021-10-19 05:30:16 +08:00
struct sk_buff * skb ;
int err ;
2019-04-18 17:08:02 +08:00
pm_runtime_get_sync ( bdev - > dev ) ;
sdio_claim_host ( bdev - > func ) ;
2019-03-08 09:15:44 +08:00
/* Disable interrupt */
2021-10-19 05:30:16 +08:00
sdio_writel ( bdev - > func , C_INT_EN_CLR , MTK_REG_CHLPCR , 0 ) ;
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
txrx_timeout = jiffies + 5 * HZ ;
do {
int_status = sdio_readl ( bdev - > func , MTK_REG_CHISR , NULL ) ;
/* Ack an interrupt as soon as possible before any operation on
* hardware .
*
* Note that we don ' t ack any status during operations to avoid race
* condition between the host and the device such as it ' s possible to
* mistakenly ack RX_DONE for the next packet and then cause interrupts
* not be raised again but there is still pending data in the hardware
* FIFO .
*/
sdio_writel ( bdev - > func , int_status , MTK_REG_CHISR , NULL ) ;
2022-01-12 15:39:17 +08:00
int_status & = INT_MASK ;
2021-10-19 05:30:16 +08:00
2022-01-12 15:39:11 +08:00
if ( ( int_status & FW_MAILBOX_INT ) & &
bdev - > data - > chipid = = 0x7921 ) {
sdio_writel ( bdev - > func , PH2DSM0R_DRIVER_OWN ,
MTK_REG_PH2DSM0R , 0 ) ;
}
2021-10-19 05:30:16 +08:00
if ( int_status & FW_OWN_BACK_INT )
bt_dev_dbg ( bdev - > hdev , " Get fw own back " ) ;
if ( int_status & TX_EMPTY )
2021-11-20 06:25:43 +08:00
set_bit ( BTMTKSDIO_HW_TX_READY , & bdev - > tx_state ) ;
2021-10-19 05:30:16 +08:00
else if ( unlikely ( int_status & TX_FIFO_OVERFLOW ) )
bt_dev_warn ( bdev - > hdev , " Tx fifo overflow " ) ;
2021-11-20 06:25:43 +08:00
if ( test_bit ( BTMTKSDIO_HW_TX_READY , & bdev - > tx_state ) ) {
2021-10-19 05:30:19 +08:00
skb = skb_dequeue ( & bdev - > txq ) ;
if ( skb ) {
err = btmtksdio_tx_packet ( bdev , skb ) ;
if ( err < 0 ) {
bdev - > hdev - > stat . err_tx + + ;
skb_queue_head ( & bdev - > txq , skb ) ;
}
}
}
2021-10-19 05:30:16 +08:00
if ( int_status & RX_DONE_INT ) {
2021-10-19 05:30:18 +08:00
rx_size = sdio_readl ( bdev - > func , MTK_REG_CRPLR , NULL ) ;
rx_size = ( rx_size & RX_PKT_LEN ) > > 16 ;
2021-10-19 05:30:16 +08:00
if ( btmtksdio_rx_packet ( bdev , rx_size ) < 0 )
bdev - > hdev - > stat . err_rx + + ;
}
} while ( int_status | | time_is_before_jiffies ( txrx_timeout ) ) ;
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
/* Enable interrupt */
sdio_writel ( bdev - > func , C_INT_EN_SET , MTK_REG_CHLPCR , 0 ) ;
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
sdio_release_host ( bdev - > func ) ;
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
pm_runtime_mark_last_busy ( bdev - > dev ) ;
pm_runtime_put_autosuspend ( bdev - > dev ) ;
}
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
static void btmtksdio_interrupt ( struct sdio_func * func )
{
struct btmtksdio_dev * bdev = sdio_get_drvdata ( func ) ;
2019-03-08 09:15:44 +08:00
2021-10-19 05:30:16 +08:00
/* Disable interrupt */
sdio_writel ( bdev - > func , C_INT_EN_CLR , MTK_REG_CHLPCR , 0 ) ;
2019-04-18 17:08:02 +08:00
2021-10-19 05:30:16 +08:00
schedule_work ( & bdev - > txrx_work ) ;
2019-03-08 09:15:44 +08:00
}
static int btmtksdio_open ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
2022-01-12 15:39:12 +08:00
u32 val ;
2019-03-08 09:15:44 +08:00
int err ;
sdio_claim_host ( bdev - > func ) ;
err = sdio_enable_func ( bdev - > func ) ;
if ( err < 0 )
goto err_release_host ;
2021-12-02 02:02:46 +08:00
set_bit ( BTMTKSDIO_FUNC_ENABLED , & bdev - > tx_state ) ;
2022-01-12 15:39:12 +08:00
err = btmtksdio_drv_pmctrl ( bdev ) ;
2019-03-08 09:15:44 +08:00
if ( err < 0 )
goto err_disable_func ;
/* Disable interrupt & mask out all interrupt sources */
sdio_writel ( bdev - > func , C_INT_EN_CLR , MTK_REG_CHLPCR , & err ) ;
if ( err < 0 )
goto err_disable_func ;
sdio_writel ( bdev - > func , 0 , MTK_REG_CHIER , & err ) ;
if ( err < 0 )
goto err_disable_func ;
err = sdio_claim_irq ( bdev - > func , btmtksdio_interrupt ) ;
if ( err < 0 )
goto err_disable_func ;
err = sdio_set_block_size ( bdev - > func , MTK_SDIO_BLOCK_SIZE ) ;
if ( err < 0 )
goto err_release_irq ;
/* SDIO CMD 5 allows the SDIO device back to idle state an
* synchronous interrupt is supported in SDIO 4 - bit mode
*/
2021-10-19 05:30:17 +08:00
val = sdio_readl ( bdev - > func , MTK_REG_CSDIOCSR , & err ) ;
if ( err < 0 )
goto err_release_irq ;
val | = SDIO_INT_CTL ;
sdio_writel ( bdev - > func , val , MTK_REG_CSDIOCSR , & err ) ;
2019-03-08 09:15:44 +08:00
if ( err < 0 )
goto err_release_irq ;
2021-10-19 05:30:15 +08:00
/* Explitly set write-1-clear method */
val = sdio_readl ( bdev - > func , MTK_REG_CHCR , & err ) ;
if ( err < 0 )
goto err_release_irq ;
val | = C_INT_CLR_CTRL ;
sdio_writel ( bdev - > func , val , MTK_REG_CHCR , & err ) ;
2019-03-08 09:15:44 +08:00
if ( err < 0 )
goto err_release_irq ;
/* Setup interrupt sources */
sdio_writel ( bdev - > func , RX_DONE_INT | TX_EMPTY | TX_FIFO_OVERFLOW ,
MTK_REG_CHIER , & err ) ;
if ( err < 0 )
goto err_release_irq ;
/* Enable interrupt */
sdio_writel ( bdev - > func , C_INT_EN_SET , MTK_REG_CHLPCR , & err ) ;
if ( err < 0 )
goto err_release_irq ;
sdio_release_host ( bdev - > func ) ;
return 0 ;
err_release_irq :
sdio_release_irq ( bdev - > func ) ;
err_disable_func :
sdio_disable_func ( bdev - > func ) ;
err_release_host :
sdio_release_host ( bdev - > func ) ;
return err ;
}
static int btmtksdio_close ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
sdio_claim_host ( bdev - > func ) ;
/* Disable interrupt */
sdio_writel ( bdev - > func , C_INT_EN_CLR , MTK_REG_CHLPCR , NULL ) ;
sdio_release_irq ( bdev - > func ) ;
2021-10-19 05:30:16 +08:00
cancel_work_sync ( & bdev - > txrx_work ) ;
2022-01-12 15:39:12 +08:00
btmtksdio_fw_pmctrl ( bdev ) ;
2019-03-08 09:15:44 +08:00
2021-12-02 02:02:46 +08:00
clear_bit ( BTMTKSDIO_FUNC_ENABLED , & bdev - > tx_state ) ;
2019-03-08 09:15:44 +08:00
sdio_disable_func ( bdev - > func ) ;
sdio_release_host ( bdev - > func ) ;
return 0 ;
}
static int btmtksdio_flush ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
skb_queue_purge ( & bdev - > txq ) ;
2021-10-19 05:30:16 +08:00
cancel_work_sync ( & bdev - > txrx_work ) ;
2019-03-08 09:15:44 +08:00
return 0 ;
}
static int btmtksdio_func_query ( struct hci_dev * hdev )
{
struct btmtk_hci_wmt_params wmt_params ;
int status , err ;
u8 param = 0 ;
/* Query whether the function is enabled */
2021-10-19 05:30:13 +08:00
wmt_params . op = BTMTK_WMT_FUNC_CTRL ;
2019-03-08 09:15:44 +08:00
wmt_params . flag = 4 ;
wmt_params . dlen = sizeof ( param ) ;
wmt_params . data = & param ;
wmt_params . status = & status ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to query function status (%d) " , err ) ;
return err ;
}
return status ;
}
2021-10-19 05:30:21 +08:00
static int mt76xx_setup ( struct hci_dev * hdev , const char * fwname )
2019-03-08 09:15:44 +08:00
{
2022-01-12 15:39:14 +08:00
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
2019-03-08 09:15:44 +08:00
struct btmtk_hci_wmt_params wmt_params ;
struct btmtk_tci_sleep tci_sleep ;
struct sk_buff * skb ;
int err , status ;
u8 param = 0x1 ;
/* Query whether the firmware is already download */
2021-10-19 05:30:13 +08:00
wmt_params . op = BTMTK_WMT_SEMAPHORE ;
2019-03-08 09:15:44 +08:00
wmt_params . flag = 1 ;
wmt_params . dlen = 0 ;
wmt_params . data = NULL ;
wmt_params . status = & status ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to query firmware status (%d) " , err ) ;
return err ;
}
if ( status = = BTMTK_WMT_PATCH_DONE ) {
bt_dev_info ( hdev , " Firmware already downloaded " ) ;
goto ignore_setup_fw ;
}
/* Setup a firmware which the device definitely requires */
2021-10-19 05:30:21 +08:00
err = btmtk_setup_firmware ( hdev , fwname , mtk_hci_wmt_sync ) ;
2019-03-08 09:15:44 +08:00
if ( err < 0 )
return err ;
ignore_setup_fw :
/* Query whether the device is already enabled */
err = readx_poll_timeout ( btmtksdio_func_query , hdev , status ,
status < 0 | | status ! = BTMTK_WMT_ON_PROGRESS ,
2000 , 5000000 ) ;
/* -ETIMEDOUT happens */
if ( err < 0 )
return err ;
/* The other errors happen in btusb_mtk_func_query */
if ( status < 0 )
return status ;
if ( status = = BTMTK_WMT_ON_DONE ) {
bt_dev_info ( hdev , " function already on " ) ;
goto ignore_func_on ;
}
/* Enable Bluetooth protocol */
2021-10-19 05:30:13 +08:00
wmt_params . op = BTMTK_WMT_FUNC_CTRL ;
2019-03-08 09:15:44 +08:00
wmt_params . flag = 0 ;
wmt_params . dlen = sizeof ( param ) ;
wmt_params . data = & param ;
wmt_params . status = NULL ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to send wmt func ctrl (%d) " , err ) ;
return err ;
}
2022-01-12 15:39:14 +08:00
set_bit ( BTMTKSDIO_PATCH_ENABLED , & bdev - > tx_state ) ;
2019-03-08 09:15:44 +08:00
ignore_func_on :
/* Apply the low power environment setup */
tci_sleep . mode = 0x5 ;
tci_sleep . duration = cpu_to_le16 ( 0x640 ) ;
tci_sleep . host_duration = cpu_to_le16 ( 0x640 ) ;
tci_sleep . host_wakeup_pin = 0 ;
tci_sleep . time_compensation = 0 ;
skb = __hci_cmd_sync ( hdev , 0xfc7a , sizeof ( tci_sleep ) , & tci_sleep ,
HCI_INIT_TIMEOUT ) ;
if ( IS_ERR ( skb ) ) {
err = PTR_ERR ( skb ) ;
bt_dev_err ( hdev , " Failed to apply low power setting (%d) " , err ) ;
return err ;
}
kfree_skb ( skb ) ;
2021-10-19 05:30:21 +08:00
return 0 ;
}
static int mt79xx_setup ( struct hci_dev * hdev , const char * fwname )
{
2022-01-12 15:39:14 +08:00
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
2021-10-19 05:30:21 +08:00
struct btmtk_hci_wmt_params wmt_params ;
u8 param = 0x1 ;
int err ;
err = btmtk_setup_firmware_79xx ( hdev , fwname , mtk_hci_wmt_sync ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to setup 79xx firmware (%d) " , err ) ;
return err ;
}
2022-05-13 05:38:12 +08:00
err = btmtksdio_fw_pmctrl ( bdev ) ;
if ( err < 0 )
return err ;
err = btmtksdio_drv_pmctrl ( bdev ) ;
if ( err < 0 )
return err ;
2021-10-19 05:30:21 +08:00
/* Enable Bluetooth protocol */
wmt_params . op = BTMTK_WMT_FUNC_CTRL ;
wmt_params . flag = 0 ;
wmt_params . dlen = sizeof ( param ) ;
wmt_params . data = & param ;
wmt_params . status = NULL ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to send wmt func ctrl (%d) " , err ) ;
return err ;
}
2021-12-01 21:13:51 +01:00
hci_set_msft_opcode ( hdev , 0xFD30 ) ;
2021-12-03 11:30:04 +08:00
hci_set_aosp_capable ( hdev ) ;
2022-01-12 15:39:14 +08:00
set_bit ( BTMTKSDIO_PATCH_ENABLED , & bdev - > tx_state ) ;
2021-12-01 21:13:51 +01:00
2021-10-19 05:30:21 +08:00
return err ;
}
2022-01-10 03:23:57 +08:00
static int btmtksdio_mtk_reg_read ( struct hci_dev * hdev , u32 reg , u32 * val )
2021-10-19 05:30:21 +08:00
{
struct btmtk_hci_wmt_params wmt_params ;
2022-01-10 03:24:00 +08:00
struct reg_read_cmd reg_read = {
2021-10-19 05:30:21 +08:00
. type = 1 ,
. num = 1 ,
} ;
u32 status ;
int err ;
reg_read . addr = cpu_to_le32 ( reg ) ;
wmt_params . op = BTMTK_WMT_REGISTER ;
wmt_params . flag = BTMTK_WMT_REG_READ ;
wmt_params . dlen = sizeof ( reg_read ) ;
wmt_params . data = & reg_read ;
wmt_params . status = & status ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 ) {
2022-01-10 03:24:01 +08:00
bt_dev_err ( hdev , " Failed to read reg (%d) " , err ) ;
2021-10-19 05:30:21 +08:00
return err ;
}
* val = status ;
return err ;
}
2022-01-10 03:23:59 +08:00
static int btmtksdio_mtk_reg_write ( struct hci_dev * hdev , u32 reg , u32 val , u32 mask )
{
struct btmtk_hci_wmt_params wmt_params ;
const struct reg_write_cmd reg_write = {
. type = 1 ,
. num = 1 ,
. addr = cpu_to_le32 ( reg ) ,
. data = cpu_to_le32 ( val ) ,
. mask = cpu_to_le32 ( mask ) ,
} ;
int err , status ;
wmt_params . op = BTMTK_WMT_REGISTER ;
wmt_params . flag = BTMTK_WMT_REG_WRITE ;
wmt_params . dlen = sizeof ( reg_write ) ;
wmt_params . data = & reg_write ;
wmt_params . status = & status ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 )
bt_dev_err ( hdev , " Failed to write reg (%d) " , err ) ;
return err ;
}
2022-03-17 07:15:21 +08:00
static int btmtksdio_get_data_path_id ( struct hci_dev * hdev , __u8 * data_path_id )
{
/* uses 1 as data path id for all the usecases */
* data_path_id = 1 ;
return 0 ;
}
2022-03-17 07:15:22 +08:00
static int btmtksdio_get_codec_config_data ( struct hci_dev * hdev ,
__u8 link , struct bt_codec * codec ,
__u8 * ven_len , __u8 * * ven_data )
{
int err = 0 ;
if ( ! ven_data | | ! ven_len )
return - EINVAL ;
* ven_len = 0 ;
* ven_data = NULL ;
if ( link ! = ESCO_LINK ) {
bt_dev_err ( hdev , " Invalid link type(%u) " , link ) ;
return - EINVAL ;
}
* ven_data = kmalloc ( sizeof ( __u8 ) , GFP_KERNEL ) ;
2022-03-23 07:30:40 +08:00
if ( ! * ven_data ) {
2022-03-17 07:15:22 +08:00
err = - ENOMEM ;
goto error ;
}
/* supports only CVSD and mSBC offload codecs */
switch ( codec - > id ) {
case 0x02 :
* * ven_data = 0x00 ;
break ;
case 0x05 :
* * ven_data = 0x01 ;
break ;
default :
err = - EINVAL ;
bt_dev_err ( hdev , " Invalid codec id(%u) " , codec - > id ) ;
goto error ;
}
/* codec and its capabilities are pre-defined to ids
* preset id = 0x00 represents CVSD codec with sampling rate 8 K
* preset id = 0x01 represents mSBC codec with sampling rate 16 K
*/
* ven_len = sizeof ( __u8 ) ;
return err ;
error :
kfree ( * ven_data ) ;
* ven_data = NULL ;
return err ;
}
2022-01-10 03:23:59 +08:00
static int btmtksdio_sco_setting ( struct hci_dev * hdev )
{
const struct btmtk_sco sco_setting = {
. clock_config = 0x49 ,
. channel_format_config = 0x80 ,
} ;
struct sk_buff * skb ;
u32 val ;
int err ;
/* Enable SCO over I2S/PCM for MediaTek chipset */
skb = __hci_cmd_sync ( hdev , 0xfc72 , sizeof ( sco_setting ) ,
& sco_setting , HCI_CMD_TIMEOUT ) ;
if ( IS_ERR ( skb ) )
return PTR_ERR ( skb ) ;
kfree_skb ( skb ) ;
err = btmtksdio_mtk_reg_read ( hdev , MT7921_PINMUX_0 , & val ) ;
if ( err < 0 )
return err ;
val | = 0x11000000 ;
err = btmtksdio_mtk_reg_write ( hdev , MT7921_PINMUX_0 , val , ~ 0 ) ;
if ( err < 0 )
return err ;
err = btmtksdio_mtk_reg_read ( hdev , MT7921_PINMUX_1 , & val ) ;
if ( err < 0 )
return err ;
val | = 0x00000101 ;
2022-03-17 07:15:21 +08:00
err = btmtksdio_mtk_reg_write ( hdev , MT7921_PINMUX_1 , val , ~ 0 ) ;
if ( err < 0 )
return err ;
hdev - > get_data_path_id = btmtksdio_get_data_path_id ;
2022-03-17 07:15:22 +08:00
hdev - > get_codec_config_data = btmtksdio_get_codec_config_data ;
2022-03-17 07:15:21 +08:00
return err ;
2022-01-10 03:23:59 +08:00
}
2022-02-09 02:17:40 +08:00
static int btmtksdio_reset_setting ( struct hci_dev * hdev )
{
int err ;
u32 val ;
err = btmtksdio_mtk_reg_read ( hdev , MT7921_PINMUX_1 , & val ) ;
if ( err < 0 )
return err ;
val | = 0x20 ; /* set the pin (bit field 11:8) work as GPIO mode */
err = btmtksdio_mtk_reg_write ( hdev , MT7921_PINMUX_1 , val , ~ 0 ) ;
if ( err < 0 )
return err ;
err = btmtksdio_mtk_reg_read ( hdev , MT7921_BTSYS_RST , & val ) ;
if ( err < 0 )
return err ;
val | = MT7921_BTSYS_RST_WITH_GPIO ;
return btmtksdio_mtk_reg_write ( hdev , MT7921_BTSYS_RST , val , ~ 0 ) ;
}
2021-10-19 05:30:21 +08:00
static int btmtksdio_setup ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
ktime_t calltime , delta , rettime ;
unsigned long long duration ;
char fwname [ 64 ] ;
int err , dev_id ;
2022-02-09 02:17:40 +08:00
u32 fw_version = 0 , val ;
2021-10-19 05:30:21 +08:00
calltime = ktime_get ( ) ;
2021-11-20 06:25:43 +08:00
set_bit ( BTMTKSDIO_HW_TX_READY , & bdev - > tx_state ) ;
2021-10-19 05:30:21 +08:00
switch ( bdev - > data - > chipid ) {
case 0x7921 :
2022-02-09 02:17:40 +08:00
if ( test_bit ( BTMTKSDIO_HW_RESET_ACTIVE , & bdev - > tx_state ) ) {
err = btmtksdio_mtk_reg_read ( hdev , MT7921_DLSTATUS ,
& val ) ;
if ( err < 0 )
return err ;
val & = ~ BT_DL_STATE ;
err = btmtksdio_mtk_reg_write ( hdev , MT7921_DLSTATUS ,
val , ~ 0 ) ;
if ( err < 0 )
return err ;
btmtksdio_fw_pmctrl ( bdev ) ;
msleep ( 20 ) ;
btmtksdio_drv_pmctrl ( bdev ) ;
clear_bit ( BTMTKSDIO_HW_RESET_ACTIVE , & bdev - > tx_state ) ;
}
2022-01-10 03:23:57 +08:00
err = btmtksdio_mtk_reg_read ( hdev , 0x70010200 , & dev_id ) ;
2021-10-19 05:30:21 +08:00
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to get device id (%d) " , err ) ;
return err ;
}
2022-01-10 03:23:57 +08:00
err = btmtksdio_mtk_reg_read ( hdev , 0x80021004 , & fw_version ) ;
2021-10-19 05:30:21 +08:00
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to get fw version (%d) " , err ) ;
return err ;
}
snprintf ( fwname , sizeof ( fwname ) ,
" mediatek/BT_RAM_CODE_MT%04x_1_%x_hdr.bin " ,
dev_id & 0xffff , ( fw_version & 0xff ) + 1 ) ;
err = mt79xx_setup ( hdev , fwname ) ;
if ( err < 0 )
return err ;
2022-01-10 03:23:59 +08:00
/* Enable SCO over I2S/PCM */
err = btmtksdio_sco_setting ( hdev ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to enable SCO setting (%d) " , err ) ;
return err ;
}
2022-03-17 07:15:23 +08:00
/* Enable WBS with mSBC codec */
set_bit ( HCI_QUIRK_WIDEBAND_SPEECH_SUPPORTED , & hdev - > quirks ) ;
2022-02-09 02:17:40 +08:00
/* Enable GPIO reset mechanism */
if ( bdev - > reset ) {
err = btmtksdio_reset_setting ( hdev ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to enable Reset setting (%d) " , err ) ;
devm_gpiod_put ( bdev - > dev , bdev - > reset ) ;
bdev - > reset = NULL ;
}
}
2022-03-17 07:15:20 +08:00
/* Valid LE States quirk for MediaTek 7921 */
set_bit ( HCI_QUIRK_VALID_LE_STATES , & hdev - > quirks ) ;
2021-10-19 05:30:21 +08:00
break ;
case 0x7663 :
case 0x7668 :
err = mt76xx_setup ( hdev , bdev - > data - > fwname ) ;
if ( err < 0 )
return err ;
break ;
default :
return - ENODEV ;
}
2019-03-08 09:15:44 +08:00
rettime = ktime_get ( ) ;
delta = ktime_sub ( rettime , calltime ) ;
duration = ( unsigned long long ) ktime_to_ns ( delta ) > > 10 ;
2019-04-18 17:08:02 +08:00
pm_runtime_set_autosuspend_delay ( bdev - > dev ,
MTKBTSDIO_AUTOSUSPEND_DELAY ) ;
pm_runtime_use_autosuspend ( bdev - > dev ) ;
err = pm_runtime_set_active ( bdev - > dev ) ;
if ( err < 0 )
return err ;
/* Default forbid runtime auto suspend, that can be allowed by
* enable_autosuspend flag or the PM runtime entry under sysfs .
*/
pm_runtime_forbid ( bdev - > dev ) ;
pm_runtime_enable ( bdev - > dev ) ;
if ( enable_autosuspend )
pm_runtime_allow ( bdev - > dev ) ;
2019-03-08 09:15:44 +08:00
bt_dev_info ( hdev , " Device setup in %llu usecs " , duration ) ;
return 0 ;
}
static int btmtksdio_shutdown ( struct hci_dev * hdev )
{
2019-04-18 17:08:02 +08:00
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
2019-03-08 09:15:44 +08:00
struct btmtk_hci_wmt_params wmt_params ;
u8 param = 0x0 ;
int err ;
2019-04-18 17:08:02 +08:00
/* Get back the state to be consistent with the state
* in btmtksdio_setup .
*/
pm_runtime_get_sync ( bdev - > dev ) ;
2022-05-13 05:38:11 +08:00
/* wmt command only works until the reset is complete */
if ( test_bit ( BTMTKSDIO_HW_RESET_ACTIVE , & bdev - > tx_state ) )
goto ignore_wmt_cmd ;
2019-03-08 09:15:44 +08:00
/* Disable the device */
2021-10-19 05:30:13 +08:00
wmt_params . op = BTMTK_WMT_FUNC_CTRL ;
2019-03-08 09:15:44 +08:00
wmt_params . flag = 0 ;
wmt_params . dlen = sizeof ( param ) ;
wmt_params . data = & param ;
wmt_params . status = NULL ;
err = mtk_hci_wmt_sync ( hdev , & wmt_params ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to send wmt func ctrl (%d) " , err ) ;
return err ;
}
2022-05-13 05:38:11 +08:00
ignore_wmt_cmd :
2019-04-18 17:08:02 +08:00
pm_runtime_put_noidle ( bdev - > dev ) ;
pm_runtime_disable ( bdev - > dev ) ;
2019-03-08 09:15:44 +08:00
return 0 ;
}
static int btmtksdio_send_frame ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
switch ( hci_skb_pkt_type ( skb ) ) {
case HCI_COMMAND_PKT :
hdev - > stat . cmd_tx + + ;
break ;
case HCI_ACLDATA_PKT :
hdev - > stat . acl_tx + + ;
break ;
case HCI_SCODATA_PKT :
hdev - > stat . sco_tx + + ;
break ;
default :
return - EILSEQ ;
}
skb_queue_tail ( & bdev - > txq , skb ) ;
2021-10-19 05:30:16 +08:00
schedule_work ( & bdev - > txrx_work ) ;
2019-03-08 09:15:44 +08:00
return 0 ;
}
2022-02-09 02:17:40 +08:00
static void btmtksdio_cmd_timeout ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
u32 status ;
int err ;
if ( ! bdev - > reset | | bdev - > data - > chipid ! = 0x7921 )
return ;
pm_runtime_get_sync ( bdev - > dev ) ;
if ( test_and_set_bit ( BTMTKSDIO_HW_RESET_ACTIVE , & bdev - > tx_state ) )
return ;
sdio_claim_host ( bdev - > func ) ;
sdio_writel ( bdev - > func , C_INT_EN_CLR , MTK_REG_CHLPCR , NULL ) ;
skb_queue_purge ( & bdev - > txq ) ;
cancel_work_sync ( & bdev - > txrx_work ) ;
gpiod_set_value_cansleep ( bdev - > reset , 1 ) ;
msleep ( 100 ) ;
gpiod_set_value_cansleep ( bdev - > reset , 0 ) ;
err = readx_poll_timeout ( btmtksdio_chcr_query , bdev , status ,
status & BT_RST_DONE , 100000 , 2000000 ) ;
if ( err < 0 ) {
bt_dev_err ( hdev , " Failed to reset (%d) " , err ) ;
goto err ;
}
clear_bit ( BTMTKSDIO_PATCH_ENABLED , & bdev - > tx_state ) ;
err :
sdio_release_host ( bdev - > func ) ;
pm_runtime_put_noidle ( bdev - > dev ) ;
pm_runtime_disable ( bdev - > dev ) ;
hci_reset_dev ( hdev ) ;
}
2022-06-27 07:27:36 +08:00
static bool btmtksdio_sdio_inband_wakeup ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
return device_may_wakeup ( bdev - > dev ) ;
}
2022-01-10 03:23:58 +08:00
static bool btmtksdio_sdio_wakeup ( struct hci_dev * hdev )
{
struct btmtksdio_dev * bdev = hci_get_drvdata ( hdev ) ;
bool may_wakeup = device_may_wakeup ( bdev - > dev ) ;
const struct btmtk_wakeon bt_awake = {
. mode = 0x1 ,
. gpo = 0 ,
. active_high = 0x1 ,
. enable_delay = cpu_to_le16 ( 0xc80 ) ,
. wakeup_delay = cpu_to_le16 ( 0x20 ) ,
} ;
if ( may_wakeup & & bdev - > data - > chipid = = 0x7921 ) {
struct sk_buff * skb ;
skb = __hci_cmd_sync ( hdev , 0xfc27 , sizeof ( bt_awake ) ,
& bt_awake , HCI_CMD_TIMEOUT ) ;
if ( IS_ERR ( skb ) )
may_wakeup = false ;
2022-02-23 07:56:00 +08:00
else
kfree_skb ( skb ) ;
2022-01-10 03:23:58 +08:00
}
return may_wakeup ;
}
2019-03-08 09:15:44 +08:00
static int btmtksdio_probe ( struct sdio_func * func ,
const struct sdio_device_id * id )
{
struct btmtksdio_dev * bdev ;
struct hci_dev * hdev ;
int err ;
bdev = devm_kzalloc ( & func - > dev , sizeof ( * bdev ) , GFP_KERNEL ) ;
if ( ! bdev )
return - ENOMEM ;
bdev - > data = ( void * ) id - > driver_data ;
if ( ! bdev - > data )
return - ENODEV ;
2019-04-18 17:08:02 +08:00
bdev - > dev = & func - > dev ;
2019-03-08 09:15:44 +08:00
bdev - > func = func ;
2021-10-19 05:30:16 +08:00
INIT_WORK ( & bdev - > txrx_work , btmtksdio_txrx_work ) ;
2019-03-08 09:15:44 +08:00
skb_queue_head_init ( & bdev - > txq ) ;
/* Initialize and register HCI device */
hdev = hci_alloc_dev ( ) ;
if ( ! hdev ) {
dev_err ( & func - > dev , " Can't allocate HCI device \n " ) ;
return - ENOMEM ;
}
bdev - > hdev = hdev ;
hdev - > bus = HCI_SDIO ;
hci_set_drvdata ( hdev , bdev ) ;
hdev - > open = btmtksdio_open ;
hdev - > close = btmtksdio_close ;
2022-02-09 02:17:40 +08:00
hdev - > cmd_timeout = btmtksdio_cmd_timeout ;
2019-03-08 09:15:44 +08:00
hdev - > flush = btmtksdio_flush ;
hdev - > setup = btmtksdio_setup ;
hdev - > shutdown = btmtksdio_shutdown ;
hdev - > send = btmtksdio_send_frame ;
2022-01-10 03:23:58 +08:00
hdev - > wakeup = btmtksdio_sdio_wakeup ;
2022-06-27 07:27:36 +08:00
/*
* If SDIO controller supports wake on Bluetooth , sending a wakeon
* command is not necessary .
*/
if ( device_can_wakeup ( func - > card - > host - > parent ) )
hdev - > wakeup = btmtksdio_sdio_inband_wakeup ;
else
hdev - > wakeup = btmtksdio_sdio_wakeup ;
2021-10-19 05:30:14 +08:00
hdev - > set_bdaddr = btmtk_set_bdaddr ;
2019-03-08 09:15:44 +08:00
SET_HCIDEV_DEV ( hdev , & func - > dev ) ;
hdev - > manufacturer = 70 ;
set_bit ( HCI_QUIRK_NON_PERSISTENT_SETUP , & hdev - > quirks ) ;
2022-03-17 07:15:19 +08:00
sdio_set_drvdata ( func , bdev ) ;
2019-03-08 09:15:44 +08:00
err = hci_register_dev ( hdev ) ;
if ( err < 0 ) {
dev_err ( & func - > dev , " Can't register HCI device \n " ) ;
hci_free_dev ( hdev ) ;
return err ;
}
2019-04-18 17:08:02 +08:00
/* pm_runtime_enable would be done after the firmware is being
* downloaded because the core layer probably already enables
* runtime PM for this func such as the case host - > caps &
* MMC_CAP_POWER_OFF_CARD .
*/
if ( pm_runtime_enabled ( bdev - > dev ) )
pm_runtime_disable ( bdev - > dev ) ;
/* As explaination in drivers/mmc/core/sdio_bus.c tells us:
* Unbound SDIO functions are always suspended .
* During probe , the function is set active and the usage count
* is incremented . If the driver supports runtime PM ,
* it should call pm_runtime_put_noidle ( ) in its probe routine and
* pm_runtime_get_noresume ( ) in its remove routine .
*
* So , put a pm_runtime_put_noidle here !
*/
pm_runtime_put_noidle ( bdev - > dev ) ;
2022-01-10 03:23:58 +08:00
err = device_init_wakeup ( bdev - > dev , true ) ;
if ( err )
bt_dev_err ( hdev , " failed to initialize device wakeup " ) ;
2022-02-09 02:17:40 +08:00
bdev - > dev - > of_node = of_find_compatible_node ( NULL , NULL ,
" mediatek,mt7921s-bluetooth " ) ;
bdev - > reset = devm_gpiod_get_optional ( bdev - > dev , " reset " ,
GPIOD_OUT_LOW ) ;
if ( IS_ERR ( bdev - > reset ) )
err = PTR_ERR ( bdev - > reset ) ;
2022-01-10 03:23:58 +08:00
return err ;
2019-03-08 09:15:44 +08:00
}
static void btmtksdio_remove ( struct sdio_func * func )
{
struct btmtksdio_dev * bdev = sdio_get_drvdata ( func ) ;
struct hci_dev * hdev ;
if ( ! bdev )
return ;
2019-04-18 17:08:02 +08:00
/* Be consistent the state in btmtksdio_probe */
pm_runtime_get_noresume ( bdev - > dev ) ;
2019-03-08 09:15:44 +08:00
hdev = bdev - > hdev ;
sdio_set_drvdata ( func , NULL ) ;
hci_unregister_dev ( hdev ) ;
hci_free_dev ( hdev ) ;
}
2019-04-18 17:08:02 +08:00
# ifdef CONFIG_PM
static int btmtksdio_runtime_suspend ( struct device * dev )
{
struct sdio_func * func = dev_to_sdio_func ( dev ) ;
struct btmtksdio_dev * bdev ;
int err ;
bdev = sdio_get_drvdata ( func ) ;
if ( ! bdev )
return 0 ;
2021-12-02 02:02:46 +08:00
if ( ! test_bit ( BTMTKSDIO_FUNC_ENABLED , & bdev - > tx_state ) )
return 0 ;
2021-12-02 02:02:47 +08:00
sdio_set_host_pm_flags ( func , MMC_PM_KEEP_POWER ) ;
2022-01-12 15:39:12 +08:00
err = btmtksdio_fw_pmctrl ( bdev ) ;
2019-04-18 17:08:02 +08:00
2022-01-12 15:39:15 +08:00
bt_dev_dbg ( bdev - > hdev , " status (%d) return ownership to device " , err ) ;
2019-04-18 17:08:02 +08:00
return err ;
}
static int btmtksdio_runtime_resume ( struct device * dev )
{
struct sdio_func * func = dev_to_sdio_func ( dev ) ;
struct btmtksdio_dev * bdev ;
int err ;
bdev = sdio_get_drvdata ( func ) ;
if ( ! bdev )
return 0 ;
2021-12-02 02:02:46 +08:00
if ( ! test_bit ( BTMTKSDIO_FUNC_ENABLED , & bdev - > tx_state ) )
return 0 ;
2022-01-12 15:39:12 +08:00
err = btmtksdio_drv_pmctrl ( bdev ) ;
2019-04-18 17:08:02 +08:00
2022-01-12 15:39:15 +08:00
bt_dev_dbg ( bdev - > hdev , " status (%d) get ownership from device " , err ) ;
2019-04-18 17:08:02 +08:00
return err ;
}
static UNIVERSAL_DEV_PM_OPS ( btmtksdio_pm_ops , btmtksdio_runtime_suspend ,
btmtksdio_runtime_resume , NULL ) ;
# define BTMTKSDIO_PM_OPS (&btmtksdio_pm_ops)
# else /* CONFIG_PM */
# define BTMTKSDIO_PM_OPS NULL
# endif /* CONFIG_PM */
2019-03-08 09:15:44 +08:00
static struct sdio_driver btmtksdio_driver = {
. name = " btmtksdio " ,
. probe = btmtksdio_probe ,
. remove = btmtksdio_remove ,
. id_table = btmtksdio_table ,
2019-04-18 17:08:02 +08:00
. drv = {
. owner = THIS_MODULE ,
. pm = BTMTKSDIO_PM_OPS ,
}
2019-03-08 09:15:44 +08:00
} ;
2019-03-14 05:01:59 +08:00
module_sdio_driver ( btmtksdio_driver ) ;
2019-03-08 09:15:44 +08:00
2019-04-18 17:08:02 +08:00
module_param ( enable_autosuspend , bool , 0644 ) ;
MODULE_PARM_DESC ( enable_autosuspend , " Enable autosuspend by default " ) ;
2019-03-08 09:15:44 +08:00
MODULE_AUTHOR ( " Sean Wang <sean.wang@mediatek.com> " ) ;
MODULE_DESCRIPTION ( " MediaTek Bluetooth SDIO driver ver " VERSION ) ;
MODULE_VERSION ( VERSION ) ;
MODULE_LICENSE ( " GPL " ) ;