389 lines
12 KiB
Diff
389 lines
12 KiB
Diff
diff -abBur linux-5.14.11/arch/arm64/boot/dts/allwinner/sun50i-a64-pinebook.dts linux-5.14.11-patched/arch/arm64/boot/dts/allwinner/sun50i-a64-pinebook.dts
|
|
--- linux-5.14.11/arch/arm64/boot/dts/allwinner/sun50i-a64-pinebook.dts 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/arch/arm64/boot/dts/allwinner/sun50i-a64-pinebook.dts 2021-10-11 23:16:51.530740961 +0200
|
|
@@ -405,6 +405,32 @@
|
|
status = "okay";
|
|
};
|
|
|
|
+
|
|
+&uart1 {
|
|
+ pinctrl-names = "default";
|
|
+ pinctrl-0 = <&uart1_pins>, <&uart1_rts_cts_pins>;
|
|
+ status = "okay";
|
|
+
|
|
+ bluetooth {
|
|
+ compatible = "realtek,rtl8723cs-bt";
|
|
+ device-wake-gpios = <&r_pio 0 5 GPIO_ACTIVE_LOW>; /* PL5 */
|
|
+ host-wake-gpios = <&r_pio 0 6 GPIO_ACTIVE_HIGH>; /* PL6 */
|
|
+ };
|
|
+};
|
|
+
|
|
+&uart1 {
|
|
+ pinctrl-names = "default";
|
|
+ pinctrl-0 = <&uart1_pins>, <&uart1_rts_cts_pins>;
|
|
+ uart-has-rtscts;
|
|
+ status = "okay";
|
|
+
|
|
+ bluetooth {
|
|
+ compatible = "realtek,rtl8723cs-bt";
|
|
+ device-wake-gpios = <&r_pio 0 5 GPIO_ACTIVE_LOW>; /* PL5 */
|
|
+ host-wake-gpios = <&r_pio 0 6 GPIO_ACTIVE_HIGH>; /* PL6 */
|
|
+ };
|
|
+};
|
|
+
|
|
&usb_otg {
|
|
dr_mode = "host";
|
|
};
|
|
diff -abBur linux-5.14.11/drivers/bluetooth/btrtl.c linux-5.14.11-patched/drivers/bluetooth/btrtl.c
|
|
--- linux-5.14.11/drivers/bluetooth/btrtl.c 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/drivers/bluetooth/btrtl.c 2021-10-11 23:06:01.518740651 +0200
|
|
@@ -17,7 +17,11 @@
|
|
|
|
#define VERSION "0.1"
|
|
|
|
+#define RTL_CHIP_8723CS_CG 3
|
|
+#define RTL_CHIP_8723CS_VF 4
|
|
+#define RTL_CHIP_8723CS_XX 5
|
|
#define RTL_EPATCH_SIGNATURE "Realtech"
|
|
+#define RTL_ROM_LMP_8703B 0x8703
|
|
#define RTL_ROM_LMP_8723A 0x1200
|
|
#define RTL_ROM_LMP_8723B 0x8723
|
|
#define RTL_ROM_LMP_8821A 0x8821
|
|
@@ -30,6 +34,7 @@
|
|
#define IC_MATCH_FL_HCIREV (1 << 1)
|
|
#define IC_MATCH_FL_HCIVER (1 << 2)
|
|
#define IC_MATCH_FL_HCIBUS (1 << 3)
|
|
+#define IC_MATCH_FL_CHIP_TYPE (1 << 4)
|
|
#define IC_INFO(lmps, hcir, hciv, bus) \
|
|
.match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_HCIREV | \
|
|
IC_MATCH_FL_HCIVER | IC_MATCH_FL_HCIBUS, \
|
|
@@ -57,6 +62,7 @@
|
|
__u16 hci_rev;
|
|
__u8 hci_ver;
|
|
__u8 hci_bus;
|
|
+ __u8 chip_type;
|
|
bool config_needed;
|
|
bool has_rom_version;
|
|
char *fw_name;
|
|
@@ -96,6 +102,72 @@
|
|
.fw_name = "rtl_bt/rtl8723b_fw.bin",
|
|
.cfg_name = "rtl_bt/rtl8723b_config" },
|
|
|
|
+ /* 8723CS-CG */
|
|
+ { .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_CHIP_TYPE |
|
|
+ IC_MATCH_FL_HCIBUS,
|
|
+ .lmp_subver = RTL_ROM_LMP_8703B,
|
|
+ .chip_type = RTL_CHIP_8723CS_CG,
|
|
+ .hci_bus = HCI_UART,
|
|
+ .config_needed = true,
|
|
+ .has_rom_version = true,
|
|
+ .fw_name = "rtl_bt/rtl8723cs_cg_fw.bin",
|
|
+ .cfg_name = "rtl_bt/rtl8723cs_cg_config" },
|
|
+
|
|
+ /* 8723CS-VF */
|
|
+ { .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_CHIP_TYPE |
|
|
+ IC_MATCH_FL_HCIBUS,
|
|
+ .lmp_subver = RTL_ROM_LMP_8703B,
|
|
+ .chip_type = RTL_CHIP_8723CS_VF,
|
|
+ .hci_bus = HCI_UART,
|
|
+ .config_needed = true,
|
|
+ .has_rom_version = true,
|
|
+ .fw_name = "rtl_bt/rtl8723cs_vf_fw.bin",
|
|
+ .cfg_name = "rtl_bt/rtl8723cs_vf_config" },
|
|
+
|
|
+ /* 8723CS-XX */
|
|
+ { .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_CHIP_TYPE |
|
|
+ IC_MATCH_FL_HCIBUS,
|
|
+ .lmp_subver = RTL_ROM_LMP_8703B,
|
|
+ .chip_type = RTL_CHIP_8723CS_XX,
|
|
+ .hci_bus = HCI_UART,
|
|
+ .config_needed = true,
|
|
+ .has_rom_version = true,
|
|
+ .fw_name = "rtl_bt/rtl8723cs_xx_fw.bin",
|
|
+ .cfg_name = "rtl_bt/rtl8723cs_xx_config" },
|
|
+
|
|
+ /* 8723CS-CG */
|
|
+ { .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_CHIP_TYPE |
|
|
+ IC_MATCH_FL_HCIBUS,
|
|
+ .lmp_subver = RTL_ROM_LMP_8703B,
|
|
+ .chip_type = RTL_CHIP_8723CS_CG,
|
|
+ .hci_bus = HCI_UART,
|
|
+ .config_needed = true,
|
|
+ .has_rom_version = true,
|
|
+ .fw_name = "rtl_bt/rtl8723cs_cg_fw.bin",
|
|
+ .cfg_name = "rtl_bt/rtl8723cs_cg_config" },
|
|
+
|
|
+ /* 8723CS-VF */
|
|
+ { .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_CHIP_TYPE |
|
|
+ IC_MATCH_FL_HCIBUS,
|
|
+ .lmp_subver = RTL_ROM_LMP_8703B,
|
|
+ .chip_type = RTL_CHIP_8723CS_VF,
|
|
+ .hci_bus = HCI_UART,
|
|
+ .config_needed = true,
|
|
+ .has_rom_version = true,
|
|
+ .fw_name = "rtl_bt/rtl8723cs_vf_fw.bin",
|
|
+ .cfg_name = "rtl_bt/rtl8723cs_vf_config" },
|
|
+
|
|
+ /* 8723CS-XX */
|
|
+ { .match_flags = IC_MATCH_FL_LMPSUBV | IC_MATCH_FL_CHIP_TYPE |
|
|
+ IC_MATCH_FL_HCIBUS,
|
|
+ .lmp_subver = RTL_ROM_LMP_8703B,
|
|
+ .chip_type = RTL_CHIP_8723CS_XX,
|
|
+ .hci_bus = HCI_UART,
|
|
+ .config_needed = true,
|
|
+ .has_rom_version = true,
|
|
+ .fw_name = "rtl_bt/rtl8723cs_xx_fw.bin",
|
|
+ .cfg_name = "rtl_bt/rtl8723cs_xx_config" },
|
|
+
|
|
/* 8723D */
|
|
{ IC_INFO(RTL_ROM_LMP_8723B, 0xd, 0x8, HCI_USB),
|
|
.config_needed = true,
|
|
@@ -175,7 +247,8 @@
|
|
};
|
|
|
|
static const struct id_table *btrtl_match_ic(u16 lmp_subver, u16 hci_rev,
|
|
- u8 hci_ver, u8 hci_bus)
|
|
+ u8 hci_ver, u8 hci_bus,
|
|
+ u8 chip_type)
|
|
{
|
|
int i;
|
|
|
|
@@ -192,6 +265,12 @@
|
|
if ((ic_id_table[i].match_flags & IC_MATCH_FL_HCIBUS) &&
|
|
(ic_id_table[i].hci_bus != hci_bus))
|
|
continue;
|
|
+ if ((ic_id_table[i].match_flags & IC_MATCH_FL_CHIP_TYPE) &&
|
|
+ (ic_id_table[i].chip_type != chip_type))
|
|
+ continue;
|
|
+ if ((ic_id_table[i].match_flags & IC_MATCH_FL_CHIP_TYPE) &&
|
|
+ (ic_id_table[i].chip_type != chip_type))
|
|
+ continue;
|
|
|
|
break;
|
|
}
|
|
@@ -274,6 +353,7 @@
|
|
{ RTL_ROM_LMP_8723B, 1 },
|
|
{ RTL_ROM_LMP_8821A, 2 },
|
|
{ RTL_ROM_LMP_8761A, 3 },
|
|
+ { RTL_ROM_LMP_8703B, 7 },
|
|
{ RTL_ROM_LMP_8822B, 8 },
|
|
{ RTL_ROM_LMP_8723B, 9 }, /* 8723D */
|
|
{ RTL_ROM_LMP_8821A, 10 }, /* 8821C */
|
|
@@ -552,6 +632,48 @@
|
|
return ret;
|
|
}
|
|
|
|
+static bool rtl_has_chip_type(u16 lmp_subver)
|
|
+{
|
|
+ switch (lmp_subver) {
|
|
+ case RTL_ROM_LMP_8703B:
|
|
+ return true;
|
|
+ default:
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ return false;
|
|
+}
|
|
+
|
|
+static int rtl_read_chip_type(struct hci_dev *hdev, u8 *type)
|
|
+{
|
|
+ struct rtl_chip_type_evt *chip_type;
|
|
+ struct sk_buff *skb;
|
|
+ const unsigned char cmd_buf[] = {0x00, 0x94, 0xa0, 0x00, 0xb0};
|
|
+
|
|
+ /* Read RTL chip type command */
|
|
+ skb = __hci_cmd_sync(hdev, 0xfc61, 5, cmd_buf, HCI_INIT_TIMEOUT);
|
|
+ if (IS_ERR(skb)) {
|
|
+ rtl_dev_err(hdev, "Read chip type failed (%ld)",
|
|
+ PTR_ERR(skb));
|
|
+ return PTR_ERR(skb);
|
|
+ }
|
|
+
|
|
+ if (skb->len != sizeof(*chip_type)) {
|
|
+ rtl_dev_err(hdev, "RTL chip type event length mismatch");
|
|
+ kfree_skb(skb);
|
|
+ return -EIO;
|
|
+ }
|
|
+
|
|
+ chip_type = (struct rtl_chip_type_evt *)skb->data;
|
|
+ rtl_dev_info(hdev, "chip_type status=%x type=%x",
|
|
+ chip_type->status, chip_type->type);
|
|
+
|
|
+ *type = chip_type->type & 0x0f;
|
|
+
|
|
+ kfree_skb(skb);
|
|
+ return 0;
|
|
+}
|
|
+
|
|
void btrtl_free(struct btrtl_device_info *btrtl_dev)
|
|
{
|
|
kvfree(btrtl_dev->fw_data);
|
|
@@ -568,7 +690,7 @@
|
|
struct hci_rp_read_local_version *resp;
|
|
char cfg_name[40];
|
|
u16 hci_rev, lmp_subver;
|
|
- u8 hci_ver;
|
|
+ u8 hci_ver, chip_type = 0;
|
|
int ret;
|
|
u16 opcode;
|
|
u8 cmd[2];
|
|
@@ -638,8 +760,14 @@
|
|
out_free:
|
|
kfree_skb(skb);
|
|
|
|
+ if (rtl_has_chip_type(lmp_subver)) {
|
|
+ ret = rtl_read_chip_type(hdev, &chip_type);
|
|
+ if (ret)
|
|
+ goto err_free;
|
|
+ }
|
|
+
|
|
btrtl_dev->ic_info = btrtl_match_ic(lmp_subver, hci_rev, hci_ver,
|
|
- hdev->bus);
|
|
+ hdev->bus, chip_type);
|
|
|
|
if (!btrtl_dev->ic_info) {
|
|
rtl_dev_info(hdev, "unknown IC info, lmp subver %04x, hci rev %04x, hci ver %04x",
|
|
@@ -718,6 +846,7 @@
|
|
case RTL_ROM_LMP_8761A:
|
|
case RTL_ROM_LMP_8822B:
|
|
case RTL_ROM_LMP_8852A:
|
|
+ case RTL_ROM_LMP_8703B:
|
|
return btrtl_setup_rtl8723b(hdev, btrtl_dev);
|
|
default:
|
|
rtl_dev_info(hdev, "assuming no firmware upload needed");
|
|
@@ -748,6 +877,18 @@
|
|
rtl_dev_dbg(hdev, "WBS supported not enabled.");
|
|
break;
|
|
}
|
|
+ switch (btrtl_dev->ic_info->lmp_subver) {
|
|
+ case RTL_ROM_LMP_8703B:
|
|
+ /* 8723CS reports two pages for local ext features,
|
|
+ * but it doesn't support any features from page 2 -
|
|
+ * it either responds with garbage or with error status
|
|
+ */
|
|
+ set_bit(HCI_QUIRK_BROKEN_LOCAL_EXT_FTR_MAX_PAGE,
|
|
+ &hdev->quirks);
|
|
+ break;
|
|
+ default:
|
|
+ break;
|
|
+ }
|
|
}
|
|
EXPORT_SYMBOL_GPL(btrtl_set_quirks);
|
|
|
|
@@ -906,6 +1048,12 @@
|
|
MODULE_FIRMWARE("rtl_bt/rtl8723b_config.bin");
|
|
MODULE_FIRMWARE("rtl_bt/rtl8723bs_fw.bin");
|
|
MODULE_FIRMWARE("rtl_bt/rtl8723bs_config.bin");
|
|
+MODULE_FIRMWARE("rtl_bt/rtl8723cs_cg_fw.bin");
|
|
+MODULE_FIRMWARE("rtl_bt/rtl8723cs_cg_config.bin");
|
|
+MODULE_FIRMWARE("rtl_bt/rtl8723cs_vf_fw.bin");
|
|
+MODULE_FIRMWARE("rtl_bt/rtl8723cs_vf_config.bin");
|
|
+MODULE_FIRMWARE("rtl_bt/rtl8723cs_xx_fw.bin");
|
|
+MODULE_FIRMWARE("rtl_bt/rtl8723cs_xx_config.bin");
|
|
MODULE_FIRMWARE("rtl_bt/rtl8723ds_fw.bin");
|
|
MODULE_FIRMWARE("rtl_bt/rtl8723ds_config.bin");
|
|
MODULE_FIRMWARE("rtl_bt/rtl8761a_fw.bin");
|
|
diff -abBur linux-5.14.11/drivers/bluetooth/btrtl.h linux-5.14.11-patched/drivers/bluetooth/btrtl.h
|
|
--- linux-5.14.11/drivers/bluetooth/btrtl.h 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/drivers/bluetooth/btrtl.h 2021-10-11 23:18:52.641741019 +0200
|
|
@@ -14,6 +14,11 @@
|
|
|
|
struct btrtl_device_info;
|
|
|
|
+struct rtl_chip_type_evt {
|
|
+ __u8 status;
|
|
+ __u8 type;
|
|
+} __packed;
|
|
+
|
|
struct rtl_download_cmd {
|
|
__u8 index;
|
|
__u8 data[RTL_FRAG_LEN];
|
|
diff -abBur linux-5.14.11/drivers/bluetooth/hci_h5.c linux-5.14.11-patched/drivers/bluetooth/hci_h5.c
|
|
--- linux-5.14.11/drivers/bluetooth/hci_h5.c 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/drivers/bluetooth/hci_h5.c 2021-10-11 23:17:48.642740988 +0200
|
|
@@ -905,6 +905,8 @@
|
|
err = btrtl_download_firmware(h5->hu->hdev, btrtl_dev);
|
|
/* Give the device some time before the hci-core sends it a reset */
|
|
usleep_range(10000, 20000);
|
|
+ if (err)
|
|
+ goto out_free;
|
|
|
|
btrtl_set_quirks(h5->hu->hdev, btrtl_dev);
|
|
|
|
@@ -1026,6 +1028,8 @@
|
|
.data = (const void *)&rtl_vnd },
|
|
{ .compatible = "realtek,rtl8723ds-bt",
|
|
.data = (const void *)&rtl_vnd },
|
|
+ { .compatible = "realtek,rtl8723cs-bt",
|
|
+ .data = (const void *)&rtl_vnd },
|
|
#endif
|
|
{ },
|
|
};
|
|
diff -abBur linux-5.14.11/include/net/bluetooth/hci.h linux-5.14.11-patched/include/net/bluetooth/hci.h
|
|
--- linux-5.14.11/include/net/bluetooth/hci.h 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/include/net/bluetooth/hci.h 2021-10-11 16:40:14.259729614 +0200
|
|
@@ -246,6 +246,13 @@
|
|
* HCI after resume.
|
|
*/
|
|
HCI_QUIRK_NO_SUSPEND_NOTIFIER,
|
|
+
|
|
+ /* When this quirk is set, max_page for local extended features
|
|
+ * is set to 1, even if controller reports higher number. Some
|
|
+ * controllers (e.g. RTL8723CS) report more pages, but they
|
|
+ * don't actually support features declared there.
|
|
+ */
|
|
+ HCI_QUIRK_BROKEN_LOCAL_EXT_FTR_MAX_PAGE,
|
|
};
|
|
|
|
/* HCI device flags */
|
|
diff -abBur linux-5.14.11/net/bluetooth/hci_event.c linux-5.14.11-patched/net/bluetooth/hci_event.c
|
|
--- linux-5.14.11/net/bluetooth/hci_event.c 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/net/bluetooth/hci_event.c 2021-10-11 16:29:31.196729307 +0200
|
|
@@ -725,7 +725,9 @@
|
|
if (rp->status)
|
|
return;
|
|
|
|
- if (hdev->max_page < rp->max_page)
|
|
+ if (!test_bit(HCI_QUIRK_BROKEN_LOCAL_EXT_FTR_MAX_PAGE,
|
|
+ &hdev->quirks) &&
|
|
+ hdev->max_page < rp->max_page)
|
|
hdev->max_page = rp->max_page;
|
|
|
|
if (rp->page < HCI_MAX_PAGES)
|
|
diff -abBur linux-5.14.11/scripts/dtc/include-prefixes/arm64/allwinner/sun50i-a64-pinebook.dts linux-5.14.11-patched/scripts/dtc/include-prefixes/arm64/allwinner/sun50i-a64-pinebook.dts
|
|
--- linux-5.14.11/scripts/dtc/include-prefixes/arm64/allwinner/sun50i-a64-pinebook.dts 2021-10-09 15:02:42.000000000 +0200
|
|
+++ linux-5.14.11-patched/scripts/dtc/include-prefixes/arm64/allwinner/sun50i-a64-pinebook.dts 2021-10-11 23:16:51.530740961 +0200
|
|
@@ -405,6 +405,32 @@
|
|
status = "okay";
|
|
};
|
|
|
|
+
|
|
+&uart1 {
|
|
+ pinctrl-names = "default";
|
|
+ pinctrl-0 = <&uart1_pins>, <&uart1_rts_cts_pins>;
|
|
+ status = "okay";
|
|
+
|
|
+ bluetooth {
|
|
+ compatible = "realtek,rtl8723cs-bt";
|
|
+ device-wake-gpios = <&r_pio 0 5 GPIO_ACTIVE_LOW>; /* PL5 */
|
|
+ host-wake-gpios = <&r_pio 0 6 GPIO_ACTIVE_HIGH>; /* PL6 */
|
|
+ };
|
|
+};
|
|
+
|
|
+&uart1 {
|
|
+ pinctrl-names = "default";
|
|
+ pinctrl-0 = <&uart1_pins>, <&uart1_rts_cts_pins>;
|
|
+ uart-has-rtscts;
|
|
+ status = "okay";
|
|
+
|
|
+ bluetooth {
|
|
+ compatible = "realtek,rtl8723cs-bt";
|
|
+ device-wake-gpios = <&r_pio 0 5 GPIO_ACTIVE_LOW>; /* PL5 */
|
|
+ host-wake-gpios = <&r_pio 0 6 GPIO_ACTIVE_HIGH>; /* PL6 */
|
|
+ };
|
|
+};
|
|
+
|
|
&usb_otg {
|
|
dr_mode = "host";
|
|
};
|