/* SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause */ /* Copyright(c) 2019-2020 Realtek Corporation */ #ifndef __RTW89_COEX_H__ #define __RTW89_COEX_H__ #include "core.h" #define BTC_H2C_MAXLEN 2020 #define BTC_TLV_SLOT_ID_LEN_V7 1 #define BTC_SLOT_REQ_TH 2 enum btc_mode { BTC_MODE_NORMAL, BTC_MODE_WL, BTC_MODE_BT, BTC_MODE_WLOFF, BTC_MODE_MAX }; enum btc_wl_rfk_type { BTC_WRFKT_IQK = 0, BTC_WRFKT_LCK = 1, BTC_WRFKT_DPK = 2, BTC_WRFKT_TXGAPK = 3, BTC_WRFKT_DACK = 4, BTC_WRFKT_RXDCK = 5, BTC_WRFKT_TSSI = 6, BTC_WRFKT_CHLK = 7, }; #define NM_EXEC false #define FC_EXEC true #define RTW89_COEX_ACT1_WORK_PERIOD round_jiffies_relative(HZ * 4) #define RTW89_COEX_BT_DEVINFO_WORK_PERIOD round_jiffies_relative(HZ * 16) #define RTW89_COEX_RFK_CHK_WORK_PERIOD msecs_to_jiffies(300) #define BTC_RFK_PATH_MAP GENMASK(3, 0) #define BTC_RFK_PHY_MAP GENMASK(5, 4) #define BTC_RFK_BAND_MAP GENMASK(7, 6) enum btc_wl_rfk_state { BTC_WRFK_STOP = 0, BTC_WRFK_START = 1, BTC_WRFK_ONESHOT_START = 2, BTC_WRFK_ONESHOT_STOP = 3, }; enum btc_pri { BTC_PRI_MASK_RX_RESP = 0, BTC_PRI_MASK_TX_RESP, BTC_PRI_MASK_BEACON, BTC_PRI_MASK_RX_CCK, BTC_PRI_MASK_TX_MNGQ, BTC_PRI_MASK_MAX, }; enum btc_bt_trs { BTC_BT_SS_GROUP = 0x0, BTC_BT_TX_GROUP = 0x2, BTC_BT_RX_GROUP = 0x3, BTC_BT_MAX_GROUP, }; enum btc_rssi_st { BTC_RSSI_ST_LOW = 0x0, BTC_RSSI_ST_HIGH, BTC_RSSI_ST_STAY_LOW, BTC_RSSI_ST_STAY_HIGH, BTC_RSSI_ST_MAX }; enum btc_fddt_en { BTC_FDDT_DISABLE, BTC_FDDT_ENABLE, }; #define BTC_RSSI_HIGH(_rssi_) \ ({typeof(_rssi_) __rssi = (_rssi_); \ ((__rssi == BTC_RSSI_ST_HIGH || \ __rssi == BTC_RSSI_ST_STAY_HIGH) ? 1 : 0); }) #define BTC_RSSI_LOW(_rssi_) \ ({typeof(_rssi_) __rssi = (_rssi_); \ ((__rssi == BTC_RSSI_ST_LOW || \ __rssi == BTC_RSSI_ST_STAY_LOW) ? 1 : 0); }) #define BTC_RSSI_CHANGE(_rssi_) \ ({typeof(_rssi_) __rssi = (_rssi_); \ ((__rssi == BTC_RSSI_ST_LOW || \ __rssi == BTC_RSSI_ST_HIGH) ? 1 : 0); }) enum btc_ant { BTC_ANT_SHARED = 0, BTC_ANT_DEDICATED, BTC_ANTTYPE_MAX }; enum btc_bt_btg { BTC_BT_ALONE = 0, BTC_BT_BTG }; enum btc_switch { BTC_SWITCH_INTERNAL = 0, BTC_SWITCH_EXTERNAL }; enum btc_pkt_type { PACKET_DHCP, PACKET_ARP, PACKET_EAPOL, PACKET_EAPOL_END, PACKET_ICMP, PACKET_MAX }; enum btc_bt_mailbox_id { BTC_BTINFO_REPLY = 0x23, BTC_BTINFO_AUTO = 0x27 }; enum btc_role_state { BTC_ROLE_START, BTC_ROLE_STOP, BTC_ROLE_CHG_TYPE, BTC_ROLE_MSTS_STA_CONN_START, BTC_ROLE_MSTS_STA_CONN_END, BTC_ROLE_MSTS_STA_DIS_CONN, BTC_ROLE_MSTS_AP_START, BTC_ROLE_MSTS_AP_STOP, BTC_ROLE_STATE_UNKNOWN }; enum btc_rfctrl { BTC_RFCTRL_WL_OFF, BTC_RFCTRL_WL_ON, BTC_RFCTRL_LPS_WL_ON, BTC_RFCTRL_FW_CTRL, BTC_RFCTRL_MAX }; enum btc_lps_state { BTC_LPS_OFF = 0, BTC_LPS_RF_OFF = 1, BTC_LPS_RF_ON = 2 }; #define R_BTC_BB_BTG_RX 0x980 #define R_BTC_BB_PRE_AGC_S1 0x476C #define R_BTC_BB_PRE_AGC_S0 0x4688 #define B_BTC_BB_GNT_MUX GENMASK(20, 17) #define B_BTC_BB_PRE_AGC_MASK GENMASK(31, 24) #define B_BTC_BB_PRE_AGC_VAL BIT(31) #define BTC_REG_NOTFOUND 0xff #define R_BTC_ZB_COEX_TBL_0 0xE328 #define R_BTC_ZB_COEX_TBL_1 0xE32c #define R_BTC_ZB_BREAK_TBL 0xE350 enum btc_ant_div_pos { BTC_ANT_DIV_MAIN = 0, BTC_ANT_DIV_AUX = 1, }; enum btc_get_reg_status { BTC_CSTATUS_TXDIV_POS = 0, BTC_CSTATUS_RXDIV_POS = 1, BTC_CSTATUS_BB_GNT_MUX = 2, BTC_CSTATUS_BB_GNT_MUX_MON = 3, BTC_CSTATUS_BB_PRE_AGC = 4, BTC_CSTATUS_BB_PRE_AGC_MON = 5, }; enum btc_preagc_type { BTC_PREAGC_DISABLE, BTC_PREAGC_ENABLE, BTC_PREAGC_BB_FWCTRL, BTC_PREAGC_NOTFOUND, }; enum btc_btgctrl_type { BTC_BTGCTRL_DISABLE, BTC_BTGCTRL_ENABLE, BTC_BTGCTRL_BB_GNT_FWCTRL, BTC_BTGCTRL_BB_GNT_NOTFOUND, }; enum btc_wa_type { BTC_WA_5G_HI_CH_RX = BIT(0), BTC_WA_NULL_AP = BIT(1), BTC_WA_HFP_ZB = BIT(2), /* HFP PTA req bit4 define issue */ BTC_WA_HFP_LAG = BIT(3), /* 52BT WL break BT Rx lag issue */ BTC_WA_INIT_SCAN = BIT(4) /* 52A/C/D init scan move to wl slot WA */ }; enum btc_3cx_type { BTC_3CX_NONE = 0, BTC_3CX_BT2 = BIT(0), BTC_3CX_ZB = BIT(1), BTC_3CX_LTE = BIT(2), BTC_3CX_MAX, }; enum btc_chip_feature { BTC_FEAT_PTA_ONOFF_CTRL = BIT(0), /* on/off ctrl by HW (not 0x73[2]) */ BTC_FEAT_NONBTG_GWL_THRU = BIT(1), /* non-BTG GNT_WL!=0 if GNT_BT = 1 */ BTC_FEAT_WLAN_ACT_MUX = BIT(2), /* separate wlan_act/gnt mux */ BTC_FEAT_NEW_BBAPI_FLOW = BIT(3), /* new btg_ctrl/pre_agc_ctrl */ BTC_FEAT_MLO_SUPPORT = BIT(4), BTC_FEAT_H2C_MACRO = BIT(5), }; enum btc_wl_mode { BTC_WL_MODE_11B = 0, BTC_WL_MODE_11A = 1, BTC_WL_MODE_11G = 2, BTC_WL_MODE_HT = 3, BTC_WL_MODE_VHT = 4, BTC_WL_MODE_HE = 5, BTC_WL_MODE_NUM, }; enum btc_wl_gpio_debug { BTC_DBG_GNT_BT = 0, BTC_DBG_GNT_WL = 1, BTC_DBG_BCN_EARLY = 2, BTC_DBG_WL_NULL0 = 3, BTC_DBG_WL_NULL1 = 4, BTC_DBG_WL_RXISR = 5, BTC_DBG_TDMA_ENTRY = 6, BTC_DBG_A2DP_EMPTY = 7, BTC_DBG_BT_RETRY = 8, BTC_DBG_BT_RELINK = 9, BTC_DBG_SLOT_WL = 10, BTC_DBG_SLOT_BT = 11, BTC_DBG_WL_ERR = 12, BTC_DBG_WL_OK = 13, BTC_DBG_SLOT_B2W = 14, BTC_DBG_SLOT_W1 = 15, BTC_DBG_SLOT_W2 = 16, BTC_DBG_SLOT_W2B = 17, BTC_DBG_SLOT_B1 = 18, BTC_DBG_SLOT_B2 = 19, BTC_DBG_SLOT_B3 = 20, BTC_DBG_SLOT_B4 = 21, BTC_DBG_SLOT_LK = 22, BTC_DBG_SLOT_E2G = 23, BTC_DBG_SLOT_E5G = 24, BTC_DBG_SLOT_EBT = 25, BTC_DBG_SLOT_WLK = 26, BTC_DBG_SLOT_B1FDD = 27, BTC_DBG_BT_CHANGE = 28, BTC_DBG_WL_CCA = 29, BTC_DBG_BT_LEAUDIO = 30, BTC_DBG_USER_DEF = 31, }; void rtw89_btc_ntfy_poweron(struct rtw89_dev *rtwdev); void rtw89_btc_ntfy_poweroff(struct rtw89_dev *rtwdev); void rtw89_btc_ntfy_init(struct rtw89_dev *rtwdev, u8 mode); void rtw89_btc_ntfy_scan_start(struct rtw89_dev *rtwdev, u8 phy_idx, u8 band); void rtw89_btc_ntfy_scan_finish(struct rtw89_dev *rtwdev, u8 phy_idx); void rtw89_btc_ntfy_switch_band(struct rtw89_dev *rtwdev, u8 phy_idx, u8 band); void rtw89_btc_ntfy_specific_packet(struct rtw89_dev *rtwdev, enum btc_pkt_type pkt_type); void rtw89_btc_ntfy_eapol_packet_work(struct work_struct *work); void rtw89_btc_ntfy_arp_packet_work(struct work_struct *work); void rtw89_btc_ntfy_dhcp_packet_work(struct work_struct *work); void rtw89_btc_ntfy_icmp_packet_work(struct work_struct *work); void rtw89_btc_ntfy_role_info(struct rtw89_dev *rtwdev, struct rtw89_vif *rtwvif, struct rtw89_sta *rtwsta, enum btc_role_state state); void rtw89_btc_ntfy_radio_state(struct rtw89_dev *rtwdev, enum btc_rfctrl rf_state); void rtw89_btc_ntfy_wl_rfk(struct rtw89_dev *rtwdev, u8 phy_map, enum btc_wl_rfk_type type, enum btc_wl_rfk_state state); void rtw89_btc_ntfy_wl_sta(struct rtw89_dev *rtwdev); void rtw89_btc_c2h_handle(struct rtw89_dev *rtwdev, struct sk_buff *skb, u32 len, u8 class, u8 func); void rtw89_btc_dump_info(struct rtw89_dev *rtwdev, struct seq_file *m); void rtw89_coex_act1_work(struct work_struct *work); void rtw89_coex_bt_devinfo_work(struct work_struct *work); void rtw89_coex_rfk_chk_work(struct work_struct *work); void rtw89_coex_power_on(struct rtw89_dev *rtwdev); void rtw89_btc_set_policy(struct rtw89_dev *rtwdev, u16 policy_type); void rtw89_btc_set_policy_v1(struct rtw89_dev *rtwdev, u16 policy_type); void rtw89_coex_recognize_ver(struct rtw89_dev *rtwdev); static inline u8 rtw89_btc_phymap(struct rtw89_dev *rtwdev, enum rtw89_phy_idx phy_idx, enum rtw89_rf_path_bit paths, enum rtw89_chanctx_idx chanctx_idx) { const struct rtw89_chan *chan = rtw89_chan_get(rtwdev, chanctx_idx); u8 phy_map; phy_map = FIELD_PREP(BTC_RFK_PATH_MAP, paths) | FIELD_PREP(BTC_RFK_PHY_MAP, BIT(phy_idx)) | FIELD_PREP(BTC_RFK_BAND_MAP, chan->band_type); return phy_map; } static inline u8 rtw89_btc_path_phymap(struct rtw89_dev *rtwdev, enum rtw89_phy_idx phy_idx, enum rtw89_rf_path path, enum rtw89_chanctx_idx chanctx_idx) { return rtw89_btc_phymap(rtwdev, phy_idx, BIT(path), chanctx_idx); } /* return bt req len in TU */ static inline u16 rtw89_coex_query_bt_req_len(struct rtw89_dev *rtwdev, enum rtw89_phy_idx phy_idx) { struct rtw89_btc *btc = &rtwdev->btc; return btc->bt_req_len; } static inline u32 rtw89_get_antpath_type(u8 phy_map, u8 type) { return ((phy_map << 8) + type); } static inline void _slot_set_le(struct rtw89_btc *btc, u8 sid, __le16 dura, __le32 tbl, __le16 type) { if (btc->ver->fcxslots == 1) { btc->dm.slot.v1[sid].dur = dura; btc->dm.slot.v1[sid].cxtbl = tbl; btc->dm.slot.v1[sid].cxtype = type; } else if (btc->ver->fcxslots == 7) { btc->dm.slot.v7[sid].dur = dura; btc->dm.slot.v7[sid].cxtype = type; btc->dm.slot.v7[sid].cxtbl = tbl; } } static inline void _slot_set(struct rtw89_btc *btc, u8 sid, u16 dura, u32 tbl, u16 type) { _slot_set_le(btc, sid, cpu_to_le16(dura), cpu_to_le32(tbl), cpu_to_le16(type)); } static inline void _slot_set_dur(struct rtw89_btc *btc, u8 sid, u16 dura) { if (btc->ver->fcxslots == 1) btc->dm.slot.v1[sid].dur = cpu_to_le16(dura); else if (btc->ver->fcxslots == 7) btc->dm.slot.v7[sid].dur = cpu_to_le16(dura); } static inline void _slot_set_type(struct rtw89_btc *btc, u8 sid, u16 type) { if (btc->ver->fcxslots == 1) btc->dm.slot.v1[sid].cxtype = cpu_to_le16(type); else if (btc->ver->fcxslots == 7) btc->dm.slot.v7[sid].cxtype = cpu_to_le16(type); } static inline void _slot_set_tbl(struct rtw89_btc *btc, u8 sid, u32 tbl) { if (btc->ver->fcxslots == 1) btc->dm.slot.v1[sid].cxtbl = cpu_to_le32(tbl); else if (btc->ver->fcxslots == 7) btc->dm.slot.v7[sid].cxtbl = cpu_to_le32(tbl); } #endif