mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-21 03:55:06 +00:00
195 lines
7.9 KiB
Diff
195 lines
7.9 KiB
Diff
|
From c7706b1173c77185a2ef40c7d1811021566563f3 Mon Sep 17 00:00:00 2001
|
||
|
From: Bitterblue Smith <rtl8821cerfe2@gmail.com>
|
||
|
Date: Wed, 23 Oct 2024 17:10:32 +0300
|
||
|
Subject: [PATCH] wifi: rtw88: Enable data rate fallback for older chips
|
||
|
|
||
|
RTL8811AU fails to perform the 4-way handshake when the AP is too far
|
||
|
because it transmits the EAPOL frames at MCS9 and when that doesn't
|
||
|
work it retries 48 times with the same rate, to no avail.
|
||
|
|
||
|
Retrying 48 times with the same rate seems pointless. Set the
|
||
|
appropriate field in the TX descriptor to allow it to use lower rates
|
||
|
when retrying.
|
||
|
|
||
|
Set it for RTL8723D and RTL8703B because they interpret this field the
|
||
|
same way as RTL8811A.
|
||
|
|
||
|
The newer RTL8822C, RTL8822B, RTL8821C seem to interpret this field in
|
||
|
the TX descriptor differently, so leave it alone for those chips.
|
||
|
|
||
|
Tested with RTL8811AU and RTL8723DU.
|
||
|
|
||
|
Signed-off-by: Bitterblue Smith <rtl8821cerfe2@gmail.com>
|
||
|
Signed-off-by: Ping-Ke Shih <pkshih@realtek.com>
|
||
|
Link: https://patch.msgid.link/2b3e3e6f-541b-4a3b-8ca3-65b267e6a95a@gmail.com
|
||
|
---
|
||
|
drivers/net/wireless/realtek/rtw88/fw.c | 2 +-
|
||
|
drivers/net/wireless/realtek/rtw88/main.h | 1 +
|
||
|
drivers/net/wireless/realtek/rtw88/pci.c | 2 +-
|
||
|
drivers/net/wireless/realtek/rtw88/rtw8703b.c | 1 +
|
||
|
drivers/net/wireless/realtek/rtw88/rtw8723d.c | 1 +
|
||
|
drivers/net/wireless/realtek/rtw88/rtw8821c.c | 1 +
|
||
|
drivers/net/wireless/realtek/rtw88/rtw8822b.c | 1 +
|
||
|
drivers/net/wireless/realtek/rtw88/rtw8822c.c | 1 +
|
||
|
drivers/net/wireless/realtek/rtw88/sdio.c | 2 +-
|
||
|
drivers/net/wireless/realtek/rtw88/tx.c | 6 +++++-
|
||
|
drivers/net/wireless/realtek/rtw88/tx.h | 4 +++-
|
||
|
drivers/net/wireless/realtek/rtw88/usb.c | 4 ++--
|
||
|
12 files changed, 19 insertions(+), 7 deletions(-)
|
||
|
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/fw.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/fw.c
|
||
|
@@ -1290,7 +1290,7 @@ static void rtw_fill_rsvd_page_desc(stru
|
||
|
rtw_tx_rsvd_page_pkt_info_update(rtwdev, &pkt_info, skb, type);
|
||
|
pkt_desc = skb_push(skb, chip->tx_pkt_desc_sz);
|
||
|
memset(pkt_desc, 0, chip->tx_pkt_desc_sz);
|
||
|
- rtw_tx_fill_tx_desc(&pkt_info, skb);
|
||
|
+ rtw_tx_fill_tx_desc(rtwdev, &pkt_info, skb);
|
||
|
}
|
||
|
|
||
|
static inline u8 rtw_len_to_page(unsigned int len, u8 page_size)
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/main.h
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/main.h
|
||
|
@@ -1204,6 +1204,7 @@ struct rtw_chip_info {
|
||
|
u8 usb_tx_agg_desc_num;
|
||
|
bool hw_feature_report;
|
||
|
u8 c2h_ra_report_size;
|
||
|
+ bool old_datarate_fb_limit;
|
||
|
|
||
|
u8 default_1ss_tx_path;
|
||
|
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/pci.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/pci.c
|
||
|
@@ -824,7 +824,7 @@ static int rtw_pci_tx_write_data(struct
|
||
|
pkt_desc = skb_push(skb, chip->tx_pkt_desc_sz);
|
||
|
memset(pkt_desc, 0, tx_pkt_desc_sz);
|
||
|
pkt_info->qsel = rtw_pci_get_tx_qsel(skb, queue);
|
||
|
- rtw_tx_fill_tx_desc(pkt_info, skb);
|
||
|
+ rtw_tx_fill_tx_desc(rtwdev, pkt_info, skb);
|
||
|
dma = dma_map_single(&rtwpci->pdev->dev, skb->data, skb->len,
|
||
|
DMA_TO_DEVICE);
|
||
|
if (dma_mapping_error(&rtwpci->pdev->dev, dma))
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/rtw8703b.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/rtw8703b.c
|
||
|
@@ -1964,6 +1964,7 @@ const struct rtw_chip_info rtw8703b_hw_s
|
||
|
.usb_tx_agg_desc_num = 1, /* Not sure if this chip has USB interface */
|
||
|
.hw_feature_report = true,
|
||
|
.c2h_ra_report_size = 7,
|
||
|
+ .old_datarate_fb_limit = true,
|
||
|
|
||
|
.path_div_supported = false,
|
||
|
.ht_supported = true,
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/rtw8723d.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/rtw8723d.c
|
||
|
@@ -2135,6 +2135,7 @@ const struct rtw_chip_info rtw8723d_hw_s
|
||
|
.usb_tx_agg_desc_num = 1,
|
||
|
.hw_feature_report = true,
|
||
|
.c2h_ra_report_size = 7,
|
||
|
+ .old_datarate_fb_limit = true,
|
||
|
.ht_supported = true,
|
||
|
.vht_supported = false,
|
||
|
.lps_deep_mode_supported = 0,
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/rtw8821c.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/rtw8821c.c
|
||
|
@@ -1972,6 +1972,7 @@ const struct rtw_chip_info rtw8821c_hw_s
|
||
|
.usb_tx_agg_desc_num = 3,
|
||
|
.hw_feature_report = true,
|
||
|
.c2h_ra_report_size = 7,
|
||
|
+ .old_datarate_fb_limit = false,
|
||
|
.ht_supported = true,
|
||
|
.vht_supported = true,
|
||
|
.lps_deep_mode_supported = BIT(LPS_DEEP_MODE_LCLK),
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/rtw8822b.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/rtw8822b.c
|
||
|
@@ -2513,6 +2513,7 @@ const struct rtw_chip_info rtw8822b_hw_s
|
||
|
.usb_tx_agg_desc_num = 3,
|
||
|
.hw_feature_report = true,
|
||
|
.c2h_ra_report_size = 7,
|
||
|
+ .old_datarate_fb_limit = false,
|
||
|
.ht_supported = true,
|
||
|
.vht_supported = true,
|
||
|
.lps_deep_mode_supported = BIT(LPS_DEEP_MODE_LCLK),
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/rtw8822c.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/rtw8822c.c
|
||
|
@@ -5333,6 +5333,7 @@ const struct rtw_chip_info rtw8822c_hw_s
|
||
|
.usb_tx_agg_desc_num = 3,
|
||
|
.hw_feature_report = true,
|
||
|
.c2h_ra_report_size = 7,
|
||
|
+ .old_datarate_fb_limit = false,
|
||
|
.default_1ss_tx_path = BB_PATH_A,
|
||
|
.path_div_supported = true,
|
||
|
.ht_supported = true,
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/sdio.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/sdio.c
|
||
|
@@ -864,7 +864,7 @@ static void rtw_sdio_tx_skb_prepare(stru
|
||
|
|
||
|
pkt_info->qsel = rtw_sdio_get_tx_qsel(rtwdev, skb, queue);
|
||
|
|
||
|
- rtw_tx_fill_tx_desc(pkt_info, skb);
|
||
|
+ rtw_tx_fill_tx_desc(rtwdev, pkt_info, skb);
|
||
|
rtw_tx_fill_txdesc_checksum(rtwdev, pkt_info, pkt_desc);
|
||
|
}
|
||
|
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/tx.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/tx.c
|
||
|
@@ -32,7 +32,8 @@ void rtw_tx_stats(struct rtw_dev *rtwdev
|
||
|
}
|
||
|
}
|
||
|
|
||
|
-void rtw_tx_fill_tx_desc(struct rtw_tx_pkt_info *pkt_info, struct sk_buff *skb)
|
||
|
+void rtw_tx_fill_tx_desc(struct rtw_dev *rtwdev,
|
||
|
+ struct rtw_tx_pkt_info *pkt_info, struct sk_buff *skb)
|
||
|
{
|
||
|
struct rtw_tx_desc *tx_desc = (struct rtw_tx_desc *)skb->data;
|
||
|
bool more_data = false;
|
||
|
@@ -67,6 +68,9 @@ void rtw_tx_fill_tx_desc(struct rtw_tx_p
|
||
|
|
||
|
tx_desc->w4 = le32_encode_bits(pkt_info->rate, RTW_TX_DESC_W4_DATARATE);
|
||
|
|
||
|
+ if (rtwdev->chip->old_datarate_fb_limit)
|
||
|
+ tx_desc->w4 |= le32_encode_bits(0x1f, RTW_TX_DESC_W4_DATARATE_FB_LIMIT);
|
||
|
+
|
||
|
tx_desc->w5 = le32_encode_bits(pkt_info->short_gi, RTW_TX_DESC_W5_DATA_SHORT) |
|
||
|
le32_encode_bits(pkt_info->bw, RTW_TX_DESC_W5_DATA_BW) |
|
||
|
le32_encode_bits(pkt_info->ldpc, RTW_TX_DESC_W5_DATA_LDPC) |
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/tx.h
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/tx.h
|
||
|
@@ -44,6 +44,7 @@ struct rtw_tx_desc {
|
||
|
#define RTW_TX_DESC_W3_NAVUSEHDR BIT(15)
|
||
|
#define RTW_TX_DESC_W3_MAX_AGG_NUM GENMASK(21, 17)
|
||
|
#define RTW_TX_DESC_W4_DATARATE GENMASK(6, 0)
|
||
|
+#define RTW_TX_DESC_W4_DATARATE_FB_LIMIT GENMASK(12, 8)
|
||
|
#define RTW_TX_DESC_W4_RTSRATE GENMASK(28, 24)
|
||
|
#define RTW_TX_DESC_W5_DATA_SHORT BIT(4)
|
||
|
#define RTW_TX_DESC_W5_DATA_BW GENMASK(6, 5)
|
||
|
@@ -94,7 +95,8 @@ void rtw_tx_pkt_info_update(struct rtw_d
|
||
|
struct rtw_tx_pkt_info *pkt_info,
|
||
|
struct ieee80211_sta *sta,
|
||
|
struct sk_buff *skb);
|
||
|
-void rtw_tx_fill_tx_desc(struct rtw_tx_pkt_info *pkt_info, struct sk_buff *skb);
|
||
|
+void rtw_tx_fill_tx_desc(struct rtw_dev *rtwdev,
|
||
|
+ struct rtw_tx_pkt_info *pkt_info, struct sk_buff *skb);
|
||
|
void rtw_tx_report_enqueue(struct rtw_dev *rtwdev, struct sk_buff *skb, u8 sn);
|
||
|
void rtw_tx_report_handle(struct rtw_dev *rtwdev, struct sk_buff *skb, int src);
|
||
|
void rtw_tx_rsvd_page_pkt_info_update(struct rtw_dev *rtwdev,
|
||
|
--- a/drivers/net/wireless/realtek/rtw88/usb.c
|
||
|
+++ b/drivers/net/wireless/realtek/rtw88/usb.c
|
||
|
@@ -457,7 +457,7 @@ static int rtw_usb_write_data(struct rtw
|
||
|
skb_put_data(skb, buf, size);
|
||
|
skb_push(skb, chip->tx_pkt_desc_sz);
|
||
|
memset(skb->data, 0, chip->tx_pkt_desc_sz);
|
||
|
- rtw_tx_fill_tx_desc(pkt_info, skb);
|
||
|
+ rtw_tx_fill_tx_desc(rtwdev, pkt_info, skb);
|
||
|
rtw_tx_fill_txdesc_checksum(rtwdev, pkt_info, skb->data);
|
||
|
|
||
|
ret = rtw_usb_write_port(rtwdev, qsel, skb,
|
||
|
@@ -524,7 +524,7 @@ static int rtw_usb_tx_write(struct rtw_d
|
||
|
pkt_desc = skb_push(skb, chip->tx_pkt_desc_sz);
|
||
|
memset(pkt_desc, 0, chip->tx_pkt_desc_sz);
|
||
|
ep = qsel_to_ep(rtwusb, pkt_info->qsel);
|
||
|
- rtw_tx_fill_tx_desc(pkt_info, skb);
|
||
|
+ rtw_tx_fill_tx_desc(rtwdev, pkt_info, skb);
|
||
|
rtw_tx_fill_txdesc_checksum(rtwdev, pkt_info, skb->data);
|
||
|
tx_data = rtw_usb_get_tx_data(skb);
|
||
|
tx_data->sn = pkt_info->sn;
|