root/drivers/net/wireless/realtek/rtw88/rtw8821a.c
// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
/* Copyright(c) 2024  Realtek Corporation
 */

#include "main.h"
#include "coex.h"
#include "phy.h"
#include "reg.h"
#include "rtw88xxa.h"
#include "rtw8821a.h"
#include "rtw8821a_table.h"
#include "tx.h"

static void rtw8821a_power_off(struct rtw_dev *rtwdev)
{
        rtw88xxa_power_off(rtwdev, enter_lps_flow_8821a);
}

static s8 rtw8821a_cck_rx_pwr(u8 lna_idx, u8 vga_idx)
{
        static const s8 lna_gain_table[] = {15, -1, -17, 0, -30, -38};
        s8 rx_pwr_all = 0;
        s8 lna_gain;

        switch (lna_idx) {
        case 5:
        case 4:
        case 2:
        case 1:
        case 0:
                lna_gain = lna_gain_table[lna_idx];
                rx_pwr_all = lna_gain - 2 * vga_idx;
                break;
        default:
                break;
        }

        return rx_pwr_all;
}

static void rtw8821a_query_phy_status(struct rtw_dev *rtwdev, u8 *phy_status,
                                      struct rtw_rx_pkt_stat *pkt_stat)
{
        rtw88xxa_query_phy_status(rtwdev, phy_status, pkt_stat,
                                  rtw8821a_cck_rx_pwr);
}

static void rtw8821a_cfg_ldo25(struct rtw_dev *rtwdev, bool enable)
{
}

#define CAL_NUM_8821A 3
#define MACBB_REG_NUM_8821A 8
#define AFE_REG_NUM_8821A 4
#define RF_REG_NUM_8821A 3

static void rtw8821a_iqk_backup_rf(struct rtw_dev *rtwdev, u32 *rfa_backup,
                                   const u32 *backup_rf_reg, u32 rf_num)
{
        u32 i;

        /* [31] = 0 --> Page C */
        rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);

        /* Save RF Parameters */
        for (i = 0; i < rf_num; i++)
                rfa_backup[i] = rtw_read_rf(rtwdev, RF_PATH_A,
                                            backup_rf_reg[i], MASKDWORD);
}

static void rtw8821a_iqk_restore_rf(struct rtw_dev *rtwdev,
                                    const u32 *backup_rf_reg,
                                    u32 *RF_backup, u32 rf_reg_num)
{
        u32 i;

        /* [31] = 0 --> Page C */
        rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);

        for (i = 0; i < rf_reg_num; i++)
                rtw_write_rf(rtwdev, RF_PATH_A, backup_rf_reg[i],
                             RFREG_MASK, RF_backup[i]);
}

static void rtw8821a_iqk_restore_afe(struct rtw_dev *rtwdev, u32 *afe_backup,
                                     const u32 *backup_afe_reg, u32 afe_num)
{
        u32 i;

        /* [31] = 0 --> Page C */
        rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);

        /* Reload AFE Parameters */
        for (i = 0; i < afe_num; i++)
                rtw_write32(rtwdev, backup_afe_reg[i], afe_backup[i]);

        /* [31] = 1 --> Page C1 */
        rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x1);

        rtw_write32(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE, 0x0);
        rtw_write32(rtwdev, REG_OFDM0_A_TX_AFE, 0x0);
        rtw_write32(rtwdev, REG_OFDM0_XB_TX_IQ_IMBALANCE, 0x0);
        rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x3c000000);
        rtw_write32(rtwdev, REG_LSSI_WRITE_A, 0x00000080);
        rtw_write32(rtwdev, REG_TXAGCIDX, 0x00000000);
        rtw_write32(rtwdev, REG_IQK_DPD_CFG, 0x20040000);
        rtw_write32(rtwdev, REG_CFG_PMPD, 0x20000000);
        rtw_write32(rtwdev, REG_RFECTL_A, 0x0);
}

static void rtw8821a_iqk_rx_fill(struct rtw_dev *rtwdev,
                                 unsigned int rx_x, unsigned int rx_y)
{
        /* [31] = 0 --> Page C */
        rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);

        rtw_write32_mask(rtwdev, REG_RX_IQC_AB_A,
                         0x000003ff, rx_x >> 1);
        rtw_write32_mask(rtwdev, REG_RX_IQC_AB_A,
                         0x03ff0000, (rx_y >> 1) & 0x3ff);
}

static void rtw8821a_iqk_tx_fill(struct rtw_dev *rtwdev,
                                 unsigned int tx_x, unsigned int tx_y)
{
        /* [31] = 1 --> Page C1 */
        rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x1);

        rtw_write32(rtwdev, REG_LSSI_WRITE_A, 0x00000080);
        rtw_write32(rtwdev, REG_IQK_DPD_CFG, 0x20040000);
        rtw_write32(rtwdev, REG_CFG_PMPD, 0x20000000);
        rtw_write32_mask(rtwdev, REG_IQC_Y, 0x000007ff, tx_y);
        rtw_write32_mask(rtwdev, REG_IQC_X, 0x000007ff, tx_x);
}

static void rtw8821a_iqk_tx_vdf_true(struct rtw_dev *rtwdev, u32 cal,
                                     bool *tx0iqkok,
                                     int tx_x0[CAL_NUM_8821A],
                                     int tx_y0[CAL_NUM_8821A])
{
        u32 cal_retry, delay_count, iqk_ready, tx_fail;
        int tx_dt[3], vdf_y[3], vdf_x[3];
        int k;

        for (k = 0; k < 3; k++) {
                switch (k) {
                case 0:
                        /* TX_Tone_idx[9:0], TxK_Mask[29] TX_Tone = 16 */
                        rtw_write32(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE,
                                    0x18008c38);
                        /* RX_Tone_idx[9:0], RxK_Mask[29] */
                        rtw_write32(rtwdev, REG_OFDM0_A_TX_AFE, 0x38008c38);
                        rtw_write32_mask(rtwdev, REG_INTPO_SETA, BIT(31), 0x0);
                        break;
                case 1:
                        rtw_write32_mask(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE,
                                         BIT(28), 0x0);
                        rtw_write32_mask(rtwdev, REG_OFDM0_A_TX_AFE,
                                         BIT(28), 0x0);
                        rtw_write32_mask(rtwdev, REG_INTPO_SETA, BIT(31), 0x0);
                        break;
                case 2:
                        rtw_dbg(rtwdev, RTW_DBG_RFK,
                                "vdf_y[1] = %x vdf_y[0] = %x\n",
                                vdf_y[1] >> 21 & 0x00007ff,
                                vdf_y[0] >> 21 & 0x00007ff);

                        rtw_dbg(rtwdev, RTW_DBG_RFK,
                                "vdf_x[1] = %x vdf_x[0] = %x\n",
                                vdf_x[1] >> 21 & 0x00007ff,
                                vdf_x[0] >> 21 & 0x00007ff);

                        tx_dt[cal] = (vdf_y[1] >> 20) - (vdf_y[0] >> 20);
                        tx_dt[cal] = (16 * tx_dt[cal]) * 10000 / 15708;
                        tx_dt[cal] = (tx_dt[cal] >> 1) + (tx_dt[cal] & BIT(0));

                        /* TX_Tone_idx[9:0], TxK_Mask[29] TX_Tone = 16 */
                        rtw_write32(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE,
                                    0x18008c20);
                        /* RX_Tone_idx[9:0], RxK_Mask[29] */
                        rtw_write32(rtwdev, REG_OFDM0_A_TX_AFE, 0x38008c20);
                        rtw_write32_mask(rtwdev, REG_INTPO_SETA, BIT(31), 0x1);
                        rtw_write32_mask(rtwdev, REG_INTPO_SETA, 0x3fff0000,
                                         tx_dt[cal] & 0x00003fff);
                        break;
                }

                rtw_write32(rtwdev, REG_RFECTL_A, 0x00100000);

                for (cal_retry = 0; cal_retry < 10; cal_retry++) {
                        /* one shot */
                        rtw_write32(rtwdev, REG_IQK_COM64, 0xfa000000);
                        rtw_write32(rtwdev, REG_IQK_COM64, 0xf8000000);

                        mdelay(10);

                        rtw_write32(rtwdev, REG_RFECTL_A, 0x00000000);

                        for (delay_count = 0; delay_count < 20; delay_count++) {
                                iqk_ready = rtw_read32_mask(rtwdev,
                                                            REG_IQKA_END,
                                                            BIT(10));

                                /* Originally: if (~iqk_ready || delay_count > 20)
                                 * that looks like a typo so make it more explicit
                                 */
                                iqk_ready = true;

                                if (iqk_ready)
                                        break;

                                mdelay(1);
                        }

                        if (delay_count < 20) {
                                /* ============TXIQK Check============== */
                                tx_fail = rtw_read32_mask(rtwdev,
                                                          REG_IQKA_END,
                                                          BIT(12));

                                /* Originally: if (~tx_fail) {
                                 * It looks like a typo, so make it more explicit.
                                 */
                                tx_fail = false;

                                if (!tx_fail) {
                                        rtw_write32(rtwdev, REG_RFECTL_A,
                                                    0x02000000);
                                        vdf_x[k] = rtw_read32_mask(rtwdev,
                                                                   REG_IQKA_END,
                                                                   0x07ff0000);
                                        vdf_x[k] <<= 21;

                                        rtw_write32(rtwdev, REG_RFECTL_A,
                                                    0x04000000);
                                        vdf_y[k] = rtw_read32_mask(rtwdev,
                                                                   REG_IQKA_END,
                                                                   0x07ff0000);
                                        vdf_y[k] <<= 21;

                                        *tx0iqkok = true;
                                        break;
                                }

                                rtw_write32_mask(rtwdev, REG_IQC_Y,
                                                 0x000007ff, 0x0);
                                rtw_write32_mask(rtwdev, REG_IQC_X,
                                                 0x000007ff, 0x200);
                        }

                        *tx0iqkok = false;
                }
        }

        if (k == 3) {
                tx_x0[cal] = vdf_x[k - 1];
                tx_y0[cal] = vdf_y[k - 1];
        }
}

static void rtw8821a_iqk_tx_vdf_false(struct rtw_dev *rtwdev, u32 cal,
                                      bool *tx0iqkok,
                                      int tx_x0[CAL_NUM_8821A],
                                      int tx_y0[CAL_NUM_8821A])
{
        u32 cal_retry, delay_count, iqk_ready, tx_fail;

        /* TX_Tone_idx[9:0], TxK_Mask[29] TX_Tone = 16 */
        rtw_write32(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE, 0x18008c10);
        /* RX_Tone_idx[9:0], RxK_Mask[29] */
        rtw_write32(rtwdev, REG_OFDM0_A_TX_AFE, 0x38008c10);
        rtw_write32(rtwdev, REG_RFECTL_A, 0x00100000);

        for (cal_retry = 0; cal_retry < 10; cal_retry++) {
                /* one shot */
                rtw_write32(rtwdev, REG_IQK_COM64, 0xfa000000);
                rtw_write32(rtwdev, REG_IQK_COM64, 0xf8000000);

                mdelay(10);
                rtw_write32(rtwdev, REG_RFECTL_A, 0x00000000);

                for (delay_count = 0; delay_count < 20; delay_count++) {
                        iqk_ready = rtw_read32_mask(rtwdev, REG_IQKA_END, BIT(10));

                        /* Originally: if (~iqk_ready || delay_count > 20)
                         * that looks like a typo so make it more explicit
                         */
                        iqk_ready = true;

                        if (iqk_ready)
                                break;

                        mdelay(1);
                }

                if (delay_count < 20) {
                        /* ============TXIQK Check============== */
                        tx_fail = rtw_read32_mask(rtwdev, REG_IQKA_END, BIT(12));

                        /* Originally: if (~tx_fail) {
                         * It looks like a typo, so make it more explicit.
                         */
                        tx_fail = false;

                        if (!tx_fail) {
                                rtw_write32(rtwdev, REG_RFECTL_A, 0x02000000);
                                tx_x0[cal] = rtw_read32_mask(rtwdev, REG_IQKA_END,
                                                             0x07ff0000);
                                tx_x0[cal] <<= 21;

                                rtw_write32(rtwdev, REG_RFECTL_A, 0x04000000);
                                tx_y0[cal] = rtw_read32_mask(rtwdev, REG_IQKA_END,
                                                             0x07ff0000);
                                tx_y0[cal] <<= 21;

                                *tx0iqkok = true;
                                break;
                        }

                        rtw_write32_mask(rtwdev, REG_IQC_Y, 0x000007ff, 0x0);
                        rtw_write32_mask(rtwdev, REG_IQC_X, 0x000007ff, 0x200);
                }

                *tx0iqkok = false;
        }
}

static void rtw8821a_iqk_rx(struct rtw_dev *rtwdev, u32 cal, bool *rx0iqkok,
                            int rx_x0[CAL_NUM_8821A],
                            int rx_y0[CAL_NUM_8821A])
{
        u32 cal_retry, delay_count, iqk_ready, rx_fail;

        rtw_write32(rtwdev, REG_RFECTL_A, 0x00100000);

        for (cal_retry = 0; cal_retry < 10; cal_retry++) {
                /* one shot */
                rtw_write32(rtwdev, REG_IQK_COM64, 0xfa000000);
                rtw_write32(rtwdev, REG_IQK_COM64, 0xf8000000);

                mdelay(10);

                rtw_write32(rtwdev, REG_RFECTL_A, 0x00000000);

                for (delay_count = 0; delay_count < 20; delay_count++) {
                        iqk_ready = rtw_read32_mask(rtwdev, REG_IQKA_END, BIT(10));

                        /* Originally: if (~iqk_ready || delay_count > 20)
                         * that looks like a typo so make it more explicit
                         */
                        iqk_ready = true;

                        if (iqk_ready)
                                break;

                        mdelay(1);
                }

                if (delay_count < 20) {
                        /* ============RXIQK Check============== */
                        rx_fail = rtw_read32_mask(rtwdev, REG_IQKA_END, BIT(11));
                        if (!rx_fail) {
                                rtw_write32(rtwdev, REG_RFECTL_A, 0x06000000);
                                rx_x0[cal] = rtw_read32_mask(rtwdev, REG_IQKA_END,
                                                             0x07ff0000);
                                rx_x0[cal] <<= 21;

                                rtw_write32(rtwdev, REG_RFECTL_A, 0x08000000);
                                rx_y0[cal] = rtw_read32_mask(rtwdev, REG_IQKA_END,
                                                             0x07ff0000);
                                rx_y0[cal] <<= 21;

                                *rx0iqkok = true;
                                break;
                        }

                        rtw_write32_mask(rtwdev, REG_RX_IQC_AB_A,
                                         0x000003ff, 0x200 >> 1);
                        rtw_write32_mask(rtwdev, REG_RX_IQC_AB_A,
                                         0x03ff0000, 0x0 >> 1);
                }

                *rx0iqkok = false;
        }
}

static void rtw8821a_iqk(struct rtw_dev *rtwdev)
{
        int tx_average = 0, rx_average = 0, rx_iqk_loop = 0;
        const struct rtw_efuse *efuse = &rtwdev->efuse;
        int tx_x = 0, tx_y = 0, rx_x = 0, rx_y = 0;
        const struct rtw_hal *hal = &rtwdev->hal;
        bool tx0iqkok = false, rx0iqkok = false;
        int rx_x_temp = 0, rx_y_temp = 0;
        int rx_x0[2][CAL_NUM_8821A];
        int rx_y0[2][CAL_NUM_8821A];
        int tx_x0[CAL_NUM_8821A];
        int tx_y0[CAL_NUM_8821A];
        bool rx_finish1 = false;
        bool rx_finish2 = false;
        bool vdf_enable;
        u32 cal;
        int i;

        rtw_dbg(rtwdev, RTW_DBG_RFK,
                "band_width = %d, ext_pa = %d, ext_pa_5g = %d\n",
                hal->current_band_width, efuse->ext_pa_2g, efuse->ext_pa_5g);

        vdf_enable = hal->current_band_width == RTW_CHANNEL_WIDTH_80;

        for (cal = 0; cal < CAL_NUM_8821A; cal++) {
                /* path-A LOK */

                /* [31] = 0 --> Page C */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);

                /* ========path-A AFE all on======== */
                /* Port 0 DAC/ADC on */
                rtw_write32(rtwdev, REG_AFE_PWR1_A, 0x77777777);
                rtw_write32(rtwdev, REG_AFE_PWR2_A, 0x77777777);

                rtw_write32(rtwdev, REG_RX_WAIT_CCA_TX_CCK_RFON_A, 0x19791979);

                /* hardware 3-wire off */
                rtw_write32_mask(rtwdev, REG_3WIRE_SWA, 0xf, 0x4);

                /* LOK setting */

                /* 1. DAC/ADC sampling rate (160 MHz) */
                rtw_write32_mask(rtwdev, REG_CK_MONHA, GENMASK(26, 24), 0x7);

                /* 2. LoK RF setting (at BW = 20M) */
                rtw_write_rf(rtwdev, RF_PATH_A, RF_LUTWE, RFREG_MASK, 0x80002);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_CFGCH, 0x00c00, 0x3);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_ADDR, RFREG_MASK,
                             0x20000);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_DATA0, RFREG_MASK,
                             0x0003f);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_DATA1, RFREG_MASK,
                             0xf3fc3);

                rtw_write_rf(rtwdev, RF_PATH_A, RF_TXA_PREPAD, RFREG_MASK,
                             0x931d5);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_RXBB2, RFREG_MASK, 0x8a001);
                rtw_write32(rtwdev, REG_DAC_RSTB, 0x00008000);
                rtw_write32_mask(rtwdev, REG_TXAGCIDX, BIT(0), 0x1);
                /* TX (X,Y) */
                rtw_write32(rtwdev, REG_IQK_COM00, 0x29002000);
                /* RX (X,Y) */
                rtw_write32(rtwdev, REG_IQK_COM32, 0xa9002000);
                /* [0]:AGC_en, [15]:idac_K_Mask */
                rtw_write32(rtwdev, REG_IQK_COM96, 0x00462910);

                /* [31] = 1 --> Page C1 */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x1);

                if (efuse->ext_pa_5g)
                        rtw_write32(rtwdev, REG_OFDM0_XB_TX_IQ_IMBALANCE,
                                    0x821403f7);
                else
                        rtw_write32(rtwdev, REG_OFDM0_XB_TX_IQ_IMBALANCE,
                                    0x821403f4);

                if (hal->current_band_type == RTW_BAND_5G)
                        rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x68163e96);
                else
                        rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x28163e96);

                /* TX_Tone_idx[9:0], TxK_Mask[29] TX_Tone = 16 */
                rtw_write32(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE, 0x18008c10);
                /* RX_Tone_idx[9:0], RxK_Mask[29] */
                rtw_write32(rtwdev, REG_OFDM0_A_TX_AFE, 0x38008c10);
                rtw_write32(rtwdev, REG_RFECTL_A, 0x00100000);
                rtw_write32(rtwdev, REG_IQK_COM64, 0xfa000000);
                rtw_write32(rtwdev, REG_IQK_COM64, 0xf8000000);

                mdelay(10);
                rtw_write32(rtwdev, REG_RFECTL_A, 0x00000000);

                /* [31] = 0 --> Page C */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_TXMOD, 0x7fe00,
                             rtw_read_rf(rtwdev, RF_PATH_A, RF_DTXLOK, 0xffc00));

                if (hal->current_band_width == RTW_CHANNEL_WIDTH_40)
                        rtw_write_rf(rtwdev, RF_PATH_A, RF_CFGCH,
                                     RF18_BW_MASK, 0x1);
                else if (hal->current_band_width == RTW_CHANNEL_WIDTH_80)
                        rtw_write_rf(rtwdev, RF_PATH_A, RF_CFGCH,
                                     RF18_BW_MASK, 0x0);

                /* [31] = 1 --> Page C1 */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x1);

                /* 3. TX RF setting */
                /* [31] = 0 --> Page C */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_LUTWE, RFREG_MASK, 0x80000);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_ADDR, RFREG_MASK,
                             0x20000);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_DATA0, RFREG_MASK,
                             0x0003f);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_DATA1, RFREG_MASK,
                             0xf3fc3);

                rtw_write_rf(rtwdev, RF_PATH_A, RF_TXA_PREPAD, RFREG_MASK, 0x931d5);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_RXBB2, RFREG_MASK, 0x8a001);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_LUTWE, RFREG_MASK, 0x00000);
                rtw_write32(rtwdev, REG_DAC_RSTB, 0x00008000);
                rtw_write32_mask(rtwdev, REG_TXAGCIDX, BIT(0), 0x1);
                /* TX (X,Y) */
                rtw_write32(rtwdev, REG_IQK_COM00, 0x29002000);
                /* RX (X,Y) */
                rtw_write32(rtwdev, REG_IQK_COM32, 0xa9002000);
                /* [0]:AGC_en, [15]:idac_K_Mask */
                rtw_write32(rtwdev, REG_IQK_COM96, 0x0046a910);

                /* [31] = 1 --> Page C1 */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x1);

                if (efuse->ext_pa_5g)
                        rtw_write32(rtwdev, REG_OFDM0_XB_TX_IQ_IMBALANCE,
                                    0x821403f7);
                else
                        rtw_write32(rtwdev, REG_OFDM0_XB_TX_IQ_IMBALANCE,
                                    0x821403e3);

                if (hal->current_band_type == RTW_BAND_5G)
                        rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x40163e96);
                else
                        rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x00163e96);

                if (vdf_enable)
                        rtw8821a_iqk_tx_vdf_true(rtwdev, cal, &tx0iqkok,
                                                 tx_x0, tx_y0);
                else
                        rtw8821a_iqk_tx_vdf_false(rtwdev, cal, &tx0iqkok,
                                                  tx_x0, tx_y0);

                if (!tx0iqkok)
                        break; /* TXK fail, Don't do RXK */

                /* ====== RX IQK ====== */
                /* [31] = 0 --> Page C */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x0);
                /* 1. RX RF setting */
                rtw_write_rf(rtwdev, RF_PATH_A, RF_LUTWE, RFREG_MASK, 0x80000);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_ADDR, RFREG_MASK,
                             0x30000);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_DATA0, RFREG_MASK,
                             0x0002f);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_MODE_TABLE_DATA1, RFREG_MASK,
                             0xfffbb);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_RXBB2, RFREG_MASK, 0x88001);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_TXA_PREPAD, RFREG_MASK, 0x931d8);
                rtw_write_rf(rtwdev, RF_PATH_A, RF_LUTWE, RFREG_MASK, 0x00000);

                rtw_write32_mask(rtwdev, REG_IQK_COM00, 0x03FF8000,
                                 (tx_x0[cal] >> 21) & 0x000007ff);
                rtw_write32_mask(rtwdev, REG_IQK_COM00, 0x000007FF,
                                 (tx_y0[cal] >> 21) & 0x000007ff);
                rtw_write32_mask(rtwdev, REG_IQK_COM00, BIT(31), 0x1);
                rtw_write32_mask(rtwdev, REG_IQK_COM00, BIT(31), 0x0);
                rtw_write32(rtwdev, REG_DAC_RSTB, 0x00008000);
                rtw_write32(rtwdev, REG_IQK_COM96, 0x0046a911);

                /* [31] = 1 --> Page C1 */
                rtw_write32_mask(rtwdev, REG_CCASEL, BIT(31), 0x1);

                /* TX_Tone_idx[9:0], TxK_Mask[29] TX_Tone = 16 */
                rtw_write32(rtwdev, REG_OFDM0_XA_TX_IQ_IMBALANCE, 0x38008c10);
                /* RX_Tone_idx[9:0], RxK_Mask[29] */
                rtw_write32(rtwdev, REG_OFDM0_A_TX_AFE, 0x18008c10);
                rtw_write32(rtwdev, REG_OFDM0_XB_TX_IQ_IMBALANCE, 0x02140119);

                if (rtw_hci_type(rtwdev) == RTW_HCI_TYPE_PCIE)
                        rx_iqk_loop = 2; /* for 2% fail; */
                else
                        rx_iqk_loop = 1;

                for (i = 0; i < rx_iqk_loop; i++) {
                        if (rtw_hci_type(rtwdev) == RTW_HCI_TYPE_PCIE && i == 0)
                                rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x28161100); /* Good */
                        else
                                rtw_write32(rtwdev, REG_TSSI_TRK_SW, 0x28160d00);

                        rtw8821a_iqk_rx(rtwdev, cal, &rx0iqkok,
                                        rx_x0[i], rx_y0[i]);
                }

                if (tx0iqkok)
                        tx_average++;
                if (rx0iqkok)
                        rx_average++;
        }

        /* FillIQK Result */

        if (tx_average == 0)
                return;

        for (i = 0; i < tx_average; i++)
                rtw_dbg(rtwdev, RTW_DBG_RFK,
                        "tx_x0[%d] = %x ;; tx_y0[%d] = %x\n",
                        i, (tx_x0[i] >> 21) & 0x000007ff,
                        i, (tx_y0[i] >> 21) & 0x000007ff);

        if (rtw88xxa_iqk_finish(tx_average, 3, tx_x0, tx_y0,
                                &tx_x, &tx_y, true, true))
                rtw8821a_iqk_tx_fill(rtwdev, tx_x, tx_y);
        else
                rtw8821a_iqk_tx_fill(rtwdev, 0x200, 0x0);

        if (rx_average == 0)
                return;

        for (i = 0; i < rx_average; i++) {
                rtw_dbg(rtwdev, RTW_DBG_RFK,
                        "rx_x0[0][%d] = %x ;; rx_y0[0][%d] = %x\n",
                        i, (rx_x0[0][i] >> 21) & 0x000007ff,
                        i, (rx_y0[0][i] >> 21) & 0x000007ff);

                if (rx_iqk_loop == 2)
                        rtw_dbg(rtwdev, RTW_DBG_RFK,
                                "rx_x0[1][%d] = %x ;; rx_y0[1][%d] = %x\n",
                                i, (rx_x0[1][i] >> 21) & 0x000007ff,
                                i, (rx_y0[1][i] >> 21) & 0x000007ff);
        }

        rx_finish1 = rtw88xxa_iqk_finish(rx_average, 4, rx_x0[0], rx_y0[0],
                                         &rx_x_temp, &rx_y_temp, true, true);

        if (rx_finish1) {
                rx_x = rx_x_temp;
                rx_y = rx_y_temp;
        }

        if (rx_iqk_loop == 2) {
                rx_finish2 = rtw88xxa_iqk_finish(rx_average, 4,
                                                 rx_x0[1], rx_y0[1],
                                                 &rx_x, &rx_y, true, true);

                if (rx_finish1 && rx_finish2) {
                        rx_x = (rx_x + rx_x_temp) / 2;
                        rx_y = (rx_y + rx_y_temp) / 2;
                }
        }

        if (rx_finish1 || rx_finish2)
                rtw8821a_iqk_rx_fill(rtwdev, rx_x, rx_y);
        else
                rtw8821a_iqk_rx_fill(rtwdev, 0x200, 0x0);
}

static void rtw8821a_do_iqk(struct rtw_dev *rtwdev)
{
        static const u32 backup_macbb_reg[MACBB_REG_NUM_8821A] = {
                0x520, 0x550, 0x808, 0xa04, 0x90c, 0xc00, 0x838, 0x82c
        };
        static const u32 backup_afe_reg[AFE_REG_NUM_8821A] = {
                0xc5c, 0xc60, 0xc64, 0xc68
        };
        static const u32 backup_rf_reg[RF_REG_NUM_8821A] = {
                0x65, 0x8f, 0x0
        };
        u32 macbb_backup[MACBB_REG_NUM_8821A];
        u32 afe_backup[AFE_REG_NUM_8821A];
        u32 rfa_backup[RF_REG_NUM_8821A];

        rtw88xxa_iqk_backup_mac_bb(rtwdev, macbb_backup,
                                   backup_macbb_reg, MACBB_REG_NUM_8821A);
        rtw88xxa_iqk_backup_afe(rtwdev, afe_backup,
                                backup_afe_reg, AFE_REG_NUM_8821A);
        rtw8821a_iqk_backup_rf(rtwdev, rfa_backup,
                               backup_rf_reg, RF_REG_NUM_8821A);

        rtw88xxa_iqk_configure_mac(rtwdev);

        rtw8821a_iqk(rtwdev);

        rtw8821a_iqk_restore_rf(rtwdev, backup_rf_reg,
                                rfa_backup, RF_REG_NUM_8821A);
        rtw8821a_iqk_restore_afe(rtwdev, afe_backup,
                                 backup_afe_reg, AFE_REG_NUM_8821A);
        rtw88xxa_iqk_restore_mac_bb(rtwdev, macbb_backup,
                                    backup_macbb_reg, MACBB_REG_NUM_8821A);
}

static void rtw8821a_phy_calibration(struct rtw_dev *rtwdev)
{
        rtw8821a_do_iqk(rtwdev);
}

static void rtw8821a_pwr_track(struct rtw_dev *rtwdev)
{
        struct rtw_dm_info *dm_info = &rtwdev->dm_info;

        if (!dm_info->pwr_trk_triggered) {
                rtw_write_rf(rtwdev, RF_PATH_A, RF_T_METER,
                             GENMASK(17, 16), 0x03);
                dm_info->pwr_trk_triggered = true;
                return;
        }

        rtw88xxa_phy_pwrtrack(rtwdev, NULL, rtw8821a_do_iqk);
        dm_info->pwr_trk_triggered = false;
}

static void rtw8821a_led_set(struct led_classdev *led,
                             enum led_brightness brightness)
{
        struct rtw_dev *rtwdev = container_of(led, struct rtw_dev, led_cdev);
        u32 gpio8_cfg;
        u8 ledcfg;

        if (brightness == LED_OFF) {
                gpio8_cfg = rtw_read32(rtwdev, REG_GPIO_PIN_CTRL_2);
                gpio8_cfg &= ~BIT(24);
                gpio8_cfg |= BIT(16) | BIT(8);
                rtw_write32(rtwdev, REG_GPIO_PIN_CTRL_2, gpio8_cfg);
        } else {
                ledcfg = rtw_read8(rtwdev, REG_LED_CFG + 2);
                gpio8_cfg = rtw_read32(rtwdev, REG_GPIO_PIN_CTRL_2);

                ledcfg &= BIT(7) | BIT(6);
                rtw_write8(rtwdev, REG_LED_CFG + 2, ledcfg);

                gpio8_cfg &= ~(BIT(24) | BIT(8));
                gpio8_cfg |= BIT(16);
                rtw_write32(rtwdev, REG_GPIO_PIN_CTRL_2, gpio8_cfg);
        }
}

static void rtw8821a_fill_txdesc_checksum(struct rtw_dev *rtwdev,
                                          struct rtw_tx_pkt_info *pkt_info,
                                          u8 *txdesc)
{
        fill_txdesc_checksum_common(txdesc, 16);
}

static void rtw8821a_coex_cfg_init(struct rtw_dev *rtwdev)
{
        u8 val8;

        /* BT report packet sample rate */
        rtw_write8_mask(rtwdev, REG_BT_TDMA_TIME, BIT_MASK_SAMPLE_RATE, 0x5);

        val8 = BIT_STATIS_BT_EN;
        if (rtwdev->efuse.share_ant)
                val8 |= BIT_R_GRANTALL_WLMASK;
        rtw_write8(rtwdev, REG_BT_COEX_ENH_INTR_CTRL, val8);

        /* enable BT counter statistics */
        rtw_write8(rtwdev, REG_BT_STAT_CTRL, 0x3);

        /* enable PTA */
        rtw_write32_set(rtwdev, REG_GPIO_MUXCFG, BIT_BT_PTA_EN);
}

static void rtw8821a_coex_cfg_ant_switch(struct rtw_dev *rtwdev, u8 ctrl_type,
                                         u8 pos_type)
{
        bool share_ant = rtwdev->efuse.share_ant;
        struct rtw_coex *coex = &rtwdev->coex;
        struct rtw_coex_dm *coex_dm = &coex->dm;
        u32 phase = coex_dm->cur_ant_pos_type;

        if (!rtwdev->efuse.btcoex)
                return;

        switch (phase) {
        case COEX_SET_ANT_POWERON:
        case COEX_SET_ANT_INIT:
                rtw_write32_clr(rtwdev, REG_LED_CFG, BIT_DPDT_SEL_EN);
                rtw_write32_set(rtwdev, REG_LED_CFG, BIT_DPDT_WL_SEL);
                rtw_write8_set(rtwdev, REG_GNT_BT, BIT_PTA_SW_CTL);

                rtw_write8(rtwdev, REG_RFE_CTRL8,
                           share_ant ? PTA_CTRL_PIN : DPDT_CTRL_PIN);
                rtw_write32_mask(rtwdev, REG_RFE_CTRL8, 0x30000000, 0x1);
                break;
        case COEX_SET_ANT_WONLY:
                rtw_write32_clr(rtwdev, REG_LED_CFG, BIT_DPDT_SEL_EN);
                rtw_write32_set(rtwdev, REG_LED_CFG, BIT_DPDT_WL_SEL);
                rtw_write8_clr(rtwdev, REG_GNT_BT, BIT_PTA_SW_CTL);

                rtw_write8(rtwdev, REG_RFE_CTRL8, DPDT_CTRL_PIN);
                rtw_write32_mask(rtwdev, REG_RFE_CTRL8, 0x30000000, 0x1);
                break;
        case COEX_SET_ANT_2G:
                rtw_write32_clr(rtwdev, REG_LED_CFG, BIT_DPDT_SEL_EN);
                rtw_write32_set(rtwdev, REG_LED_CFG, BIT_DPDT_WL_SEL);
                rtw_write8_clr(rtwdev, REG_GNT_BT, BIT_PTA_SW_CTL);

                rtw_write8(rtwdev, REG_RFE_CTRL8,
                           share_ant ? PTA_CTRL_PIN : DPDT_CTRL_PIN);
                rtw_write32_mask(rtwdev, REG_RFE_CTRL8, 0x30000000, 0x1);
                break;
        case COEX_SET_ANT_5G:
                rtw_write32_clr(rtwdev, REG_LED_CFG, BIT_DPDT_SEL_EN);
                rtw_write32_set(rtwdev, REG_LED_CFG, BIT_DPDT_WL_SEL);
                rtw_write8_set(rtwdev, REG_GNT_BT, BIT_PTA_SW_CTL);

                rtw_write8(rtwdev, REG_RFE_CTRL8, DPDT_CTRL_PIN);
                rtw_write32_mask(rtwdev, REG_RFE_CTRL8, 0x30000000,
                                 share_ant ? 0x2 : 0x1);
                break;
        case COEX_SET_ANT_WOFF:
                rtw_write32_clr(rtwdev, REG_LED_CFG, BIT_DPDT_SEL_EN);
                rtw_write32_clr(rtwdev, REG_LED_CFG, BIT_DPDT_WL_SEL);
                rtw_write8_set(rtwdev, REG_GNT_BT, BIT_PTA_SW_CTL);

                rtw_write8(rtwdev, REG_RFE_CTRL8, DPDT_CTRL_PIN);
                rtw_write32_mask(rtwdev, REG_RFE_CTRL8, 0x30000000,
                                 share_ant ? 0x2 : 0x1);
                break;
        default:
                rtw_warn(rtwdev, "%s: not handling phase %d\n",
                         __func__, phase);
                break;
        }
}

static void rtw8821a_coex_cfg_gnt_fix(struct rtw_dev *rtwdev)
{
}

static void rtw8821a_coex_cfg_gnt_debug(struct rtw_dev *rtwdev)
{
}

static void rtw8821a_coex_cfg_rfe_type(struct rtw_dev *rtwdev)
{
        struct rtw_coex *coex = &rtwdev->coex;
        struct rtw_coex_rfe *coex_rfe = &coex->rfe;

        coex_rfe->ant_switch_exist = true;
}

static void rtw8821a_coex_cfg_wl_tx_power(struct rtw_dev *rtwdev, u8 wl_pwr)
{
        struct rtw_coex *coex = &rtwdev->coex;
        struct rtw_coex_dm *coex_dm = &coex->dm;
        struct rtw_efuse *efuse = &rtwdev->efuse;
        bool share_ant = efuse->share_ant;

        if (share_ant)
                return;

        if (wl_pwr == coex_dm->cur_wl_pwr_lvl)
                return;

        coex_dm->cur_wl_pwr_lvl = wl_pwr;
}

static void rtw8821a_coex_cfg_wl_rx_gain(struct rtw_dev *rtwdev, bool low_gain)
{
}

static const struct rtw_chip_ops rtw8821a_ops = {
        .power_on               = rtw88xxa_power_on,
        .power_off              = rtw8821a_power_off,
        .phy_set_param          = NULL,
        .read_efuse             = rtw88xxa_read_efuse,
        .query_phy_status       = rtw8821a_query_phy_status,
        .set_channel            = rtw88xxa_set_channel,
        .mac_init               = NULL,
        .mac_postinit           = NULL,
        .read_rf                = rtw88xxa_phy_read_rf,
        .write_rf               = rtw_phy_write_rf_reg_sipi,
        .set_antenna            = NULL,
        .set_tx_power_index     = rtw88xxa_set_tx_power_index,
        .cfg_ldo25              = rtw8821a_cfg_ldo25,
        .efuse_grant            = rtw88xxa_efuse_grant,
        .set_ampdu_factor       = NULL,
        .false_alarm_statistics = rtw88xxa_false_alarm_statistics,
        .phy_calibration        = rtw8821a_phy_calibration,
        .cck_pd_set             = rtw88xxa_phy_cck_pd_set,
        .pwr_track              = rtw8821a_pwr_track,
        .config_bfee            = NULL,
        .set_gid_table          = NULL,
        .cfg_csi_rate           = NULL,
        .led_set                = rtw8821a_led_set,
        .fill_txdesc_checksum   = rtw8821a_fill_txdesc_checksum,
        .coex_set_init          = rtw8821a_coex_cfg_init,
        .coex_set_ant_switch    = rtw8821a_coex_cfg_ant_switch,
        .coex_set_gnt_fix       = rtw8821a_coex_cfg_gnt_fix,
        .coex_set_gnt_debug     = rtw8821a_coex_cfg_gnt_debug,
        .coex_set_rfe_type      = rtw8821a_coex_cfg_rfe_type,
        .coex_set_wl_tx_power   = rtw8821a_coex_cfg_wl_tx_power,
        .coex_set_wl_rx_gain    = rtw8821a_coex_cfg_wl_rx_gain,
};

static const struct rtw_page_table page_table_8821a[] = {
        /* hq_num, nq_num, lq_num, exq_num, gapq_num */
        {0, 0, 0, 0, 0},        /* SDIO */
        {0, 0, 0, 0, 0},        /* PCI */
        {8, 0, 0, 0, 1},        /* 2 bulk out endpoints */
        {8, 0, 8, 0, 1},        /* 3 bulk out endpoints */
        {8, 0, 8, 4, 1},        /* 4 bulk out endpoints */
};

static const struct rtw_rqpn rqpn_table_8821a[] = {
        {RTW_DMA_MAPPING_NORMAL, RTW_DMA_MAPPING_NORMAL,
         RTW_DMA_MAPPING_LOW, RTW_DMA_MAPPING_LOW,
         RTW_DMA_MAPPING_EXTRA, RTW_DMA_MAPPING_HIGH},

        {RTW_DMA_MAPPING_NORMAL, RTW_DMA_MAPPING_NORMAL,
         RTW_DMA_MAPPING_LOW, RTW_DMA_MAPPING_LOW,
         RTW_DMA_MAPPING_EXTRA, RTW_DMA_MAPPING_HIGH},

        {RTW_DMA_MAPPING_HIGH, RTW_DMA_MAPPING_HIGH,
         RTW_DMA_MAPPING_NORMAL, RTW_DMA_MAPPING_NORMAL,
         RTW_DMA_MAPPING_HIGH, RTW_DMA_MAPPING_HIGH},

        {RTW_DMA_MAPPING_HIGH, RTW_DMA_MAPPING_NORMAL,
         RTW_DMA_MAPPING_LOW, RTW_DMA_MAPPING_LOW,
         RTW_DMA_MAPPING_HIGH, RTW_DMA_MAPPING_HIGH},

        {RTW_DMA_MAPPING_NORMAL, RTW_DMA_MAPPING_NORMAL,
         RTW_DMA_MAPPING_LOW, RTW_DMA_MAPPING_LOW,
         RTW_DMA_MAPPING_EXTRA, RTW_DMA_MAPPING_HIGH},
};

static const struct rtw_prioq_addrs prioq_addrs_8821a = {
        .prio[RTW_DMA_MAPPING_EXTRA] = {
                .rsvd = REG_RQPN_NPQ + 2, .avail = REG_RQPN_NPQ + 3,
        },
        .prio[RTW_DMA_MAPPING_LOW] = {
                .rsvd = REG_RQPN + 1, .avail = REG_FIFOPAGE_CTRL_2 + 1,
        },
        .prio[RTW_DMA_MAPPING_NORMAL] = {
                .rsvd = REG_RQPN_NPQ, .avail = REG_RQPN_NPQ + 1,
        },
        .prio[RTW_DMA_MAPPING_HIGH] = {
                .rsvd = REG_RQPN, .avail = REG_FIFOPAGE_CTRL_2,
        },
        .wsize = false,
};

static const struct rtw_hw_reg rtw8821a_dig[] = {
        [0] = { .addr = REG_RXIGI_A, .mask = 0x7f },
};

static const struct rtw_rfe_def rtw8821a_rfe_defs[] = {
        [0] = { .phy_pg_tbl     = &rtw8821a_bb_pg_tbl,
                .txpwr_lmt_tbl  = &rtw8821a_txpwr_lmt_tbl,
                .pwr_track_tbl  = &rtw8821a_rtw_pwr_track_tbl, },
};

/* TODO */
/* rssi in percentage % (dbm = % - 100) */
static const u8 wl_rssi_step_8821a[] = {101, 45, 101, 40};
static const u8 bt_rssi_step_8821a[] = {101, 101, 101, 101};

/* table_sant_8821a, table_nsant_8821a, tdma_sant_8821a, and tdma_nsant_8821a
 * are copied from rtw8821c.c because the 8821au driver's tables are not
 * compatible with the coex code in rtw88.
 *
 * tdma case 112 (A2DP) byte 0 had to be modified from 0x61 to 0x51,
 * otherwise the firmware gets confused after pausing the music:
 * rtw_8821au 1-2:1.2: [BTCoex], Bt_info[1], len=7, data=[81 00 0a 01 00 00]
 * - 81 means PAN (personal area network) when it should be 4x (A2DP)
 * The music is not smooth with the PAN algorithm.
 */

/* Shared-Antenna Coex Table */
static const struct coex_table_para table_sant_8821a[] = {
        {0x55555555, 0x55555555}, /* case-0 */
        {0x55555555, 0x55555555},
        {0x66555555, 0x66555555},
        {0xaaaaaaaa, 0xaaaaaaaa},
        {0x5a5a5a5a, 0x5a5a5a5a},
        {0xfafafafa, 0xfafafafa}, /* case-5 */
        {0x6a5a5555, 0xaaaaaaaa},
        {0x6a5a56aa, 0x6a5a56aa},
        {0x6a5a5a5a, 0x6a5a5a5a},
        {0x66555555, 0x5a5a5a5a},
        {0x66555555, 0x6a5a5a5a}, /* case-10 */
        {0x66555555, 0xaaaaaaaa},
        {0x66555555, 0x6a5a5aaa},
        {0x66555555, 0x6aaa6aaa},
        {0x66555555, 0x6a5a5aaa},
        {0x66555555, 0xaaaaaaaa}, /* case-15 */
        {0xffff55ff, 0xfafafafa},
        {0xffff55ff, 0x6afa5afa},
        {0xaaffffaa, 0xfafafafa},
        {0xaa5555aa, 0x5a5a5a5a},
        {0xaa5555aa, 0x6a5a5a5a}, /* case-20 */
        {0xaa5555aa, 0xaaaaaaaa},
        {0xffffffff, 0x55555555},
        {0xffffffff, 0x5a5a5a5a},
        {0xffffffff, 0x5a5a5a5a},
        {0xffffffff, 0x5a5a5aaa}, /* case-25 */
        {0x55555555, 0x5a5a5a5a},
        {0x55555555, 0xaaaaaaaa},
        {0x66555555, 0x6a5a6a5a},
        {0x66556655, 0x66556655},
        {0x66556aaa, 0x6a5a6aaa}, /* case-30 */
        {0xffffffff, 0x5aaa5aaa},
        {0x56555555, 0x5a5a5aaa}
};

/* Non-Shared-Antenna Coex Table */
static const struct coex_table_para table_nsant_8821a[] = {
        {0xffffffff, 0xffffffff}, /* case-100 */
        {0xffff55ff, 0xfafafafa},
        {0x66555555, 0x66555555},
        {0xaaaaaaaa, 0xaaaaaaaa},
        {0x5a5a5a5a, 0x5a5a5a5a},
        {0xffffffff, 0xffffffff}, /* case-105 */
        {0x5afa5afa, 0x5afa5afa},
        {0x55555555, 0xfafafafa},
        {0x66555555, 0xfafafafa},
        {0x66555555, 0x5a5a5a5a},
        {0x66555555, 0x6a5a5a5a}, /* case-110 */
        {0x66555555, 0xaaaaaaaa},
        {0xffff55ff, 0xfafafafa},
        {0xffff55ff, 0x5afa5afa},
        {0xffff55ff, 0xaaaaaaaa},
        {0xffff55ff, 0xffff55ff}, /* case-115 */
        {0xaaffffaa, 0x5afa5afa},
        {0xaaffffaa, 0xaaaaaaaa},
        {0xffffffff, 0xfafafafa},
        {0xffff55ff, 0xfafafafa},
        {0xffffffff, 0xaaaaaaaa}, /* case-120 */
        {0xffff55ff, 0x5afa5afa},
        {0xffff55ff, 0x5afa5afa},
        {0x55ff55ff, 0x55ff55ff}
};

/* Shared-Antenna TDMA */
static const struct coex_tdma_para tdma_sant_8821a[] = {
        { {0x00, 0x00, 0x00, 0x00, 0x00} }, /* case-0 */
        { {0x61, 0x45, 0x03, 0x11, 0x11} }, /* case-1 */
        { {0x61, 0x3a, 0x03, 0x11, 0x11} },
        { {0x61, 0x35, 0x03, 0x11, 0x11} },
        { {0x61, 0x20, 0x03, 0x11, 0x11} },
        { {0x61, 0x3a, 0x03, 0x11, 0x11} }, /* case-5 */
        { {0x61, 0x45, 0x03, 0x11, 0x10} },
        { {0x61, 0x35, 0x03, 0x11, 0x10} },
        { {0x61, 0x30, 0x03, 0x11, 0x10} },
        { {0x61, 0x20, 0x03, 0x11, 0x10} },
        { {0x61, 0x10, 0x03, 0x11, 0x10} }, /* case-10 */
        { {0x61, 0x08, 0x03, 0x11, 0x15} },
        { {0x61, 0x08, 0x03, 0x10, 0x14} },
        { {0x51, 0x08, 0x03, 0x10, 0x54} },
        { {0x51, 0x08, 0x03, 0x10, 0x55} },
        { {0x51, 0x08, 0x07, 0x10, 0x54} }, /* case-15 */
        { {0x51, 0x45, 0x03, 0x10, 0x50} },
        { {0x51, 0x3a, 0x03, 0x11, 0x50} },
        { {0x51, 0x30, 0x03, 0x10, 0x50} },
        { {0x51, 0x21, 0x03, 0x10, 0x50} },
        { {0x51, 0x10, 0x03, 0x10, 0x50} }, /* case-20 */
        { {0x51, 0x4a, 0x03, 0x10, 0x50} },
        { {0x51, 0x08, 0x03, 0x30, 0x54} },
        { {0x55, 0x08, 0x03, 0x10, 0x54} },
        { {0x65, 0x10, 0x03, 0x11, 0x10} },
        { {0x51, 0x10, 0x03, 0x10, 0x51} }, /* case-25 */
        { {0x51, 0x21, 0x03, 0x10, 0x50} },
        { {0x61, 0x08, 0x03, 0x11, 0x11} }
};

/* Non-Shared-Antenna TDMA */
static const struct coex_tdma_para tdma_nsant_8821a[] = {
        { {0x00, 0x00, 0x00, 0x40, 0x00} }, /* case-100 */
        { {0x61, 0x45, 0x03, 0x11, 0x11} },
        { {0x61, 0x25, 0x03, 0x11, 0x11} },
        { {0x61, 0x35, 0x03, 0x11, 0x11} },
        { {0x61, 0x20, 0x03, 0x11, 0x11} },
        { {0x61, 0x10, 0x03, 0x11, 0x11} }, /* case-105 */
        { {0x61, 0x45, 0x03, 0x11, 0x10} },
        { {0x61, 0x30, 0x03, 0x11, 0x10} },
        { {0x61, 0x30, 0x03, 0x11, 0x10} },
        { {0x61, 0x20, 0x03, 0x11, 0x10} },
        { {0x61, 0x10, 0x03, 0x11, 0x10} }, /* case-110 */
        { {0x61, 0x10, 0x03, 0x11, 0x11} },
        { {0x51, 0x08, 0x03, 0x10, 0x14} }, /* a2dp high rssi */
        { {0x51, 0x08, 0x03, 0x10, 0x54} }, /* a2dp not high rssi */
        { {0x51, 0x08, 0x03, 0x10, 0x55} },
        { {0x51, 0x08, 0x07, 0x10, 0x54} }, /* case-115 */
        { {0x51, 0x45, 0x03, 0x10, 0x50} },
        { {0x51, 0x3a, 0x03, 0x10, 0x50} },
        { {0x51, 0x30, 0x03, 0x10, 0x50} },
        { {0x51, 0x21, 0x03, 0x10, 0x50} },
        { {0x51, 0x21, 0x03, 0x10, 0x50} }, /* case-120 */
        { {0x51, 0x10, 0x03, 0x10, 0x50} }
};

/* TODO */
static const struct coex_rf_para rf_para_tx_8821a[] = {
        {0, 0, false, 7},  /* for normal */
        {0, 20, false, 7}, /* for WL-CPT */
        {8, 17, true, 4},
        {7, 18, true, 4},
        {6, 19, true, 4},
        {5, 20, true, 4}
};

static const struct coex_rf_para rf_para_rx_8821a[] = {
        {0, 0, false, 7},  /* for normal */
        {0, 20, false, 7}, /* for WL-CPT */
        {3, 24, true, 5},
        {2, 26, true, 5},
        {1, 27, true, 5},
        {0, 28, true, 5}
};

static_assert(ARRAY_SIZE(rf_para_tx_8821a) == ARRAY_SIZE(rf_para_rx_8821a));

static const struct coex_5g_afh_map afh_5g_8821a[] = { {0, 0, 0} };

static const struct rtw_reg_domain coex_info_hw_regs_8821a[] = {
        {0xCB0, MASKDWORD, RTW_REG_DOMAIN_MAC32},
        {0xCB4, MASKDWORD, RTW_REG_DOMAIN_MAC32},
        {0xCBA, MASKBYTE0, RTW_REG_DOMAIN_MAC8},
        {0, 0, RTW_REG_DOMAIN_NL},
        {0x430, MASKDWORD, RTW_REG_DOMAIN_MAC32},
        {0x434, MASKDWORD, RTW_REG_DOMAIN_MAC32},
        {0x42a, MASKLWORD, RTW_REG_DOMAIN_MAC16},
        {0x426, MASKBYTE0, RTW_REG_DOMAIN_MAC8},
        {0x45e, BIT(3), RTW_REG_DOMAIN_MAC8},
        {0x454, MASKLWORD, RTW_REG_DOMAIN_MAC16},
        {0, 0, RTW_REG_DOMAIN_NL},
        {0x4c, BIT(24) | BIT(23), RTW_REG_DOMAIN_MAC32},
        {0x64, BIT(0), RTW_REG_DOMAIN_MAC8},
        {0x4c6, BIT(4), RTW_REG_DOMAIN_MAC8},
        {0x40, BIT(5), RTW_REG_DOMAIN_MAC8},
        {0x1, RFREG_MASK, RTW_REG_DOMAIN_RF_A},
        {0, 0, RTW_REG_DOMAIN_NL},
        {0x550, MASKDWORD, RTW_REG_DOMAIN_MAC32},
        {0x522, MASKBYTE0, RTW_REG_DOMAIN_MAC8},
        {0x953, BIT(1), RTW_REG_DOMAIN_MAC8},
        {0xc50,  MASKBYTE0, RTW_REG_DOMAIN_MAC8},
        {0x60A, MASKBYTE0, RTW_REG_DOMAIN_MAC8},
};

const struct rtw_chip_info rtw8821a_hw_spec = {
        .ops = &rtw8821a_ops,
        .id = RTW_CHIP_TYPE_8821A,
        .fw_name = "rtw88/rtw8821a_fw.bin",
        .wlan_cpu = RTW_WCPU_8051,
        .tx_pkt_desc_sz = 40,
        .tx_buf_desc_sz = 16,
        .rx_pkt_desc_sz = 24,
        .rx_buf_desc_sz = 8,
        .phy_efuse_size = 512,
        .log_efuse_size = 512,
        .ptct_efuse_size = 0,
        .txff_size = 65536,
        .rxff_size = 16128,
        .rsvd_drv_pg_num = 8,
        .txgi_factor = 1,
        .is_pwr_by_rate_dec = true,
        .max_power_index = 0x3f,
        .csi_buf_pg_num = 0,
        .band = RTW_BAND_2G | RTW_BAND_5G,
        .page_size = 256,
        .dig_min = 0x20,
        .ht_supported = true,
        .vht_supported = true,
        .lps_deep_mode_supported = 0,
        .sys_func_en = 0xFD,
        .pwr_on_seq = card_enable_flow_8821a,
        .pwr_off_seq = card_disable_flow_8821a,
        .page_table = page_table_8821a,
        .rqpn_table = rqpn_table_8821a,
        .prioq_addrs = &prioq_addrs_8821a,
        .intf_table = NULL,
        .dig = rtw8821a_dig,
        .rf_sipi_addr = {REG_LSSI_WRITE_A, REG_LSSI_WRITE_B},
        .ltecoex_addr = NULL,
        .mac_tbl = &rtw8821a_mac_tbl,
        .agc_tbl = &rtw8821a_agc_tbl,
        .bb_tbl = &rtw8821a_bb_tbl,
        .rf_tbl = {&rtw8821a_rf_a_tbl},
        .rfe_defs = rtw8821a_rfe_defs,
        .rfe_defs_size = ARRAY_SIZE(rtw8821a_rfe_defs),
        .rx_ldpc = false,
        .amsdu_in_ampdu = true,
        .hw_feature_report = false,
        .c2h_ra_report_size = 4,
        .old_datarate_fb_limit = true,
        .usb_tx_agg_desc_num = 6,
        .iqk_threshold = 8,
        .ampdu_density = IEEE80211_HT_MPDU_DENSITY_16,
        .max_scan_ie_len = IEEE80211_MAX_DATA_LEN,

        .coex_para_ver = 20190509, /* glcoex_ver_date_8821a_1ant */
        .bt_desired_ver = 0x62, /* But for 2 ant it's 0x5c */
        .scbd_support = false,
        .new_scbd10_def = false,
        .ble_hid_profile_support = false,
        .wl_mimo_ps_support = false,
        .pstdma_type = COEX_PSTDMA_FORCE_LPSOFF,
        .bt_rssi_type = COEX_BTRSSI_RATIO,
        .ant_isolation = 10,
        .rssi_tolerance = 2,
        .wl_rssi_step = wl_rssi_step_8821a,
        .bt_rssi_step = bt_rssi_step_8821a,
        .table_sant_num = ARRAY_SIZE(table_sant_8821a),
        .table_sant = table_sant_8821a,
        .table_nsant_num = ARRAY_SIZE(table_nsant_8821a),
        .table_nsant = table_nsant_8821a,
        .tdma_sant_num = ARRAY_SIZE(tdma_sant_8821a),
        .tdma_sant = tdma_sant_8821a,
        .tdma_nsant_num = ARRAY_SIZE(tdma_nsant_8821a),
        .tdma_nsant = tdma_nsant_8821a,
        .wl_rf_para_num = ARRAY_SIZE(rf_para_tx_8821a),
        .wl_rf_para_tx = rf_para_tx_8821a,
        .wl_rf_para_rx = rf_para_rx_8821a,
        .bt_afh_span_bw20 = 0x20,
        .bt_afh_span_bw40 = 0x30,
        .afh_5g_num = ARRAY_SIZE(afh_5g_8821a),
        .afh_5g = afh_5g_8821a,

        .coex_info_hw_regs_num = ARRAY_SIZE(coex_info_hw_regs_8821a),
        .coex_info_hw_regs = coex_info_hw_regs_8821a,
};
EXPORT_SYMBOL(rtw8821a_hw_spec);

MODULE_FIRMWARE("rtw88/rtw8821a_fw.bin");

MODULE_AUTHOR("Realtek Corporation");
MODULE_DESCRIPTION("Realtek 802.11ac wireless 8821a/8811a driver");
MODULE_LICENSE("Dual BSD/GPL");