Commit 3a722044 authored by Sean Wang's avatar Sean Wang Committed by Marcel Holtmann

Bluetooth: btmtksido: rely on BT_MTK module

Rely on btmtk module to reduce duplicated code
Signed-off-by: default avatarSean Wang <sean.wang@mediatek.com>
Signed-off-by: default avatarMarcel Holtmann <marcel@holtmann.org>
parent 8c0d17b6
...@@ -388,6 +388,7 @@ config BT_ATH3K ...@@ -388,6 +388,7 @@ config BT_ATH3K
config BT_MTKSDIO config BT_MTKSDIO
tristate "MediaTek HCI SDIO driver" tristate "MediaTek HCI SDIO driver"
depends on MMC depends on MMC
select BT_MTK
help help
MediaTek Bluetooth HCI SDIO driver. MediaTek Bluetooth HCI SDIO driver.
This driver is required if you want to use MediaTek Bluetooth This driver is required if you want to use MediaTek Bluetooth
......
...@@ -9,6 +9,9 @@ ...@@ -9,6 +9,9 @@
enum { enum {
BTMTK_WMT_PATCH_DWNLD = 0x1, BTMTK_WMT_PATCH_DWNLD = 0x1,
BTMTK_WMT_TEST = 0x2,
BTMTK_WMT_WAKEUP = 0x3,
BTMTK_WMT_HIF = 0x4,
BTMTK_WMT_FUNC_CTRL = 0x6, BTMTK_WMT_FUNC_CTRL = 0x6,
BTMTK_WMT_RST = 0x7, BTMTK_WMT_RST = 0x7,
BTMTK_WMT_SEMAPHORE = 0x17, BTMTK_WMT_SEMAPHORE = 0x17,
......
...@@ -12,7 +12,6 @@ ...@@ -12,7 +12,6 @@
#include <asm/unaligned.h> #include <asm/unaligned.h>
#include <linux/atomic.h> #include <linux/atomic.h>
#include <linux/firmware.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/iopoll.h> #include <linux/iopoll.h>
#include <linux/kernel.h> #include <linux/kernel.h>
...@@ -28,12 +27,10 @@ ...@@ -28,12 +27,10 @@
#include <net/bluetooth/hci_core.h> #include <net/bluetooth/hci_core.h>
#include "h4_recv.h" #include "h4_recv.h"
#include "btmtk.h"
#define VERSION "0.1" #define VERSION "0.1"
#define FIRMWARE_MT7663 "mediatek/mt7663pr2h.bin"
#define FIRMWARE_MT7668 "mediatek/mt7668pr2h.bin"
#define MTKBTSDIO_AUTOSUSPEND_DELAY 8000 #define MTKBTSDIO_AUTOSUSPEND_DELAY 8000
static bool enable_autosuspend; static bool enable_autosuspend;
...@@ -90,69 +87,12 @@ MODULE_DEVICE_TABLE(sdio, btmtksdio_table); ...@@ -90,69 +87,12 @@ MODULE_DEVICE_TABLE(sdio, btmtksdio_table);
#define BTMTKSDIO_TX_WAIT_VND_EVT 1 #define BTMTKSDIO_TX_WAIT_VND_EVT 1
enum {
MTK_WMT_PATCH_DWNLD = 0x1,
MTK_WMT_TEST = 0x2,
MTK_WMT_WAKEUP = 0x3,
MTK_WMT_HIF = 0x4,
MTK_WMT_FUNC_CTRL = 0x6,
MTK_WMT_RST = 0x7,
MTK_WMT_SEMAPHORE = 0x17,
};
enum {
BTMTK_WMT_INVALID,
BTMTK_WMT_PATCH_UNDONE,
BTMTK_WMT_PATCH_DONE,
BTMTK_WMT_ON_UNDONE,
BTMTK_WMT_ON_DONE,
BTMTK_WMT_ON_PROGRESS,
};
struct mtkbtsdio_hdr { struct mtkbtsdio_hdr {
__le16 len; __le16 len;
__le16 reserved; __le16 reserved;
u8 bt_type; u8 bt_type;
} __packed; } __packed;
struct mtk_wmt_hdr {
u8 dir;
u8 op;
__le16 dlen;
u8 flag;
} __packed;
struct mtk_hci_wmt_cmd {
struct mtk_wmt_hdr hdr;
u8 data[256];
} __packed;
struct btmtk_hci_wmt_evt {
struct hci_event_hdr hhdr;
struct mtk_wmt_hdr whdr;
} __packed;
struct btmtk_hci_wmt_evt_funcc {
struct btmtk_hci_wmt_evt hwhdr;
__be16 status;
} __packed;
struct btmtk_tci_sleep {
u8 mode;
__le16 duration;
__le16 host_duration;
u8 host_wakeup_pin;
u8 time_compensation;
} __packed;
struct btmtk_hci_wmt_params {
u8 op;
u8 flag;
u16 dlen;
const void *data;
u32 *status;
};
struct btmtksdio_dev { struct btmtksdio_dev {
struct hci_dev *hdev; struct hci_dev *hdev;
struct sdio_func *func; struct sdio_func *func;
...@@ -174,27 +114,32 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev, ...@@ -174,27 +114,32 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev,
struct btmtk_hci_wmt_evt_funcc *wmt_evt_funcc; struct btmtk_hci_wmt_evt_funcc *wmt_evt_funcc;
u32 hlen, status = BTMTK_WMT_INVALID; u32 hlen, status = BTMTK_WMT_INVALID;
struct btmtk_hci_wmt_evt *wmt_evt; struct btmtk_hci_wmt_evt *wmt_evt;
struct mtk_hci_wmt_cmd wc; struct btmtk_hci_wmt_cmd *wc;
struct mtk_wmt_hdr *hdr; struct btmtk_wmt_hdr *hdr;
int err; int err;
/* Send the WMT command and wait until the WMT event returns */
hlen = sizeof(*hdr) + wmt_params->dlen; hlen = sizeof(*hdr) + wmt_params->dlen;
if (hlen > 255) if (hlen > 255)
return -EINVAL; return -EINVAL;
hdr = (struct mtk_wmt_hdr *)&wc; wc = kzalloc(hlen, GFP_KERNEL);
if (!wc)
return -ENOMEM;
hdr = &wc->hdr;
hdr->dir = 1; hdr->dir = 1;
hdr->op = wmt_params->op; hdr->op = wmt_params->op;
hdr->dlen = cpu_to_le16(wmt_params->dlen + 1); hdr->dlen = cpu_to_le16(wmt_params->dlen + 1);
hdr->flag = wmt_params->flag; hdr->flag = wmt_params->flag;
memcpy(wc.data, wmt_params->data, wmt_params->dlen); memcpy(wc->data, wmt_params->data, wmt_params->dlen);
set_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state); set_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state);
err = __hci_cmd_send(hdev, 0xfc6f, hlen, &wc); err = __hci_cmd_send(hdev, 0xfc6f, hlen, wc);
if (err < 0) { if (err < 0) {
clear_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state); clear_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state);
return err; goto err_free_wc;
} }
/* The vendor specific WMT commands are all answered by a vendor /* The vendor specific WMT commands are all answered by a vendor
...@@ -211,13 +156,14 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev, ...@@ -211,13 +156,14 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev,
if (err == -EINTR) { if (err == -EINTR) {
bt_dev_err(hdev, "Execution of wmt command interrupted"); bt_dev_err(hdev, "Execution of wmt command interrupted");
clear_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state); clear_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state);
return err; goto err_free_wc;
} }
if (err) { if (err) {
bt_dev_err(hdev, "Execution of wmt command timed out"); bt_dev_err(hdev, "Execution of wmt command timed out");
clear_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state); clear_bit(BTMTKSDIO_TX_WAIT_VND_EVT, &bdev->tx_state);
return -ETIMEDOUT; err = -ETIMEDOUT;
goto err_free_wc;
} }
/* Parse and handle the return WMT event */ /* Parse and handle the return WMT event */
...@@ -230,13 +176,13 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev, ...@@ -230,13 +176,13 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev,
} }
switch (wmt_evt->whdr.op) { switch (wmt_evt->whdr.op) {
case MTK_WMT_SEMAPHORE: case BTMTK_WMT_SEMAPHORE:
if (wmt_evt->whdr.flag == 2) if (wmt_evt->whdr.flag == 2)
status = BTMTK_WMT_PATCH_UNDONE; status = BTMTK_WMT_PATCH_UNDONE;
else else
status = BTMTK_WMT_PATCH_DONE; status = BTMTK_WMT_PATCH_DONE;
break; break;
case MTK_WMT_FUNC_CTRL: case BTMTK_WMT_FUNC_CTRL:
wmt_evt_funcc = (struct btmtk_hci_wmt_evt_funcc *)wmt_evt; wmt_evt_funcc = (struct btmtk_hci_wmt_evt_funcc *)wmt_evt;
if (be16_to_cpu(wmt_evt_funcc->status) == 0x404) if (be16_to_cpu(wmt_evt_funcc->status) == 0x404)
status = BTMTK_WMT_ON_DONE; status = BTMTK_WMT_ON_DONE;
...@@ -253,6 +199,8 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev, ...@@ -253,6 +199,8 @@ static int mtk_hci_wmt_sync(struct hci_dev *hdev,
err_free_skb: err_free_skb:
kfree_skb(bdev->evt_skb); kfree_skb(bdev->evt_skb);
bdev->evt_skb = NULL; bdev->evt_skb = NULL;
err_free_wc:
kfree(wc);
return err; return err;
} }
...@@ -663,7 +611,7 @@ static int btmtksdio_func_query(struct hci_dev *hdev) ...@@ -663,7 +611,7 @@ static int btmtksdio_func_query(struct hci_dev *hdev)
u8 param = 0; u8 param = 0;
/* Query whether the function is enabled */ /* Query whether the function is enabled */
wmt_params.op = MTK_WMT_FUNC_CTRL; wmt_params.op = BTMTK_WMT_FUNC_CTRL;
wmt_params.flag = 4; wmt_params.flag = 4;
wmt_params.dlen = sizeof(param); wmt_params.dlen = sizeof(param);
wmt_params.data = &param; wmt_params.data = &param;
...@@ -678,96 +626,6 @@ static int btmtksdio_func_query(struct hci_dev *hdev) ...@@ -678,96 +626,6 @@ static int btmtksdio_func_query(struct hci_dev *hdev)
return status; return status;
} }
static int mtk_setup_firmware(struct hci_dev *hdev, const char *fwname)
{
struct btmtk_hci_wmt_params wmt_params;
const struct firmware *fw;
const u8 *fw_ptr;
size_t fw_size;
int err, dlen;
u8 flag, param;
err = request_firmware(&fw, fwname, &hdev->dev);
if (err < 0) {
bt_dev_err(hdev, "Failed to load firmware file (%d)", err);
return err;
}
/* Power on data RAM the firmware relies on. */
param = 1;
wmt_params.op = MTK_WMT_FUNC_CTRL;
wmt_params.flag = 3;
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 power on data RAM (%d)", err);
goto free_fw;
}
fw_ptr = fw->data;
fw_size = fw->size;
/* The size of patch header is 30 bytes, should be skip */
if (fw_size < 30) {
err = -EINVAL;
goto free_fw;
}
fw_size -= 30;
fw_ptr += 30;
flag = 1;
wmt_params.op = MTK_WMT_PATCH_DWNLD;
wmt_params.status = NULL;
while (fw_size > 0) {
dlen = min_t(int, 250, fw_size);
/* Tell device the position in sequence */
if (fw_size - dlen <= 0)
flag = 3;
else if (fw_size < fw->size - 30)
flag = 2;
wmt_params.flag = flag;
wmt_params.dlen = dlen;
wmt_params.data = fw_ptr;
err = mtk_hci_wmt_sync(hdev, &wmt_params);
if (err < 0) {
bt_dev_err(hdev, "Failed to send wmt patch dwnld (%d)",
err);
goto free_fw;
}
fw_size -= dlen;
fw_ptr += dlen;
}
wmt_params.op = MTK_WMT_RST;
wmt_params.flag = 4;
wmt_params.dlen = 0;
wmt_params.data = NULL;
wmt_params.status = NULL;
/* Activate funciton the firmware providing to */
err = mtk_hci_wmt_sync(hdev, &wmt_params);
if (err < 0) {
bt_dev_err(hdev, "Failed to send wmt rst (%d)", err);
goto free_fw;
}
/* Wait a few moments for firmware activation done */
usleep_range(10000, 12000);
free_fw:
release_firmware(fw);
return err;
}
static int btmtksdio_setup(struct hci_dev *hdev) static int btmtksdio_setup(struct hci_dev *hdev)
{ {
struct btmtksdio_dev *bdev = hci_get_drvdata(hdev); struct btmtksdio_dev *bdev = hci_get_drvdata(hdev);
...@@ -782,7 +640,7 @@ static int btmtksdio_setup(struct hci_dev *hdev) ...@@ -782,7 +640,7 @@ static int btmtksdio_setup(struct hci_dev *hdev)
calltime = ktime_get(); calltime = ktime_get();
/* Query whether the firmware is already download */ /* Query whether the firmware is already download */
wmt_params.op = MTK_WMT_SEMAPHORE; wmt_params.op = BTMTK_WMT_SEMAPHORE;
wmt_params.flag = 1; wmt_params.flag = 1;
wmt_params.dlen = 0; wmt_params.dlen = 0;
wmt_params.data = NULL; wmt_params.data = NULL;
...@@ -800,7 +658,7 @@ static int btmtksdio_setup(struct hci_dev *hdev) ...@@ -800,7 +658,7 @@ static int btmtksdio_setup(struct hci_dev *hdev)
} }
/* Setup a firmware which the device definitely requires */ /* Setup a firmware which the device definitely requires */
err = mtk_setup_firmware(hdev, bdev->data->fwname); err = btmtk_setup_firmware(hdev, bdev->data->fwname, mtk_hci_wmt_sync);
if (err < 0) if (err < 0)
return err; return err;
...@@ -823,7 +681,7 @@ static int btmtksdio_setup(struct hci_dev *hdev) ...@@ -823,7 +681,7 @@ static int btmtksdio_setup(struct hci_dev *hdev)
} }
/* Enable Bluetooth protocol */ /* Enable Bluetooth protocol */
wmt_params.op = MTK_WMT_FUNC_CTRL; wmt_params.op = BTMTK_WMT_FUNC_CTRL;
wmt_params.flag = 0; wmt_params.flag = 0;
wmt_params.dlen = sizeof(param); wmt_params.dlen = sizeof(param);
wmt_params.data = &param; wmt_params.data = &param;
...@@ -891,7 +749,7 @@ static int btmtksdio_shutdown(struct hci_dev *hdev) ...@@ -891,7 +749,7 @@ static int btmtksdio_shutdown(struct hci_dev *hdev)
pm_runtime_get_sync(bdev->dev); pm_runtime_get_sync(bdev->dev);
/* Disable the device */ /* Disable the device */
wmt_params.op = MTK_WMT_FUNC_CTRL; wmt_params.op = BTMTK_WMT_FUNC_CTRL;
wmt_params.flag = 0; wmt_params.flag = 0;
wmt_params.dlen = sizeof(param); wmt_params.dlen = sizeof(param);
wmt_params.data = &param; wmt_params.data = &param;
...@@ -1112,5 +970,3 @@ MODULE_AUTHOR("Sean Wang <sean.wang@mediatek.com>"); ...@@ -1112,5 +970,3 @@ MODULE_AUTHOR("Sean Wang <sean.wang@mediatek.com>");
MODULE_DESCRIPTION("MediaTek Bluetooth SDIO driver ver " VERSION); MODULE_DESCRIPTION("MediaTek Bluetooth SDIO driver ver " VERSION);
MODULE_VERSION(VERSION); MODULE_VERSION(VERSION);
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_FIRMWARE(FIRMWARE_MT7663);
MODULE_FIRMWARE(FIRMWARE_MT7668);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment