xref: /openbmc/linux/drivers/net/wireless/realtek/rtw89/efuse.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
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