1e3ec7017SPing-Ke Shih // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2e3ec7017SPing-Ke Shih /* Copyright(c) 2019-2020 Realtek Corporation
3e3ec7017SPing-Ke Shih */
4e3ec7017SPing-Ke Shih
5e3ec7017SPing-Ke Shih #include "debug.h"
6e3ec7017SPing-Ke Shih #include "efuse.h"
7bdfbf06cSPing-Ke Shih #include "mac.h"
8e3ec7017SPing-Ke Shih #include "reg.h"
9e3ec7017SPing-Ke Shih
10*40bb2ab4SPing-Ke Shih #define EF_FV_OFSET 0x5ea
11*40bb2ab4SPing-Ke Shih #define EF_CV_MASK GENMASK(7, 4)
12*40bb2ab4SPing-Ke Shih #define EF_CV_INV 15
13*40bb2ab4SPing-Ke Shih
14e3ec7017SPing-Ke Shih enum rtw89_efuse_bank {
15e3ec7017SPing-Ke Shih RTW89_EFUSE_BANK_WIFI,
16e3ec7017SPing-Ke Shih RTW89_EFUSE_BANK_BT,
17e3ec7017SPing-Ke Shih };
18e3ec7017SPing-Ke Shih
rtw89_switch_efuse_bank(struct rtw89_dev * rtwdev,enum rtw89_efuse_bank bank)19e3ec7017SPing-Ke Shih static int rtw89_switch_efuse_bank(struct rtw89_dev *rtwdev,
20e3ec7017SPing-Ke Shih enum rtw89_efuse_bank bank)
21e3ec7017SPing-Ke Shih {
22e3ec7017SPing-Ke Shih u8 val;
23e3ec7017SPing-Ke Shih
24bdfbf06cSPing-Ke Shih if (rtwdev->chip->chip_id != RTL8852A)
25bdfbf06cSPing-Ke Shih return 0;
26bdfbf06cSPing-Ke Shih
27e3ec7017SPing-Ke Shih val = rtw89_read32_mask(rtwdev, R_AX_EFUSE_CTRL_1,
28e3ec7017SPing-Ke Shih B_AX_EF_CELL_SEL_MASK);
29e3ec7017SPing-Ke Shih if (bank == val)
30e3ec7017SPing-Ke Shih return 0;
31e3ec7017SPing-Ke Shih
32e3ec7017SPing-Ke Shih rtw89_write32_mask(rtwdev, R_AX_EFUSE_CTRL_1, B_AX_EF_CELL_SEL_MASK,
33e3ec7017SPing-Ke Shih bank);
34e3ec7017SPing-Ke Shih
35e3ec7017SPing-Ke Shih val = rtw89_read32_mask(rtwdev, R_AX_EFUSE_CTRL_1,
36e3ec7017SPing-Ke Shih B_AX_EF_CELL_SEL_MASK);
37e3ec7017SPing-Ke Shih if (bank == val)
38e3ec7017SPing-Ke Shih return 0;
39e3ec7017SPing-Ke Shih
40e3ec7017SPing-Ke Shih return -EBUSY;
41e3ec7017SPing-Ke Shih }
42e3ec7017SPing-Ke Shih
rtw89_enable_otp_burst_mode(struct rtw89_dev * rtwdev,bool en)43bdfbf06cSPing-Ke Shih static void rtw89_enable_otp_burst_mode(struct rtw89_dev *rtwdev, bool en)
44bdfbf06cSPing-Ke Shih {
45bdfbf06cSPing-Ke Shih if (en)
46bdfbf06cSPing-Ke Shih rtw89_write32_set(rtwdev, R_AX_EFUSE_CTRL_1_V1, B_AX_EF_BURST);
47bdfbf06cSPing-Ke Shih else
48bdfbf06cSPing-Ke Shih rtw89_write32_clr(rtwdev, R_AX_EFUSE_CTRL_1_V1, B_AX_EF_BURST);
49bdfbf06cSPing-Ke Shih }
50bdfbf06cSPing-Ke Shih
rtw89_enable_efuse_pwr_cut_ddv(struct rtw89_dev * rtwdev)51bdfbf06cSPing-Ke Shih static void rtw89_enable_efuse_pwr_cut_ddv(struct rtw89_dev *rtwdev)
52bdfbf06cSPing-Ke Shih {
53bdfbf06cSPing-Ke Shih enum rtw89_core_chip_id chip_id = rtwdev->chip->chip_id;
54bdfbf06cSPing-Ke Shih struct rtw89_hal *hal = &rtwdev->hal;
55bdfbf06cSPing-Ke Shih
56bdfbf06cSPing-Ke Shih if (chip_id == RTL8852A)
57bdfbf06cSPing-Ke Shih return;
58bdfbf06cSPing-Ke Shih
59bdfbf06cSPing-Ke Shih rtw89_write8_set(rtwdev, R_AX_PMC_DBG_CTRL2, B_AX_SYSON_DIS_PMCR_AX_WRMSK);
60bdfbf06cSPing-Ke Shih rtw89_write16_set(rtwdev, R_AX_SYS_ISO_CTRL, B_AX_PWC_EV2EF_B14);
61bdfbf06cSPing-Ke Shih
62bdfbf06cSPing-Ke Shih fsleep(1000);
63bdfbf06cSPing-Ke Shih
64bdfbf06cSPing-Ke Shih rtw89_write16_set(rtwdev, R_AX_SYS_ISO_CTRL, B_AX_PWC_EV2EF_B15);
65bdfbf06cSPing-Ke Shih rtw89_write16_clr(rtwdev, R_AX_SYS_ISO_CTRL, B_AX_ISO_EB2CORE);
66bdfbf06cSPing-Ke Shih if (chip_id == RTL8852B && hal->cv == CHIP_CAV)
67bdfbf06cSPing-Ke Shih rtw89_enable_otp_burst_mode(rtwdev, true);
68bdfbf06cSPing-Ke Shih }
69bdfbf06cSPing-Ke Shih
rtw89_disable_efuse_pwr_cut_ddv(struct rtw89_dev * rtwdev)70bdfbf06cSPing-Ke Shih static void rtw89_disable_efuse_pwr_cut_ddv(struct rtw89_dev *rtwdev)
71bdfbf06cSPing-Ke Shih {
72bdfbf06cSPing-Ke Shih enum rtw89_core_chip_id chip_id = rtwdev->chip->chip_id;
73bdfbf06cSPing-Ke Shih struct rtw89_hal *hal = &rtwdev->hal;
74bdfbf06cSPing-Ke Shih
75bdfbf06cSPing-Ke Shih if (chip_id == RTL8852A)
76bdfbf06cSPing-Ke Shih return;
77bdfbf06cSPing-Ke Shih
78bdfbf06cSPing-Ke Shih if (chip_id == RTL8852B && hal->cv == CHIP_CAV)
79bdfbf06cSPing-Ke Shih rtw89_enable_otp_burst_mode(rtwdev, false);
80bdfbf06cSPing-Ke Shih
81bdfbf06cSPing-Ke Shih rtw89_write16_set(rtwdev, R_AX_SYS_ISO_CTRL, B_AX_ISO_EB2CORE);
82bdfbf06cSPing-Ke Shih rtw89_write16_clr(rtwdev, R_AX_SYS_ISO_CTRL, B_AX_PWC_EV2EF_B15);
83bdfbf06cSPing-Ke Shih
84bdfbf06cSPing-Ke Shih fsleep(1000);
85bdfbf06cSPing-Ke Shih
86bdfbf06cSPing-Ke Shih rtw89_write16_clr(rtwdev, R_AX_SYS_ISO_CTRL, B_AX_PWC_EV2EF_B14);
87bdfbf06cSPing-Ke Shih rtw89_write8_clr(rtwdev, R_AX_PMC_DBG_CTRL2, B_AX_SYSON_DIS_PMCR_AX_WRMSK);
88bdfbf06cSPing-Ke Shih }
89bdfbf06cSPing-Ke Shih
rtw89_dump_physical_efuse_map_ddv(struct rtw89_dev * rtwdev,u8 * map,u32 dump_addr,u32 dump_size)90bdfbf06cSPing-Ke Shih static int rtw89_dump_physical_efuse_map_ddv(struct rtw89_dev *rtwdev, u8 *map,
91e3ec7017SPing-Ke Shih u32 dump_addr, u32 dump_size)
92e3ec7017SPing-Ke Shih {
93e3ec7017SPing-Ke Shih u32 efuse_ctl;
94e3ec7017SPing-Ke Shih u32 addr;
95e3ec7017SPing-Ke Shih int ret;
96e3ec7017SPing-Ke Shih
97bdfbf06cSPing-Ke Shih rtw89_enable_efuse_pwr_cut_ddv(rtwdev);
98e3ec7017SPing-Ke Shih
99e3ec7017SPing-Ke Shih for (addr = dump_addr; addr < dump_addr + dump_size; addr++) {
100e3ec7017SPing-Ke Shih efuse_ctl = u32_encode_bits(addr, B_AX_EF_ADDR_MASK);
101e3ec7017SPing-Ke Shih rtw89_write32(rtwdev, R_AX_EFUSE_CTRL, efuse_ctl & ~B_AX_EF_RDY);
102e3ec7017SPing-Ke Shih
103e3ec7017SPing-Ke Shih ret = read_poll_timeout_atomic(rtw89_read32, efuse_ctl,
104e3ec7017SPing-Ke Shih efuse_ctl & B_AX_EF_RDY, 1, 1000000,
105e3ec7017SPing-Ke Shih true, rtwdev, R_AX_EFUSE_CTRL);
106e3ec7017SPing-Ke Shih if (ret)
107e3ec7017SPing-Ke Shih return -EBUSY;
108e3ec7017SPing-Ke Shih
109e3ec7017SPing-Ke Shih *map++ = (u8)(efuse_ctl & 0xff);
110e3ec7017SPing-Ke Shih }
111e3ec7017SPing-Ke Shih
112bdfbf06cSPing-Ke Shih rtw89_disable_efuse_pwr_cut_ddv(rtwdev);
113bdfbf06cSPing-Ke Shih
114bdfbf06cSPing-Ke Shih return 0;
115bdfbf06cSPing-Ke Shih }
116bdfbf06cSPing-Ke Shih
rtw89_dump_physical_efuse_map_dav(struct rtw89_dev * rtwdev,u8 * map,u32 dump_addr,u32 dump_size)117bdfbf06cSPing-Ke Shih static int rtw89_dump_physical_efuse_map_dav(struct rtw89_dev *rtwdev, u8 *map,
118bdfbf06cSPing-Ke Shih u32 dump_addr, u32 dump_size)
119bdfbf06cSPing-Ke Shih {
120bdfbf06cSPing-Ke Shih u32 addr;
121bdfbf06cSPing-Ke Shih u8 val8;
122bdfbf06cSPing-Ke Shih int err;
123bdfbf06cSPing-Ke Shih int ret;
124bdfbf06cSPing-Ke Shih
125bdfbf06cSPing-Ke Shih for (addr = dump_addr; addr < dump_addr + dump_size; addr++) {
126bdfbf06cSPing-Ke Shih ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_CTRL, 0x40, FULL_BIT_MASK);
127bdfbf06cSPing-Ke Shih if (ret)
128bdfbf06cSPing-Ke Shih return ret;
129bdfbf06cSPing-Ke Shih ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_LOW_ADDR,
130bdfbf06cSPing-Ke Shih addr & 0xff, XTAL_SI_LOW_ADDR_MASK);
131bdfbf06cSPing-Ke Shih if (ret)
132bdfbf06cSPing-Ke Shih return ret;
133bdfbf06cSPing-Ke Shih ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_CTRL, addr >> 8,
134bdfbf06cSPing-Ke Shih XTAL_SI_HIGH_ADDR_MASK);
135bdfbf06cSPing-Ke Shih if (ret)
136bdfbf06cSPing-Ke Shih return ret;
137bdfbf06cSPing-Ke Shih ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_CTRL, 0,
138bdfbf06cSPing-Ke Shih XTAL_SI_MODE_SEL_MASK);
139bdfbf06cSPing-Ke Shih if (ret)
140bdfbf06cSPing-Ke Shih return ret;
141bdfbf06cSPing-Ke Shih
142bdfbf06cSPing-Ke Shih ret = read_poll_timeout_atomic(rtw89_mac_read_xtal_si, err,
143bdfbf06cSPing-Ke Shih !err && (val8 & XTAL_SI_RDY),
144bdfbf06cSPing-Ke Shih 1, 10000, false,
145bdfbf06cSPing-Ke Shih rtwdev, XTAL_SI_CTRL, &val8);
146bdfbf06cSPing-Ke Shih if (ret) {
147bdfbf06cSPing-Ke Shih rtw89_warn(rtwdev, "failed to read dav efuse\n");
148bdfbf06cSPing-Ke Shih return ret;
149bdfbf06cSPing-Ke Shih }
150bdfbf06cSPing-Ke Shih
151bdfbf06cSPing-Ke Shih ret = rtw89_mac_read_xtal_si(rtwdev, XTAL_SI_READ_VAL, &val8);
152bdfbf06cSPing-Ke Shih if (ret)
153bdfbf06cSPing-Ke Shih return ret;
154bdfbf06cSPing-Ke Shih *map++ = val8;
155bdfbf06cSPing-Ke Shih }
156bdfbf06cSPing-Ke Shih
157bdfbf06cSPing-Ke Shih return 0;
158bdfbf06cSPing-Ke Shih }
159bdfbf06cSPing-Ke Shih
rtw89_dump_physical_efuse_map(struct rtw89_dev * rtwdev,u8 * map,u32 dump_addr,u32 dump_size,bool dav)160bdfbf06cSPing-Ke Shih static int rtw89_dump_physical_efuse_map(struct rtw89_dev *rtwdev, u8 *map,
161bdfbf06cSPing-Ke Shih u32 dump_addr, u32 dump_size, bool dav)
162bdfbf06cSPing-Ke Shih {
163bdfbf06cSPing-Ke Shih int ret;
164bdfbf06cSPing-Ke Shih
165bdfbf06cSPing-Ke Shih if (!map || dump_size == 0)
166bdfbf06cSPing-Ke Shih return 0;
167bdfbf06cSPing-Ke Shih
168bdfbf06cSPing-Ke Shih rtw89_switch_efuse_bank(rtwdev, RTW89_EFUSE_BANK_WIFI);
169bdfbf06cSPing-Ke Shih
170bdfbf06cSPing-Ke Shih if (dav) {
171bdfbf06cSPing-Ke Shih ret = rtw89_dump_physical_efuse_map_dav(rtwdev, map, dump_addr, dump_size);
172bdfbf06cSPing-Ke Shih if (ret)
173bdfbf06cSPing-Ke Shih return ret;
174bdfbf06cSPing-Ke Shih } else {
175bdfbf06cSPing-Ke Shih ret = rtw89_dump_physical_efuse_map_ddv(rtwdev, map, dump_addr, dump_size);
176bdfbf06cSPing-Ke Shih if (ret)
177bdfbf06cSPing-Ke Shih return ret;
178bdfbf06cSPing-Ke Shih }
179bdfbf06cSPing-Ke Shih
180e3ec7017SPing-Ke Shih return 0;
181e3ec7017SPing-Ke Shih }
182e3ec7017SPing-Ke Shih
183e3ec7017SPing-Ke Shih #define invalid_efuse_header(hdr1, hdr2) \
184e3ec7017SPing-Ke Shih ((hdr1) == 0xff || (hdr2) == 0xff)
185e3ec7017SPing-Ke Shih #define invalid_efuse_content(word_en, i) \
186e3ec7017SPing-Ke Shih (((word_en) & BIT(i)) != 0x0)
187e3ec7017SPing-Ke Shih #define get_efuse_blk_idx(hdr1, hdr2) \
188e3ec7017SPing-Ke Shih ((((hdr2) & 0xf0) >> 4) | (((hdr1) & 0x0f) << 4))
189e3ec7017SPing-Ke Shih #define block_idx_to_logical_idx(blk_idx, i) \
190e3ec7017SPing-Ke Shih (((blk_idx) << 3) + ((i) << 1))
rtw89_dump_logical_efuse_map(struct rtw89_dev * rtwdev,u8 * phy_map,u8 * log_map)191e3ec7017SPing-Ke Shih static int rtw89_dump_logical_efuse_map(struct rtw89_dev *rtwdev, u8 *phy_map,
192e3ec7017SPing-Ke Shih u8 *log_map)
193e3ec7017SPing-Ke Shih {
194e3ec7017SPing-Ke Shih u32 physical_size = rtwdev->chip->physical_efuse_size;
195e3ec7017SPing-Ke Shih u32 logical_size = rtwdev->chip->logical_efuse_size;
196e3ec7017SPing-Ke Shih u8 sec_ctrl_size = rtwdev->chip->sec_ctrl_efuse_size;
197e3ec7017SPing-Ke Shih u32 phy_idx = sec_ctrl_size;
198e3ec7017SPing-Ke Shih u32 log_idx;
199e3ec7017SPing-Ke Shih u8 hdr1, hdr2;
200e3ec7017SPing-Ke Shih u8 blk_idx;
201e3ec7017SPing-Ke Shih u8 word_en;
202e3ec7017SPing-Ke Shih int i;
203e3ec7017SPing-Ke Shih
204bdfbf06cSPing-Ke Shih if (!phy_map)
205bdfbf06cSPing-Ke Shih return 0;
206bdfbf06cSPing-Ke Shih
207e3ec7017SPing-Ke Shih while (phy_idx < physical_size - sec_ctrl_size) {
208e3ec7017SPing-Ke Shih hdr1 = phy_map[phy_idx];
209e3ec7017SPing-Ke Shih hdr2 = phy_map[phy_idx + 1];
210e3ec7017SPing-Ke Shih if (invalid_efuse_header(hdr1, hdr2))
211e3ec7017SPing-Ke Shih break;
212e3ec7017SPing-Ke Shih
213e3ec7017SPing-Ke Shih blk_idx = get_efuse_blk_idx(hdr1, hdr2);
214e3ec7017SPing-Ke Shih word_en = hdr2 & 0xf;
215e3ec7017SPing-Ke Shih phy_idx += 2;
216e3ec7017SPing-Ke Shih
217e3ec7017SPing-Ke Shih for (i = 0; i < 4; i++) {
218e3ec7017SPing-Ke Shih if (invalid_efuse_content(word_en, i))
219e3ec7017SPing-Ke Shih continue;
220e3ec7017SPing-Ke Shih
221e3ec7017SPing-Ke Shih log_idx = block_idx_to_logical_idx(blk_idx, i);
222e3ec7017SPing-Ke Shih if (phy_idx + 1 > physical_size - sec_ctrl_size - 1 ||
223e3ec7017SPing-Ke Shih log_idx + 1 > logical_size)
224e3ec7017SPing-Ke Shih return -EINVAL;
225e3ec7017SPing-Ke Shih
226e3ec7017SPing-Ke Shih log_map[log_idx] = phy_map[phy_idx];
227e3ec7017SPing-Ke Shih log_map[log_idx + 1] = phy_map[phy_idx + 1];
228e3ec7017SPing-Ke Shih phy_idx += 2;
229e3ec7017SPing-Ke Shih }
230e3ec7017SPing-Ke Shih }
231e3ec7017SPing-Ke Shih return 0;
232e3ec7017SPing-Ke Shih }
233e3ec7017SPing-Ke Shih
rtw89_parse_efuse_map(struct rtw89_dev * rtwdev)234e3ec7017SPing-Ke Shih int rtw89_parse_efuse_map(struct rtw89_dev *rtwdev)
235e3ec7017SPing-Ke Shih {
236e3ec7017SPing-Ke Shih u32 phy_size = rtwdev->chip->physical_efuse_size;
237e3ec7017SPing-Ke Shih u32 log_size = rtwdev->chip->logical_efuse_size;
238bdfbf06cSPing-Ke Shih u32 dav_phy_size = rtwdev->chip->dav_phy_efuse_size;
239bdfbf06cSPing-Ke Shih u32 dav_log_size = rtwdev->chip->dav_log_efuse_size;
240bdfbf06cSPing-Ke Shih u32 full_log_size = log_size + dav_log_size;
241e3ec7017SPing-Ke Shih u8 *phy_map = NULL;
242e3ec7017SPing-Ke Shih u8 *log_map = NULL;
243bdfbf06cSPing-Ke Shih u8 *dav_phy_map = NULL;
244bdfbf06cSPing-Ke Shih u8 *dav_log_map = NULL;
245e3ec7017SPing-Ke Shih int ret;
246e3ec7017SPing-Ke Shih
247e3ec7017SPing-Ke Shih if (rtw89_read16(rtwdev, R_AX_SYS_WL_EFUSE_CTRL) & B_AX_AUTOLOAD_SUS)
248e3ec7017SPing-Ke Shih rtwdev->efuse.valid = true;
249e3ec7017SPing-Ke Shih else
250e3ec7017SPing-Ke Shih rtw89_warn(rtwdev, "failed to check efuse autoload\n");
251e3ec7017SPing-Ke Shih
252e3ec7017SPing-Ke Shih phy_map = kmalloc(phy_size, GFP_KERNEL);
253bdfbf06cSPing-Ke Shih log_map = kmalloc(full_log_size, GFP_KERNEL);
254bdfbf06cSPing-Ke Shih if (dav_phy_size && dav_log_size) {
255bdfbf06cSPing-Ke Shih dav_phy_map = kmalloc(dav_phy_size, GFP_KERNEL);
256bdfbf06cSPing-Ke Shih dav_log_map = log_map + log_size;
257bdfbf06cSPing-Ke Shih }
258e3ec7017SPing-Ke Shih
259bdfbf06cSPing-Ke Shih if (!phy_map || !log_map || (dav_phy_size && !dav_phy_map)) {
260e3ec7017SPing-Ke Shih ret = -ENOMEM;
261e3ec7017SPing-Ke Shih goto out_free;
262e3ec7017SPing-Ke Shih }
263e3ec7017SPing-Ke Shih
264bdfbf06cSPing-Ke Shih ret = rtw89_dump_physical_efuse_map(rtwdev, phy_map, 0, phy_size, false);
265e3ec7017SPing-Ke Shih if (ret) {
266e3ec7017SPing-Ke Shih rtw89_warn(rtwdev, "failed to dump efuse physical map\n");
267e3ec7017SPing-Ke Shih goto out_free;
268e3ec7017SPing-Ke Shih }
269bdfbf06cSPing-Ke Shih ret = rtw89_dump_physical_efuse_map(rtwdev, dav_phy_map, 0, dav_phy_size, true);
270bdfbf06cSPing-Ke Shih if (ret) {
271bdfbf06cSPing-Ke Shih rtw89_warn(rtwdev, "failed to dump efuse dav physical map\n");
272bdfbf06cSPing-Ke Shih goto out_free;
273bdfbf06cSPing-Ke Shih }
274e3ec7017SPing-Ke Shih
275bdfbf06cSPing-Ke Shih memset(log_map, 0xff, full_log_size);
276e3ec7017SPing-Ke Shih ret = rtw89_dump_logical_efuse_map(rtwdev, phy_map, log_map);
277e3ec7017SPing-Ke Shih if (ret) {
278e3ec7017SPing-Ke Shih rtw89_warn(rtwdev, "failed to dump efuse logical map\n");
279e3ec7017SPing-Ke Shih goto out_free;
280e3ec7017SPing-Ke Shih }
281bdfbf06cSPing-Ke Shih ret = rtw89_dump_logical_efuse_map(rtwdev, dav_phy_map, dav_log_map);
282bdfbf06cSPing-Ke Shih if (ret) {
283bdfbf06cSPing-Ke Shih rtw89_warn(rtwdev, "failed to dump efuse dav logical map\n");
284bdfbf06cSPing-Ke Shih goto out_free;
285bdfbf06cSPing-Ke Shih }
286e3ec7017SPing-Ke Shih
287bdfbf06cSPing-Ke Shih rtw89_hex_dump(rtwdev, RTW89_DBG_FW, "log_map: ", log_map, full_log_size);
288e3ec7017SPing-Ke Shih
289e3ec7017SPing-Ke Shih ret = rtwdev->chip->ops->read_efuse(rtwdev, log_map);
290e3ec7017SPing-Ke Shih if (ret) {
291e3ec7017SPing-Ke Shih rtw89_warn(rtwdev, "failed to read efuse map\n");
292e3ec7017SPing-Ke Shih goto out_free;
293e3ec7017SPing-Ke Shih }
294e3ec7017SPing-Ke Shih
295e3ec7017SPing-Ke Shih out_free:
296bdfbf06cSPing-Ke Shih kfree(dav_phy_map);
297e3ec7017SPing-Ke Shih kfree(log_map);
298e3ec7017SPing-Ke Shih kfree(phy_map);
299e3ec7017SPing-Ke Shih
300e3ec7017SPing-Ke Shih return ret;
301e3ec7017SPing-Ke Shih }
302e3ec7017SPing-Ke Shih
rtw89_parse_phycap_map(struct rtw89_dev * rtwdev)303e3ec7017SPing-Ke Shih int rtw89_parse_phycap_map(struct rtw89_dev *rtwdev)
304e3ec7017SPing-Ke Shih {
305e3ec7017SPing-Ke Shih u32 phycap_addr = rtwdev->chip->phycap_addr;
306e3ec7017SPing-Ke Shih u32 phycap_size = rtwdev->chip->phycap_size;
307e3ec7017SPing-Ke Shih u8 *phycap_map = NULL;
308e3ec7017SPing-Ke Shih int ret = 0;
309e3ec7017SPing-Ke Shih
310e3ec7017SPing-Ke Shih if (!phycap_size)
311e3ec7017SPing-Ke Shih return 0;
312e3ec7017SPing-Ke Shih
313e3ec7017SPing-Ke Shih phycap_map = kmalloc(phycap_size, GFP_KERNEL);
314e3ec7017SPing-Ke Shih if (!phycap_map)
315e3ec7017SPing-Ke Shih return -ENOMEM;
316e3ec7017SPing-Ke Shih
317e3ec7017SPing-Ke Shih ret = rtw89_dump_physical_efuse_map(rtwdev, phycap_map,
318bdfbf06cSPing-Ke Shih phycap_addr, phycap_size, false);
319e3ec7017SPing-Ke Shih if (ret) {
320e3ec7017SPing-Ke Shih rtw89_warn(rtwdev, "failed to dump phycap map\n");
321e3ec7017SPing-Ke Shih goto out_free;
322e3ec7017SPing-Ke Shih }
323e3ec7017SPing-Ke Shih
324e3ec7017SPing-Ke Shih ret = rtwdev->chip->ops->read_phycap(rtwdev, phycap_map);
325e3ec7017SPing-Ke Shih if (ret) {
326e3ec7017SPing-Ke Shih rtw89_warn(rtwdev, "failed to read phycap map\n");
327e3ec7017SPing-Ke Shih goto out_free;
328e3ec7017SPing-Ke Shih }
329e3ec7017SPing-Ke Shih
330e3ec7017SPing-Ke Shih out_free:
331e3ec7017SPing-Ke Shih kfree(phycap_map);
332e3ec7017SPing-Ke Shih
333e3ec7017SPing-Ke Shih return ret;
334e3ec7017SPing-Ke Shih }
335*40bb2ab4SPing-Ke Shih
rtw89_read_efuse_ver(struct rtw89_dev * rtwdev,u8 * ecv)336*40bb2ab4SPing-Ke Shih int rtw89_read_efuse_ver(struct rtw89_dev *rtwdev, u8 *ecv)
337*40bb2ab4SPing-Ke Shih {
338*40bb2ab4SPing-Ke Shih int ret;
339*40bb2ab4SPing-Ke Shih u8 val;
340*40bb2ab4SPing-Ke Shih
341*40bb2ab4SPing-Ke Shih ret = rtw89_dump_physical_efuse_map(rtwdev, &val, EF_FV_OFSET, 1, false);
342*40bb2ab4SPing-Ke Shih if (ret)
343*40bb2ab4SPing-Ke Shih return ret;
344*40bb2ab4SPing-Ke Shih
345*40bb2ab4SPing-Ke Shih *ecv = u8_get_bits(val, EF_CV_MASK);
346*40bb2ab4SPing-Ke Shih if (*ecv == EF_CV_INV)
347*40bb2ab4SPing-Ke Shih return -ENOENT;
348*40bb2ab4SPing-Ke Shih
349*40bb2ab4SPing-Ke Shih return 0;
350*40bb2ab4SPing-Ke Shih }
351*40bb2ab4SPing-Ke Shih EXPORT_SYMBOL(rtw89_read_efuse_ver);
352