xref: /OK3568_Linux_fs/external/rkwifibt/tools/rtk_hciattach/rtb_fwc.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  *  Copyright (C) 2018 Realtek Semiconductor Corporation.
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  *  This program is free software; you can redistribute it and/or modify
5*4882a593Smuzhiyun  *  it under the terms of the GNU General Public License as published by
6*4882a593Smuzhiyun  *  the Free Software Foundation; either version 2 of the License, or
7*4882a593Smuzhiyun  *  (at your option) any later version.
8*4882a593Smuzhiyun  *
9*4882a593Smuzhiyun  *  This program is distributed in the hope that it will be useful,
10*4882a593Smuzhiyun  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
11*4882a593Smuzhiyun  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12*4882a593Smuzhiyun  *  GNU General Public License for more details.
13*4882a593Smuzhiyun  */
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun #include <stdio.h>
16*4882a593Smuzhiyun #include <stdlib.h>
17*4882a593Smuzhiyun #include <stdint.h>
18*4882a593Smuzhiyun #include <stdbool.h>
19*4882a593Smuzhiyun #include <string.h>
20*4882a593Smuzhiyun #include <sys/types.h>
21*4882a593Smuzhiyun #include <sys/stat.h>
22*4882a593Smuzhiyun #include <fcntl.h>
23*4882a593Smuzhiyun #include <errno.h>
24*4882a593Smuzhiyun #include <unistd.h>
25*4882a593Smuzhiyun #include <ctype.h>
26*4882a593Smuzhiyun #include <time.h>
27*4882a593Smuzhiyun #include <sys/time.h>
28*4882a593Smuzhiyun #include <sys/ioctl.h>
29*4882a593Smuzhiyun 
30*4882a593Smuzhiyun #ifndef PATH_MAX
31*4882a593Smuzhiyun #define PATH_MAX	1024
32*4882a593Smuzhiyun #endif
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #include "hciattach.h"
35*4882a593Smuzhiyun #include "rtb_fwc.h"
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun //#define BT_ADDR_FROM_VENDOR_STORAGE
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun #define FIRMWARE_DIRECTORY	"/lib/firmware/rtlbt/"
40*4882a593Smuzhiyun #define BT_CONFIG_DIRECTORY	"/lib/firmware/rtlbt/"
41*4882a593Smuzhiyun #define EXTRA_CONFIG_FILE	"/opt/rtk_btconfig.txt"
42*4882a593Smuzhiyun #define BT_ADDR_FILE		"/opt/bdaddr"
43*4882a593Smuzhiyun #define BDADDR_STRING_LEN	17
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun struct list_head {
46*4882a593Smuzhiyun 	struct list_head *next, *prev;
47*4882a593Smuzhiyun };
48*4882a593Smuzhiyun 
49*4882a593Smuzhiyun struct cfg_list_item {
50*4882a593Smuzhiyun 	struct list_head list;
51*4882a593Smuzhiyun 	uint16_t offset;
52*4882a593Smuzhiyun 	uint8_t len;
53*4882a593Smuzhiyun 	uint8_t data[0];
54*4882a593Smuzhiyun };
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun const uint8_t cfg_magic[4] = { 0x55, 0xab, 0x23, 0x87 };
57*4882a593Smuzhiyun static struct list_head list_configs;
58*4882a593Smuzhiyun static struct list_head list_extracfgs;
59*4882a593Smuzhiyun 
60*4882a593Smuzhiyun struct rtb_cfg_item {
61*4882a593Smuzhiyun 	uint16_t offset;
62*4882a593Smuzhiyun 	uint8_t len;
63*4882a593Smuzhiyun 	uint8_t data[0];
64*4882a593Smuzhiyun } __attribute__ ((packed));
65*4882a593Smuzhiyun 
66*4882a593Smuzhiyun #define RTB_CFG_HDR_LEN		6
67*4882a593Smuzhiyun 
68*4882a593Smuzhiyun struct rtb_patch_entry {
69*4882a593Smuzhiyun 	uint16_t chip_id;
70*4882a593Smuzhiyun 	uint16_t patch_len;
71*4882a593Smuzhiyun 	uint32_t soffset;
72*4882a593Smuzhiyun 	uint32_t svn_ver;
73*4882a593Smuzhiyun 	uint32_t coex_ver;
74*4882a593Smuzhiyun } __attribute__ ((packed));
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun struct rtb_patch_hdr {
77*4882a593Smuzhiyun 	uint8_t signature[8];
78*4882a593Smuzhiyun 	uint32_t fw_version;
79*4882a593Smuzhiyun 	uint16_t number_of_patch;
80*4882a593Smuzhiyun 	struct rtb_patch_entry entry[0];
81*4882a593Smuzhiyun } __attribute__ ((packed));
82*4882a593Smuzhiyun 
83*4882a593Smuzhiyun uint16_t project_id[]=
84*4882a593Smuzhiyun {
85*4882a593Smuzhiyun 	ROM_LMP_8723a,
86*4882a593Smuzhiyun 	ROM_LMP_8723b, /* RTL8723BS */
87*4882a593Smuzhiyun 	ROM_LMP_8821a, /* RTL8821AS */
88*4882a593Smuzhiyun 	ROM_LMP_8761a, /* RTL8761ATV */
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun 	ROM_LMP_8703a,
91*4882a593Smuzhiyun 	ROM_LMP_8763a,
92*4882a593Smuzhiyun 	ROM_LMP_8703b,
93*4882a593Smuzhiyun 	ROM_LMP_8723c, /* index 7 for 8723CS. What is for other 8723CS  */
94*4882a593Smuzhiyun 	ROM_LMP_8822b, /* RTL8822BS */
95*4882a593Smuzhiyun 	ROM_LMP_8723b, /* RTL8723DS */
96*4882a593Smuzhiyun 	ROM_LMP_8821a, /* id 10 for RTL8821CS, lmp subver 0x8821 */
97*4882a593Smuzhiyun 	ROM_LMP_NONE,
98*4882a593Smuzhiyun 	ROM_LMP_NONE,
99*4882a593Smuzhiyun 	ROM_LMP_8822c, /* id 13 for RTL8822CS, lmp subver 0x8822 */
100*4882a593Smuzhiyun 	ROM_LMP_8761a, /* id 14 for 8761B */
101*4882a593Smuzhiyun 	ROM_LMP_NONE,
102*4882a593Smuzhiyun 	ROM_LMP_NONE,
103*4882a593Smuzhiyun 	ROM_LMP_NONE,
104*4882a593Smuzhiyun 	ROM_LMP_8852a, /* id 18 for 8852AS */
105*4882a593Smuzhiyun 	ROM_LMP_8723b, /* id 19 for 8723FS */
106*4882a593Smuzhiyun 	ROM_LMP_8852a, /* id 20 for 8852BS */
107*4882a593Smuzhiyun 	ROM_LMP_NONE,
108*4882a593Smuzhiyun 	ROM_LMP_NONE,
109*4882a593Smuzhiyun 	ROM_LMP_NONE,
110*4882a593Smuzhiyun 	ROM_LMP_NONE,
111*4882a593Smuzhiyun 	ROM_LMP_8852a, /* id 25 for 8852CS */
112*4882a593Smuzhiyun 	ROM_LMP_NONE,
113*4882a593Smuzhiyun 	ROM_LMP_NONE,
114*4882a593Smuzhiyun 	ROM_LMP_NONE,
115*4882a593Smuzhiyun 	ROM_LMP_NONE,
116*4882a593Smuzhiyun 	ROM_LMP_NONE,
117*4882a593Smuzhiyun 	ROM_LMP_NONE,
118*4882a593Smuzhiyun 	ROM_LMP_NONE,
119*4882a593Smuzhiyun 	ROM_LMP_NONE,
120*4882a593Smuzhiyun 	ROM_LMP_8852a, /* id 34 for 8852BP */
121*4882a593Smuzhiyun };
122*4882a593Smuzhiyun 
123*4882a593Smuzhiyun static struct patch_info h4_patch_table[] = {
124*4882a593Smuzhiyun 	/* match flags, chip type, lmp subver, proj id(unused), hci_ver,
125*4882a593Smuzhiyun 	 * hci_rev, ...
126*4882a593Smuzhiyun 	 */
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun 	/* RTL8761AT */
129*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8761AT,
130*4882a593Smuzhiyun 		0x8761, 0xffff, 0, 0x000a,
131*4882a593Smuzhiyun 		"rtl8761at_fw", "rtl8761at_config", "RTL8761AT" },
132*4882a593Smuzhiyun 	/* RTL8761ATF */
133*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8761ATF,
134*4882a593Smuzhiyun 		0x8761, 0xffff, 0, 0x000a,
135*4882a593Smuzhiyun 		"rtl8761atf_fw", "rtl8761atf_config", "RTL8761ATF" },
136*4882a593Smuzhiyun 	/* RTL8761B(8763) H4 Test Chip without download
137*4882a593Smuzhiyun 	 * FW/Config is not used.
138*4882a593Smuzhiyun 	 */
139*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8761BTC,
140*4882a593Smuzhiyun 		0x8763, 0xffff, 0, 0x000b,
141*4882a593Smuzhiyun 		"rtl8761btc_fw", "rtl8761btc_config", "RTL8761BTC" },
142*4882a593Smuzhiyun 	/* RTL8761B H4 Test Chip wihtout download*/
143*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8761BH4,
144*4882a593Smuzhiyun 		0x8761, 0xffff, 0, 0x000b,
145*4882a593Smuzhiyun 		"rtl8761bh4_fw", "rtl8761bh4_config", "RTL8761BH4" },
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun 	/* RTL8723DS */
148*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8723DS,
149*4882a593Smuzhiyun 		ROM_LMP_8723b, ROM_LMP_8723b, 8, 0x000d,
150*4882a593Smuzhiyun 		"rtl8723dsh4_fw", "rtl8723dsh4_config", "RTL8723DSH4"},
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun 	{ 0, 0, 0, ROM_LMP_NONE, 0, 0, "rtl_none_fw", "rtl_none_config", "NONE"}
153*4882a593Smuzhiyun };
154*4882a593Smuzhiyun 
155*4882a593Smuzhiyun static struct patch_info patch_table[] = {
156*4882a593Smuzhiyun 	/* match flags, chip type, lmp subver, proj id(unused), hci_ver,
157*4882a593Smuzhiyun 	 * hci_rev, ...
158*4882a593Smuzhiyun 	 */
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun 	/* RTL8723AS */
161*4882a593Smuzhiyun 	{ 0, 0, ROM_LMP_8723a, ROM_LMP_8723a, 0, 0,
162*4882a593Smuzhiyun 		"rtl8723a_fw", "rtl8723a_config", "RTL8723AS"},
163*4882a593Smuzhiyun 	/* RTL8821CS */
164*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_REV, CHIP_8821CS,
165*4882a593Smuzhiyun 		ROM_LMP_8821a, ROM_LMP_8821a, 0, 0x000c,
166*4882a593Smuzhiyun 		"rtl8821c_fw", "rtl8821c_config", "RTL8821CS"},
167*4882a593Smuzhiyun 	/* RTL8821AS */
168*4882a593Smuzhiyun 	{ 0, 0, ROM_LMP_8821a, ROM_LMP_8821a, 0, 0,
169*4882a593Smuzhiyun 		"rtl8821a_fw", "rtl8821a_config", "RTL8821AS"},
170*4882a593Smuzhiyun 	/* RTL8761ATV */
171*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, 0,
172*4882a593Smuzhiyun 		ROM_LMP_8761a, ROM_LMP_8761a, 0x06, 0x000a,
173*4882a593Smuzhiyun 		"rtl8761a_fw", "rtl8761a_config", "RTL8761ATV"},
174*4882a593Smuzhiyun 	/* RTL8725AS */
175*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE | RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV,
176*4882a593Smuzhiyun 		CHIP_8725AS,
177*4882a593Smuzhiyun 		ROM_LMP_8761a, ROM_LMP_8761a, 0x0a, 0x000b,
178*4882a593Smuzhiyun 		"rtl8725as_fw", "rtl8725as_config", "RTL8725AS"},
179*4882a593Smuzhiyun 	/* RTL8761BTV */
180*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8761B,
181*4882a593Smuzhiyun 		ROM_LMP_8761a, ROM_LMP_8761a, 0x0a, 0x000b,
182*4882a593Smuzhiyun 		"rtl8761b_fw", "rtl8761b_config", "RTL8761BTV"},
183*4882a593Smuzhiyun 
184*4882a593Smuzhiyun 	/* RTL8703AS
185*4882a593Smuzhiyun 	 * RTL8822BS
186*4882a593Smuzhiyun 	 * */
187*4882a593Smuzhiyun #ifdef RTL_8703A_SUPPORT
188*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8703AS,
189*4882a593Smuzhiyun 		ROM_LMP_8723b, ROM_LMP_8723b, 0, 0,
190*4882a593Smuzhiyun 		"rtl8703a_fw", "rtl8703a_config", "RTL8703AS"},
191*4882a593Smuzhiyun #endif
192*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_REV, CHIP_8822BS,
193*4882a593Smuzhiyun 		ROM_LMP_8822b, ROM_LMP_8822b, 0, 0x000b,
194*4882a593Smuzhiyun 		"rtl8822b_fw", "rtl8822b_config", "RTL8822BS"},
195*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_REV, CHIP_8822CS,
196*4882a593Smuzhiyun 		ROM_LMP_8822c, ROM_LMP_8822c, 0, 0x000c,
197*4882a593Smuzhiyun 		"rtl8822cs_fw", "rtl8822cs_config", "RTL8822CS"},
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun 	/* RTL8852AS */
200*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_REV, CHIP_8852AS,
201*4882a593Smuzhiyun 		ROM_LMP_8852a, ROM_LMP_8852a, 11, 0x000a,
202*4882a593Smuzhiyun 		"rtl8852as_fw", "rtl8852as_config", "RTL8852AS" },
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun 	/* RTL8852BS */
205*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE | RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8852BS,
206*4882a593Smuzhiyun 		ROM_LMP_8852a, ROM_LMP_8852a, 11, 0x000b,
207*4882a593Smuzhiyun 		"rtl8852bs_fw", "rtl8852bs_config", "RTL8852BS" },
208*4882a593Smuzhiyun 
209*4882a593Smuzhiyun 	/* RTL8852BP */
210*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE | RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8852BP,
211*4882a593Smuzhiyun 		ROM_LMP_8852a, ROM_LMP_8852a, 11, 0x000b,
212*4882a593Smuzhiyun 		"rtl8852bps_fw", "rtl8852bps_config", "RTL8852BP" },
213*4882a593Smuzhiyun 
214*4882a593Smuzhiyun 	/* RTL8852CS */
215*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE | RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8852CS,
216*4882a593Smuzhiyun 		ROM_LMP_8852a, ROM_LMP_8852a, 11, 0x000c,
217*4882a593Smuzhiyun 		"rtl8852cs_fw", "rtl8852cs_config", "RTL8852CS" },
218*4882a593Smuzhiyun 
219*4882a593Smuzhiyun 	/* RTL8703BS
220*4882a593Smuzhiyun 	 * RTL8723CS_XX
221*4882a593Smuzhiyun 	 * RTL8723CS_CG
222*4882a593Smuzhiyun 	 * RTL8723CS_VF
223*4882a593Smuzhiyun 	 * Use the sampe lmp subversion 0x8703
224*4882a593Smuzhiyun 	 * */
225*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8703BS,
226*4882a593Smuzhiyun 		ROM_LMP_8703b, ROM_LMP_8703b, 0, 0,
227*4882a593Smuzhiyun 		"rtl8703b_fw", "rtl8703b_config", "RTL8703BS"},
228*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8723CS_XX,
229*4882a593Smuzhiyun 		ROM_LMP_8703b, ROM_LMP_8723cs_xx, 0, 0,
230*4882a593Smuzhiyun 		"rtl8723cs_fw", "rtl8723cs_config", "RTL8723CS_XX"},
231*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8723CS_CG,
232*4882a593Smuzhiyun 		ROM_LMP_8703b, ROM_LMP_8723cs_cg, 0, 0,
233*4882a593Smuzhiyun 		"rtl8723cs_fw", "rtl8723cs_config", "RTL8723CS_CG"},
234*4882a593Smuzhiyun 	{ RTL_FW_MATCH_CHIP_TYPE, CHIP_8723CS_VF,
235*4882a593Smuzhiyun 		ROM_LMP_8703b, ROM_LMP_8723cs_vf, 0, 0,
236*4882a593Smuzhiyun 		"rtl8723cs_fw", "rtl8723cs_config", "RTL8723CS_VF"},
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun 	/* RTL8723BS */
239*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, 0,
240*4882a593Smuzhiyun 		ROM_LMP_8723b, ROM_LMP_8723b, 6, 0x000b,
241*4882a593Smuzhiyun 		"rtl8723b_fw", "rtl8723b_config", "RTL8723BS"},
242*4882a593Smuzhiyun 	/* RTL8723DS */
243*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8723DS,
244*4882a593Smuzhiyun 		ROM_LMP_8723b, ROM_LMP_8723b, 8, 0x000d,
245*4882a593Smuzhiyun 		"rtl8723d_fw", "rtl8723d_config", "RTL8723DS"},
246*4882a593Smuzhiyun 	/* RTL8723FS */
247*4882a593Smuzhiyun 	{ RTL_FW_MATCH_HCI_VER | RTL_FW_MATCH_HCI_REV, CHIP_8723FS,
248*4882a593Smuzhiyun 		ROM_LMP_8723b, ROM_LMP_8723b, 11, 0x000f,
249*4882a593Smuzhiyun 		"rtl8723fs_fw", "rtl8723fs_config", "RTL8723FS"},
250*4882a593Smuzhiyun 	/* add entries here*/
251*4882a593Smuzhiyun 
252*4882a593Smuzhiyun 	{ 0, 0, 0, ROM_LMP_NONE, 0, 0, "rtl_none_fw", "rtl_none_config", "NONE"}
253*4882a593Smuzhiyun };
254*4882a593Smuzhiyun 
get_unaligned_le16(uint8_t * p)255*4882a593Smuzhiyun static __inline uint16_t get_unaligned_le16(uint8_t * p)
256*4882a593Smuzhiyun {
257*4882a593Smuzhiyun 	return (uint16_t) (*p) + ((uint16_t) (*(p + 1)) << 8);
258*4882a593Smuzhiyun }
259*4882a593Smuzhiyun 
get_unaligned_le32(uint8_t * p)260*4882a593Smuzhiyun static __inline uint32_t get_unaligned_le32(uint8_t * p)
261*4882a593Smuzhiyun {
262*4882a593Smuzhiyun 	return (uint32_t) (*p) + ((uint32_t) (*(p + 1)) << 8) +
263*4882a593Smuzhiyun 	    ((uint32_t) (*(p + 2)) << 16) + ((uint32_t) (*(p + 3)) << 24);
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun 
266*4882a593Smuzhiyun /* list head from kernel */
267*4882a593Smuzhiyun #define offsetof(TYPE, MEMBER)	((size_t)&((TYPE *)0)->MEMBER)
268*4882a593Smuzhiyun 
269*4882a593Smuzhiyun #define container_of(ptr, type, member) ({                      \
270*4882a593Smuzhiyun 	const typeof( ((type *)0)->member ) *__mptr = (ptr);    \
271*4882a593Smuzhiyun 	(type *)( (char *)__mptr - offsetof(type,member) );})
272*4882a593Smuzhiyun 
273*4882a593Smuzhiyun #define list_entry(ptr, type, member) \
274*4882a593Smuzhiyun 	container_of(ptr, type, member)
275*4882a593Smuzhiyun 
276*4882a593Smuzhiyun #define list_for_each_safe(pos, n, head) \
277*4882a593Smuzhiyun 	for (pos = (head)->next, n = pos->next; pos != (head); \
278*4882a593Smuzhiyun 		pos = n, n = pos->next)
279*4882a593Smuzhiyun 
INIT_LIST_HEAD(struct list_head * list)280*4882a593Smuzhiyun static inline void INIT_LIST_HEAD(struct list_head *list)
281*4882a593Smuzhiyun {
282*4882a593Smuzhiyun 	list->next = list;
283*4882a593Smuzhiyun 	list->prev = list;
284*4882a593Smuzhiyun }
285*4882a593Smuzhiyun 
list_empty(const struct list_head * head)286*4882a593Smuzhiyun static inline int list_empty(const struct list_head *head)
287*4882a593Smuzhiyun {
288*4882a593Smuzhiyun 	return head->next == head;
289*4882a593Smuzhiyun }
290*4882a593Smuzhiyun 
__list_add(struct list_head * _new,struct list_head * prev,struct list_head * next)291*4882a593Smuzhiyun static inline void __list_add(struct list_head *_new,
292*4882a593Smuzhiyun 			      struct list_head *prev,
293*4882a593Smuzhiyun 			      struct list_head *next)
294*4882a593Smuzhiyun {
295*4882a593Smuzhiyun 	next->prev = _new;
296*4882a593Smuzhiyun 	_new->next = next;
297*4882a593Smuzhiyun 	_new->prev = prev;
298*4882a593Smuzhiyun 	prev->next = _new;
299*4882a593Smuzhiyun }
300*4882a593Smuzhiyun 
list_add_tail(struct list_head * _new,struct list_head * head)301*4882a593Smuzhiyun static inline void list_add_tail(struct list_head *_new, struct list_head *head)
302*4882a593Smuzhiyun {
303*4882a593Smuzhiyun 	__list_add(_new, head->prev, head);
304*4882a593Smuzhiyun }
305*4882a593Smuzhiyun 
__list_del(struct list_head * prev,struct list_head * next)306*4882a593Smuzhiyun static inline void __list_del(struct list_head *prev, struct list_head *next)
307*4882a593Smuzhiyun {
308*4882a593Smuzhiyun 	next->prev = prev;
309*4882a593Smuzhiyun 	prev->next = next;
310*4882a593Smuzhiyun }
311*4882a593Smuzhiyun 
312*4882a593Smuzhiyun #define LIST_POISON1  ((void *) 0x00100100)
313*4882a593Smuzhiyun #define LIST_POISON2  ((void *) 0x00200200)
list_del(struct list_head * entry)314*4882a593Smuzhiyun static inline void list_del(struct list_head *entry)
315*4882a593Smuzhiyun {
316*4882a593Smuzhiyun 	__list_del(entry->prev, entry->next);
317*4882a593Smuzhiyun 	entry->next = (struct list_head*)LIST_POISON1;
318*4882a593Smuzhiyun 	entry->prev = (struct list_head*)LIST_POISON2;
319*4882a593Smuzhiyun }
320*4882a593Smuzhiyun 
__list_splice(const struct list_head * list,struct list_head * prev,struct list_head * next)321*4882a593Smuzhiyun static inline void __list_splice(const struct list_head *list,
322*4882a593Smuzhiyun 				 struct list_head *prev,
323*4882a593Smuzhiyun 				 struct list_head *next)
324*4882a593Smuzhiyun {
325*4882a593Smuzhiyun 	struct list_head *first = list->next;
326*4882a593Smuzhiyun 	struct list_head *last = list->prev;
327*4882a593Smuzhiyun 
328*4882a593Smuzhiyun 	first->prev = prev;
329*4882a593Smuzhiyun 	prev->next = first;
330*4882a593Smuzhiyun 
331*4882a593Smuzhiyun 	last->next = next;
332*4882a593Smuzhiyun 	next->prev = last;
333*4882a593Smuzhiyun }
334*4882a593Smuzhiyun 
list_splice_tail(struct list_head * list,struct list_head * head)335*4882a593Smuzhiyun static inline void list_splice_tail(struct list_head *list,
336*4882a593Smuzhiyun 				struct list_head *head)
337*4882a593Smuzhiyun {
338*4882a593Smuzhiyun 	if (!list_empty(list))
339*4882a593Smuzhiyun 		__list_splice(list, head->prev, head);
340*4882a593Smuzhiyun }
341*4882a593Smuzhiyun 
list_replace(struct list_head * old,struct list_head * new)342*4882a593Smuzhiyun static inline void list_replace(struct list_head *old,
343*4882a593Smuzhiyun 				struct list_head *new)
344*4882a593Smuzhiyun {
345*4882a593Smuzhiyun 	new->next = old->next;
346*4882a593Smuzhiyun 	new->next->prev = new;
347*4882a593Smuzhiyun 	new->prev = old->prev;
348*4882a593Smuzhiyun 	new->prev->next = new;
349*4882a593Smuzhiyun }
350*4882a593Smuzhiyun 
list_replace_init(struct list_head * old,struct list_head * new)351*4882a593Smuzhiyun static inline void list_replace_init(struct list_head *old,
352*4882a593Smuzhiyun 					struct list_head *new)
353*4882a593Smuzhiyun {
354*4882a593Smuzhiyun 	list_replace(old, new);
355*4882a593Smuzhiyun 	INIT_LIST_HEAD(old);
356*4882a593Smuzhiyun }
357*4882a593Smuzhiyun 
config_lists_init(void)358*4882a593Smuzhiyun static int config_lists_init(void)
359*4882a593Smuzhiyun {
360*4882a593Smuzhiyun 	INIT_LIST_HEAD(&list_configs);
361*4882a593Smuzhiyun 	INIT_LIST_HEAD(&list_extracfgs);
362*4882a593Smuzhiyun 
363*4882a593Smuzhiyun 	return 0;
364*4882a593Smuzhiyun }
365*4882a593Smuzhiyun 
config_lists_free(void)366*4882a593Smuzhiyun static void config_lists_free(void)
367*4882a593Smuzhiyun {
368*4882a593Smuzhiyun 	struct list_head *iter;
369*4882a593Smuzhiyun 	struct list_head *tmp;
370*4882a593Smuzhiyun 	struct list_head *head;
371*4882a593Smuzhiyun 	struct cfg_list_item *n;
372*4882a593Smuzhiyun 
373*4882a593Smuzhiyun 	if (!list_empty(&list_extracfgs))
374*4882a593Smuzhiyun 		list_splice_tail(&list_extracfgs, &list_configs);
375*4882a593Smuzhiyun 	head = &list_configs;
376*4882a593Smuzhiyun 	list_for_each_safe(iter, tmp, head) {
377*4882a593Smuzhiyun 		n = list_entry(iter, struct cfg_list_item, list);
378*4882a593Smuzhiyun 		if (n) {
379*4882a593Smuzhiyun 			list_del(&n->list);
380*4882a593Smuzhiyun 			free(n);
381*4882a593Smuzhiyun 		}
382*4882a593Smuzhiyun 	}
383*4882a593Smuzhiyun 
384*4882a593Smuzhiyun 	INIT_LIST_HEAD(&list_configs);
385*4882a593Smuzhiyun 	INIT_LIST_HEAD(&list_extracfgs);
386*4882a593Smuzhiyun }
387*4882a593Smuzhiyun 
line_process(char * buf,int len)388*4882a593Smuzhiyun static void line_process(char *buf, int len /*@unused@*/)
389*4882a593Smuzhiyun {
390*4882a593Smuzhiyun 	char *argv[32];
391*4882a593Smuzhiyun 	int argc = 0;
392*4882a593Smuzhiyun 	char *ptr = buf;
393*4882a593Smuzhiyun 	char *head = buf;
394*4882a593Smuzhiyun 	unsigned long int offset;
395*4882a593Smuzhiyun 	uint8_t l;
396*4882a593Smuzhiyun 	uint8_t i = 0;
397*4882a593Smuzhiyun 	struct cfg_list_item *item;
398*4882a593Smuzhiyun 
399*4882a593Smuzhiyun 	RS_INFO("%s", buf);
400*4882a593Smuzhiyun 
401*4882a593Smuzhiyun 	while ((ptr = strsep(&head, ", \t")) != NULL) {
402*4882a593Smuzhiyun 		if (!ptr[0])
403*4882a593Smuzhiyun 			continue;
404*4882a593Smuzhiyun 		argv[argc++] = ptr;
405*4882a593Smuzhiyun 		if (argc >= 32) {
406*4882a593Smuzhiyun 			RS_WARN("%s: Config item is too long", __func__);
407*4882a593Smuzhiyun 			break;
408*4882a593Smuzhiyun 		}
409*4882a593Smuzhiyun 	}
410*4882a593Smuzhiyun 
411*4882a593Smuzhiyun 	if (argc < 4) {
412*4882a593Smuzhiyun 		RS_WARN("%s: Invalid Config item, ignore", __func__);
413*4882a593Smuzhiyun 		return;
414*4882a593Smuzhiyun 	}
415*4882a593Smuzhiyun 
416*4882a593Smuzhiyun 	offset = strtoul(argv[0], NULL, 16);
417*4882a593Smuzhiyun 	offset = offset | (strtoul(argv[1], NULL, 16) << 8);
418*4882a593Smuzhiyun 	l = (uint8_t)strtoul(argv[2], NULL, 16);
419*4882a593Smuzhiyun 	if (l != (uint8_t)(argc - 3)) {
420*4882a593Smuzhiyun 		RS_ERR("Invalid Config item len %u", l);
421*4882a593Smuzhiyun 		return;
422*4882a593Smuzhiyun 	}
423*4882a593Smuzhiyun 
424*4882a593Smuzhiyun 	item = malloc(sizeof(*item) + l);
425*4882a593Smuzhiyun 	if (!item) {
426*4882a593Smuzhiyun 		RS_WARN("%s: Cannot alloc mem for item, %04lx, %u", __func__,
427*4882a593Smuzhiyun 			offset, l);
428*4882a593Smuzhiyun 		return;
429*4882a593Smuzhiyun 	}
430*4882a593Smuzhiyun 	memset(item, 0, sizeof(*item));
431*4882a593Smuzhiyun 	item->offset = (uint16_t)offset;
432*4882a593Smuzhiyun 	item->len = l;
433*4882a593Smuzhiyun 	for (i = 0; i < l; i++)
434*4882a593Smuzhiyun 		item->data[i] = (uint8_t)strtoul(argv[3 + i], NULL, 16);
435*4882a593Smuzhiyun 	list_add_tail(&item->list, &list_extracfgs);
436*4882a593Smuzhiyun }
437*4882a593Smuzhiyun 
config_process(uint8_t * buf,int len)438*4882a593Smuzhiyun static void config_process(uint8_t *buf, int len /*@unused@*/)
439*4882a593Smuzhiyun {
440*4882a593Smuzhiyun 	char *head = (void *)buf;
441*4882a593Smuzhiyun 	char *ptr = (void *)buf;
442*4882a593Smuzhiyun 
443*4882a593Smuzhiyun 	while ((ptr = strsep(&head, "\n\r")) != NULL) {
444*4882a593Smuzhiyun 		if (!ptr[0])
445*4882a593Smuzhiyun 			continue;
446*4882a593Smuzhiyun 		line_process(ptr, strlen(ptr) + 1);
447*4882a593Smuzhiyun 	}
448*4882a593Smuzhiyun }
449*4882a593Smuzhiyun 
parse_extra_config(const char * path)450*4882a593Smuzhiyun static void parse_extra_config(const char *path)
451*4882a593Smuzhiyun {
452*4882a593Smuzhiyun 	int fd;
453*4882a593Smuzhiyun 	uint8_t buf[256];
454*4882a593Smuzhiyun 	int result;
455*4882a593Smuzhiyun 
456*4882a593Smuzhiyun 	fd = open(path, O_RDONLY);
457*4882a593Smuzhiyun 	if (fd == -1) {
458*4882a593Smuzhiyun 		RS_INFO("Couldnt open extra config %s, %s", path,
459*4882a593Smuzhiyun 			strerror(errno));
460*4882a593Smuzhiyun 		return;
461*4882a593Smuzhiyun 	}
462*4882a593Smuzhiyun 
463*4882a593Smuzhiyun 	result = read(fd, buf, sizeof(buf));
464*4882a593Smuzhiyun 	if (result == -1) {
465*4882a593Smuzhiyun 		RS_ERR("Couldnt read %s, %s", path, strerror(errno));
466*4882a593Smuzhiyun 		goto done;
467*4882a593Smuzhiyun 	} else if (result == 0) {
468*4882a593Smuzhiyun 		RS_ERR("File is empty");
469*4882a593Smuzhiyun 		goto done;
470*4882a593Smuzhiyun 	}
471*4882a593Smuzhiyun 
472*4882a593Smuzhiyun 	if (result > 254) {
473*4882a593Smuzhiyun 		RS_ERR("Extra Config file is too big");
474*4882a593Smuzhiyun 		goto done;
475*4882a593Smuzhiyun 	}
476*4882a593Smuzhiyun 	buf[result++] = '\n';
477*4882a593Smuzhiyun 	buf[result++] = '\0';
478*4882a593Smuzhiyun 
479*4882a593Smuzhiyun 	config_process(buf, result);
480*4882a593Smuzhiyun 
481*4882a593Smuzhiyun done:
482*4882a593Smuzhiyun 	close(fd);
483*4882a593Smuzhiyun }
484*4882a593Smuzhiyun 
485*4882a593Smuzhiyun /* Get the entry from patch_table according to LMP subversion */
get_patch_entry(struct rtb_struct * btrtl)486*4882a593Smuzhiyun struct patch_info *get_patch_entry(struct rtb_struct *btrtl)
487*4882a593Smuzhiyun {
488*4882a593Smuzhiyun 	struct patch_info *n = NULL;
489*4882a593Smuzhiyun 
490*4882a593Smuzhiyun 	if (btrtl->proto == HCI_UART_3WIRE)
491*4882a593Smuzhiyun 		n = patch_table;
492*4882a593Smuzhiyun 	else
493*4882a593Smuzhiyun 		n = h4_patch_table;
494*4882a593Smuzhiyun 	for (; n->lmp_subver; n++) {
495*4882a593Smuzhiyun 		if ((n->match_flags & RTL_FW_MATCH_CHIP_TYPE) &&
496*4882a593Smuzhiyun 		    n->chip_type != btrtl->chip_type)
497*4882a593Smuzhiyun 			continue;
498*4882a593Smuzhiyun 		if ((n->match_flags & RTL_FW_MATCH_HCI_VER) &&
499*4882a593Smuzhiyun 		    n->hci_ver != btrtl->hci_ver)
500*4882a593Smuzhiyun 			continue;
501*4882a593Smuzhiyun 		if ((n->match_flags & RTL_FW_MATCH_HCI_REV) &&
502*4882a593Smuzhiyun 		    n->hci_rev != btrtl->hci_rev)
503*4882a593Smuzhiyun 			continue;
504*4882a593Smuzhiyun 		if (n->lmp_subver != btrtl->lmp_subver)
505*4882a593Smuzhiyun 			continue;
506*4882a593Smuzhiyun 
507*4882a593Smuzhiyun 		break;
508*4882a593Smuzhiyun 	}
509*4882a593Smuzhiyun 
510*4882a593Smuzhiyun 	return n;
511*4882a593Smuzhiyun }
512*4882a593Smuzhiyun 
is_mac(uint8_t chip_type,uint16_t offset)513*4882a593Smuzhiyun static int is_mac(uint8_t chip_type, uint16_t offset)
514*4882a593Smuzhiyun {
515*4882a593Smuzhiyun 	int result = 0;
516*4882a593Smuzhiyun 
517*4882a593Smuzhiyun 	switch (chip_type) {
518*4882a593Smuzhiyun 	case CHIP_8822BS:
519*4882a593Smuzhiyun 	case CHIP_8723DS:
520*4882a593Smuzhiyun 	case CHIP_8821CS:
521*4882a593Smuzhiyun 	case CHIP_8723CS_XX:
522*4882a593Smuzhiyun 	case CHIP_8723CS_CG:
523*4882a593Smuzhiyun 	case CHIP_8723CS_VF:
524*4882a593Smuzhiyun 		if (offset == 0x0044)
525*4882a593Smuzhiyun 			return 1;
526*4882a593Smuzhiyun 		break;
527*4882a593Smuzhiyun 	case CHIP_8822CS:
528*4882a593Smuzhiyun 	case CHIP_8761B:
529*4882a593Smuzhiyun 	case CHIP_8852AS:
530*4882a593Smuzhiyun 	case CHIP_8723FS:
531*4882a593Smuzhiyun 	case CHIP_8852BS:
532*4882a593Smuzhiyun 	case CHIP_8852CS:
533*4882a593Smuzhiyun 		if (offset == 0x0030)
534*4882a593Smuzhiyun 			return 1;
535*4882a593Smuzhiyun 		break;
536*4882a593Smuzhiyun //	case 0: /* special for not setting chip_type */
537*4882a593Smuzhiyun 	case CHIP_8761AT:
538*4882a593Smuzhiyun 	case CHIP_8761ATF:
539*4882a593Smuzhiyun 	case CHIP_8761BTC:
540*4882a593Smuzhiyun 	case CHIP_8723BS:
541*4882a593Smuzhiyun 		if (offset == 0x003c)
542*4882a593Smuzhiyun 			return 1;
543*4882a593Smuzhiyun 		break;
544*4882a593Smuzhiyun 	default:
545*4882a593Smuzhiyun 		break;
546*4882a593Smuzhiyun 	}
547*4882a593Smuzhiyun 
548*4882a593Smuzhiyun 	return result;
549*4882a593Smuzhiyun }
550*4882a593Smuzhiyun 
get_mac_offset(uint8_t chip_type)551*4882a593Smuzhiyun static uint16_t get_mac_offset(uint8_t chip_type)
552*4882a593Smuzhiyun {
553*4882a593Smuzhiyun 	switch (chip_type) {
554*4882a593Smuzhiyun 	case CHIP_8822BS:
555*4882a593Smuzhiyun 	case CHIP_8723DS:
556*4882a593Smuzhiyun 	case CHIP_8821CS:
557*4882a593Smuzhiyun 		return 0x0044;
558*4882a593Smuzhiyun 	case CHIP_8822CS:
559*4882a593Smuzhiyun 	case CHIP_8761B:
560*4882a593Smuzhiyun 	case CHIP_8852AS:
561*4882a593Smuzhiyun 	case CHIP_8723FS:
562*4882a593Smuzhiyun 	case CHIP_8852BS:
563*4882a593Smuzhiyun 	case CHIP_8852CS:
564*4882a593Smuzhiyun 		return 0x0030;
565*4882a593Smuzhiyun //	case 0: /* special for not setting chip_type */
566*4882a593Smuzhiyun 	case CHIP_8761AT:
567*4882a593Smuzhiyun 	case CHIP_8761ATF:
568*4882a593Smuzhiyun 	case CHIP_8761BTC:
569*4882a593Smuzhiyun 	case CHIP_8723BS:
570*4882a593Smuzhiyun 		return 0x003c;
571*4882a593Smuzhiyun 	default:
572*4882a593Smuzhiyun 		return 0x003c;
573*4882a593Smuzhiyun 	}
574*4882a593Smuzhiyun }
575*4882a593Smuzhiyun 
merge_configs(struct list_head * head,struct list_head * head2)576*4882a593Smuzhiyun static void merge_configs(struct list_head *head, struct list_head *head2)
577*4882a593Smuzhiyun {
578*4882a593Smuzhiyun 	struct list_head *epos, *enext;
579*4882a593Smuzhiyun 	struct list_head *pos, *next;
580*4882a593Smuzhiyun 	struct cfg_list_item *n;
581*4882a593Smuzhiyun 	struct cfg_list_item *extra;
582*4882a593Smuzhiyun 
583*4882a593Smuzhiyun 	if (!head || !head2)
584*4882a593Smuzhiyun 		return;
585*4882a593Smuzhiyun 
586*4882a593Smuzhiyun 	if (list_empty(head2))
587*4882a593Smuzhiyun 		return;
588*4882a593Smuzhiyun 
589*4882a593Smuzhiyun 	if (list_empty(head)) {
590*4882a593Smuzhiyun 		list_splice_tail(head2, head);
591*4882a593Smuzhiyun 		INIT_LIST_HEAD(head2);
592*4882a593Smuzhiyun 		return;
593*4882a593Smuzhiyun 	}
594*4882a593Smuzhiyun 
595*4882a593Smuzhiyun 	/* Add or update & replace */
596*4882a593Smuzhiyun 	list_for_each_safe(epos, enext, head2) {
597*4882a593Smuzhiyun 		extra = list_entry(epos, struct cfg_list_item, list);
598*4882a593Smuzhiyun 
599*4882a593Smuzhiyun 		list_for_each_safe(pos, next, head) {
600*4882a593Smuzhiyun 			n = list_entry(pos, struct cfg_list_item, list);
601*4882a593Smuzhiyun 			if (extra->offset == n->offset) {
602*4882a593Smuzhiyun 				if (extra->len < n->len) {
603*4882a593Smuzhiyun 					/* Update the cfg data */
604*4882a593Smuzhiyun 					RS_INFO("Update cfg: ofs %04x len %u",
605*4882a593Smuzhiyun 						n->offset, n->len);
606*4882a593Smuzhiyun 					memcpy(n->data, extra->data,
607*4882a593Smuzhiyun 					       extra->len);
608*4882a593Smuzhiyun 					list_del(epos);
609*4882a593Smuzhiyun 					free(extra);
610*4882a593Smuzhiyun 				} else {
611*4882a593Smuzhiyun 					/* Replace the item */
612*4882a593Smuzhiyun 					list_del(epos);
613*4882a593Smuzhiyun 					list_replace_init(pos, epos);
614*4882a593Smuzhiyun 					/* free the old item */
615*4882a593Smuzhiyun 					free(n);
616*4882a593Smuzhiyun 				}
617*4882a593Smuzhiyun 			}
618*4882a593Smuzhiyun 
619*4882a593Smuzhiyun 		}
620*4882a593Smuzhiyun 
621*4882a593Smuzhiyun 	}
622*4882a593Smuzhiyun 
623*4882a593Smuzhiyun 	if (list_empty(head2))
624*4882a593Smuzhiyun 		return;
625*4882a593Smuzhiyun 	list_for_each_safe(epos, enext, head2) {
626*4882a593Smuzhiyun 		extra = list_entry(epos, struct cfg_list_item, list);
627*4882a593Smuzhiyun 		RS_INFO("Add new cfg: ofs %04x, len %u", extra->offset,
628*4882a593Smuzhiyun 			extra->len);
629*4882a593Smuzhiyun 		/* Add the item to list */
630*4882a593Smuzhiyun 		list_del(epos);
631*4882a593Smuzhiyun 		list_add_tail(epos, head);
632*4882a593Smuzhiyun 	}
633*4882a593Smuzhiyun }
634*4882a593Smuzhiyun 
635*4882a593Smuzhiyun /*
636*4882a593Smuzhiyun  * Parse realtek Bluetooth config file.
637*4882a593Smuzhiyun  * The content starts with vendor magic: 55 ab 23 87
638*4882a593Smuzhiyun  */
rtb_parse_config(uint8_t * cfg_buf,size_t len)639*4882a593Smuzhiyun int rtb_parse_config(uint8_t *cfg_buf, size_t len)
640*4882a593Smuzhiyun {
641*4882a593Smuzhiyun 	uint16_t cfg_len;
642*4882a593Smuzhiyun 	uint16_t tmp;
643*4882a593Smuzhiyun 	struct rtb_cfg_item *entry;
644*4882a593Smuzhiyun 	uint16_t i;
645*4882a593Smuzhiyun 	struct cfg_list_item *item;
646*4882a593Smuzhiyun 
647*4882a593Smuzhiyun 	if (!cfg_buf || !len) {
648*4882a593Smuzhiyun 		RS_ERR("%s: Invalid parameter", __func__);
649*4882a593Smuzhiyun 		return -1;
650*4882a593Smuzhiyun 	}
651*4882a593Smuzhiyun 
652*4882a593Smuzhiyun 	if (memcmp(cfg_buf, cfg_magic, 4)) {
653*4882a593Smuzhiyun 		RS_ERR("Signature %02x %02x %02x %02x is incorrect",
654*4882a593Smuzhiyun 		       cfg_buf[0], cfg_buf[1], cfg_buf[2], cfg_buf[3]);
655*4882a593Smuzhiyun 		return -1;
656*4882a593Smuzhiyun 	}
657*4882a593Smuzhiyun 
658*4882a593Smuzhiyun 	cfg_len = ((uint16_t)cfg_buf[5] << 8) + cfg_buf[4];
659*4882a593Smuzhiyun 	if (cfg_len != len - RTB_CFG_HDR_LEN) {
660*4882a593Smuzhiyun 		RS_ERR("Config len %u is incorrect(%zd)", cfg_len,
661*4882a593Smuzhiyun 		       len - RTB_CFG_HDR_LEN);
662*4882a593Smuzhiyun 		return -1;
663*4882a593Smuzhiyun 	}
664*4882a593Smuzhiyun 
665*4882a593Smuzhiyun 	entry = (struct rtb_cfg_item *)(cfg_buf + 6);
666*4882a593Smuzhiyun 	i = 0;
667*4882a593Smuzhiyun 	while (i < cfg_len) {
668*4882a593Smuzhiyun 		/* Add config item to list */
669*4882a593Smuzhiyun 		item = malloc(sizeof(*item) + entry->len);
670*4882a593Smuzhiyun 		if (item) {
671*4882a593Smuzhiyun 			memset(item, 0, sizeof(*item));
672*4882a593Smuzhiyun 			item->offset = le16_to_cpu(entry->offset);
673*4882a593Smuzhiyun 			item->len = entry->len;
674*4882a593Smuzhiyun 			memcpy(item->data, entry->data, item->len);
675*4882a593Smuzhiyun 			list_add_tail(&item->list, &list_configs);
676*4882a593Smuzhiyun 		} else {
677*4882a593Smuzhiyun 			RS_ERR("Cannot alloc mem for entry %04x, %u",
678*4882a593Smuzhiyun 			       entry->offset, entry->len);
679*4882a593Smuzhiyun 		}
680*4882a593Smuzhiyun 
681*4882a593Smuzhiyun 		tmp = entry->len + sizeof(struct rtb_cfg_item);
682*4882a593Smuzhiyun 		i += tmp;
683*4882a593Smuzhiyun 		entry = (struct rtb_cfg_item *)((uint8_t *)entry + tmp);
684*4882a593Smuzhiyun 	}
685*4882a593Smuzhiyun 
686*4882a593Smuzhiyun 	return 0;
687*4882a593Smuzhiyun }
688*4882a593Smuzhiyun 
bachk(const char * str)689*4882a593Smuzhiyun int bachk(const char *str)
690*4882a593Smuzhiyun {
691*4882a593Smuzhiyun 	if (!str)
692*4882a593Smuzhiyun 		return -1;
693*4882a593Smuzhiyun 
694*4882a593Smuzhiyun 	if (strlen(str) != 17)
695*4882a593Smuzhiyun 		return -1;
696*4882a593Smuzhiyun 
697*4882a593Smuzhiyun 	while (*str) {
698*4882a593Smuzhiyun 		if (!isxdigit(*str++))
699*4882a593Smuzhiyun 			return -1;
700*4882a593Smuzhiyun 
701*4882a593Smuzhiyun 		if (!isxdigit(*str++))
702*4882a593Smuzhiyun 			return -1;
703*4882a593Smuzhiyun 
704*4882a593Smuzhiyun 		if (*str == 0)
705*4882a593Smuzhiyun 			break;
706*4882a593Smuzhiyun 
707*4882a593Smuzhiyun 		if (*str++ != ':')
708*4882a593Smuzhiyun 			return -1;
709*4882a593Smuzhiyun 	}
710*4882a593Smuzhiyun 
711*4882a593Smuzhiyun 	return 0;
712*4882a593Smuzhiyun }
713*4882a593Smuzhiyun /*
714*4882a593Smuzhiyun  * Get random Bluetooth addr.
715*4882a593Smuzhiyun  */
716*4882a593Smuzhiyun /* static void rtb_get_ram_addr(char bt_addr[0])
717*4882a593Smuzhiyun  * {
718*4882a593Smuzhiyun  * 	srand(time(NULL) + getpid() + getpid() * 987654 + rand());
719*4882a593Smuzhiyun  *
720*4882a593Smuzhiyun  * 	uint32_t addr = rand();
721*4882a593Smuzhiyun  * 	memcpy(bt_addr, &addr, sizeof(uint8_t));
722*4882a593Smuzhiyun  * }
723*4882a593Smuzhiyun  */
724*4882a593Smuzhiyun typedef uint8_t RT_U8, *PRT_U8;
725*4882a593Smuzhiyun typedef int8_t RT_S8, *PRT_S8;
726*4882a593Smuzhiyun typedef uint16_t RT_U16, *PRT_U16;
727*4882a593Smuzhiyun typedef int32_t RT_S32, *PRT_S32;
728*4882a593Smuzhiyun typedef uint32_t RT_U32, *PRT_U32;
rtk_get_ram_addr(uint8_t bt_addr[0])729*4882a593Smuzhiyun static void rtk_get_ram_addr(uint8_t bt_addr[0])
730*4882a593Smuzhiyun {
731*4882a593Smuzhiyun 	srand(time(NULL) + getpid() + getpid() * 987654 + rand());
732*4882a593Smuzhiyun 
733*4882a593Smuzhiyun 	RT_U32 addr = rand();
734*4882a593Smuzhiyun 	memcpy(bt_addr, &addr, sizeof(RT_U8));
735*4882a593Smuzhiyun }
736*4882a593Smuzhiyun 
737*4882a593Smuzhiyun #ifdef BT_ADDR_FROM_VENDOR_STORAGE
738*4882a593Smuzhiyun 
739*4882a593Smuzhiyun typedef         unsigned short      uint16;
740*4882a593Smuzhiyun typedef         unsigned int        uint32;
741*4882a593Smuzhiyun typedef         unsigned char       uint8;
742*4882a593Smuzhiyun 
743*4882a593Smuzhiyun #define VENDOR_REQ_TAG          0x56524551
744*4882a593Smuzhiyun #define VENDOR_READ_IO          _IOW('v', 0x01, unsigned int)
745*4882a593Smuzhiyun #define VENDOR_WRITE_IO         _IOW('v', 0x02, unsigned int)
746*4882a593Smuzhiyun 
747*4882a593Smuzhiyun #define VENDOR_ID_MAX   5
748*4882a593Smuzhiyun static char *vendor_id_table[] = {
749*4882a593Smuzhiyun         "VENDOR_SN_ID",
750*4882a593Smuzhiyun         "VENDOR_WIFI_MAC_ID",
751*4882a593Smuzhiyun         "VENDOR_LAN_MAC_ID",
752*4882a593Smuzhiyun         "VENDOR_BT_MAC_ID",
753*4882a593Smuzhiyun         "VENDOR_IMEI_ID",
754*4882a593Smuzhiyun };
755*4882a593Smuzhiyun 
756*4882a593Smuzhiyun #define VENDOR_SN_ID            1
757*4882a593Smuzhiyun #define VENDOR_WIFI_MAC_ID      2
758*4882a593Smuzhiyun #define VENDOR_LAN_MAC_ID       3
759*4882a593Smuzhiyun #define VENDOR_BT_MAC_ID        4
760*4882a593Smuzhiyun #define VENDOR_IMEI_ID          5
761*4882a593Smuzhiyun 
762*4882a593Smuzhiyun #define VENDOR_STORAGE_DEBUG
763*4882a593Smuzhiyun 
764*4882a593Smuzhiyun struct rk_vendor_req {
765*4882a593Smuzhiyun         uint32 tag;
766*4882a593Smuzhiyun         uint16 id;
767*4882a593Smuzhiyun         uint16 len;
768*4882a593Smuzhiyun         uint8 data[1024];
769*4882a593Smuzhiyun };
770*4882a593Smuzhiyun 
rknand_get_randeom_btaddr(uint8_t * bt_addr)771*4882a593Smuzhiyun static void rknand_get_randeom_btaddr(uint8_t *bt_addr)
772*4882a593Smuzhiyun {
773*4882a593Smuzhiyun 	int i;
774*4882a593Smuzhiyun 	/* No autogen BDA. Generate one now. */
775*4882a593Smuzhiyun 	bt_addr[4] = 0x22;
776*4882a593Smuzhiyun 	bt_addr[5] = 0x22;
777*4882a593Smuzhiyun 	for (i = 0; i < 4; i++)
778*4882a593Smuzhiyun 		rtk_get_ram_addr(&bt_addr[i]);
779*4882a593Smuzhiyun }
780*4882a593Smuzhiyun 
rknand_print_hex_data(char * s,struct rk_vendor_req * buf,uint32 len)781*4882a593Smuzhiyun static void rknand_print_hex_data(char *s, struct rk_vendor_req *buf, uint32 len)
782*4882a593Smuzhiyun {
783*4882a593Smuzhiyun         unsigned char i = 0;
784*4882a593Smuzhiyun 
785*4882a593Smuzhiyun #ifdef VENDOR_STORAGE_DEBUG
786*4882a593Smuzhiyun         fprintf(stdout, "%s\n",s);
787*4882a593Smuzhiyun         fprintf(stdout, "tag = %d // id = %d // len = %d // data = 0x%p\n", buf->tag, buf->id, buf->len, buf->data);
788*4882a593Smuzhiyun #endif
789*4882a593Smuzhiyun 
790*4882a593Smuzhiyun         printf("%s: ", vendor_id_table[buf->id - 1]);
791*4882a593Smuzhiyun         if (buf->id == VENDOR_SN_ID ||
792*4882a593Smuzhiyun             buf->id == VENDOR_IMEI_ID) {
793*4882a593Smuzhiyun                 for (i = 0; i < len; i++)
794*4882a593Smuzhiyun                         printf("%c", buf->data[i]);
795*4882a593Smuzhiyun         } else {
796*4882a593Smuzhiyun                 for (i = 0; i < len; i++)
797*4882a593Smuzhiyun                         printf("%02x", buf->data[i]);
798*4882a593Smuzhiyun         }
799*4882a593Smuzhiyun         fprintf(stdout, "\n");
800*4882a593Smuzhiyun }
801*4882a593Smuzhiyun 
vendor_storage_read(int cmd,uint8_t * buf,int buf_len)802*4882a593Smuzhiyun static int vendor_storage_read(int cmd, uint8_t *buf, int buf_len)
803*4882a593Smuzhiyun {
804*4882a593Smuzhiyun         int ret ;
805*4882a593Smuzhiyun         uint8 p_buf[100]; /* malloc req buffer or used extern buffer */
806*4882a593Smuzhiyun         struct rk_vendor_req *req;
807*4882a593Smuzhiyun 
808*4882a593Smuzhiyun         req = (struct rk_vendor_req *)p_buf;
809*4882a593Smuzhiyun         memset(p_buf, 0, 100);
810*4882a593Smuzhiyun         int sys_fd = open("/dev/vendor_storage", O_RDWR, 0);
811*4882a593Smuzhiyun         if(sys_fd < 0){
812*4882a593Smuzhiyun                 printf("vendor_storage open fail\n");
813*4882a593Smuzhiyun                 return -1;
814*4882a593Smuzhiyun         }
815*4882a593Smuzhiyun 
816*4882a593Smuzhiyun         req->tag = VENDOR_REQ_TAG;
817*4882a593Smuzhiyun         req->id = cmd;
818*4882a593Smuzhiyun         req->len = 50;
819*4882a593Smuzhiyun 
820*4882a593Smuzhiyun         ret = ioctl(sys_fd, VENDOR_READ_IO, req);
821*4882a593Smuzhiyun 
822*4882a593Smuzhiyun         if(ret){
823*4882a593Smuzhiyun                 printf("vendor read error %d\n", ret);
824*4882a593Smuzhiyun                 return -1;
825*4882a593Smuzhiyun         }
826*4882a593Smuzhiyun         close(sys_fd);
827*4882a593Smuzhiyun 
828*4882a593Smuzhiyun         rknand_print_hex_data("vendor read:", req, req->len);
829*4882a593Smuzhiyun 		memcpy(buf, req->data, req->len);
830*4882a593Smuzhiyun         return req->len;
831*4882a593Smuzhiyun }
vendor_storage_write(int cmd,uint8_t * num)832*4882a593Smuzhiyun static int vendor_storage_write(int cmd, uint8_t *num)
833*4882a593Smuzhiyun {
834*4882a593Smuzhiyun 		int ret;
835*4882a593Smuzhiyun         uint8 p_buf[100]; /* malloc req buffer or used extern buffer */
836*4882a593Smuzhiyun         struct rk_vendor_req *req;
837*4882a593Smuzhiyun 
838*4882a593Smuzhiyun         req = (struct rk_vendor_req *)p_buf;
839*4882a593Smuzhiyun         int sys_fd = open("/dev/vendor_storage",O_RDWR,0);
840*4882a593Smuzhiyun         if(sys_fd < 0){
841*4882a593Smuzhiyun                 printf("vendor_storage open fail\n");
842*4882a593Smuzhiyun                 return -1;
843*4882a593Smuzhiyun         }
844*4882a593Smuzhiyun 
845*4882a593Smuzhiyun         req->tag = VENDOR_REQ_TAG;
846*4882a593Smuzhiyun         req->id = cmd;
847*4882a593Smuzhiyun 
848*4882a593Smuzhiyun         if (cmd != VENDOR_SN_ID && cmd != VENDOR_IMEI_ID)
849*4882a593Smuzhiyun                 req->len = 6;
850*4882a593Smuzhiyun         else
851*4882a593Smuzhiyun                 req->len = strlen((char *)num);
852*4882a593Smuzhiyun         memcpy(req->data, num, req->len);
853*4882a593Smuzhiyun 
854*4882a593Smuzhiyun         ret = ioctl(sys_fd, VENDOR_WRITE_IO, req);
855*4882a593Smuzhiyun         if(ret){
856*4882a593Smuzhiyun                 printf("vendor write error\n");
857*4882a593Smuzhiyun                 return -1;
858*4882a593Smuzhiyun         }
859*4882a593Smuzhiyun 
860*4882a593Smuzhiyun         rknand_print_hex_data("vendor write:", req, req->len);
861*4882a593Smuzhiyun         return 0;
862*4882a593Smuzhiyun }
vendor_storage_read_bt_addr(char * tbuf)863*4882a593Smuzhiyun static int vendor_storage_read_bt_addr(char *tbuf)
864*4882a593Smuzhiyun {
865*4882a593Smuzhiyun 	uint8 raw_buf[100] = {0};
866*4882a593Smuzhiyun 	int raw_len = 0;
867*4882a593Smuzhiyun     int i;
868*4882a593Smuzhiyun 
869*4882a593Smuzhiyun 	raw_len = vendor_storage_read(VENDOR_BT_MAC_ID, raw_buf, 100);
870*4882a593Smuzhiyun 	if (raw_len < 0)
871*4882a593Smuzhiyun 		return -1;
872*4882a593Smuzhiyun     for (i = 0; i < 6; i++) {
873*4882a593Smuzhiyun         sprintf(tbuf + i * 3, "%02x:", raw_buf[i]);
874*4882a593Smuzhiyun 	}
875*4882a593Smuzhiyun 	tbuf[17] = '\0';
876*4882a593Smuzhiyun 	return 17;
877*4882a593Smuzhiyun }
vendor_storage_write_bt_addr(uint8_t * tbuf)878*4882a593Smuzhiyun static int vendor_storage_write_bt_addr(uint8_t *tbuf)
879*4882a593Smuzhiyun {
880*4882a593Smuzhiyun 	return vendor_storage_write(VENDOR_BT_MAC_ID, tbuf);
881*4882a593Smuzhiyun }
882*4882a593Smuzhiyun #endif
883*4882a593Smuzhiyun 
884*4882a593Smuzhiyun /*
885*4882a593Smuzhiyun  * Write the random addr to the BT_ADDR_FILE.
886*4882a593Smuzhiyun  */
887*4882a593Smuzhiyun /* static void rtb_write_btmac2file(char bt_addr[6])
888*4882a593Smuzhiyun  * {
889*4882a593Smuzhiyun  * 	int fd;
890*4882a593Smuzhiyun  * 	fd = open(BT_ADDR_FILE, O_CREAT | O_RDWR | O_TRUNC);
891*4882a593Smuzhiyun  *
892*4882a593Smuzhiyun  * 	if (fd > 0) {
893*4882a593Smuzhiyun  * 		chmod(BT_ADDR_FILE, 0666);
894*4882a593Smuzhiyun  * 		char addr[18] = { 0 };
895*4882a593Smuzhiyun  * 		addr[17] = '\0';
896*4882a593Smuzhiyun  * 		sprintf(addr, "%2x:%2x:%2x:%2x:%2x:%2x", bt_addr[0], bt_addr[1],
897*4882a593Smuzhiyun  * 			bt_addr[2], bt_addr[3], bt_addr[4], bt_addr[5]);
898*4882a593Smuzhiyun  * 		write(fd, addr, strlen(addr));
899*4882a593Smuzhiyun  * 		close(fd);
900*4882a593Smuzhiyun  * 	} else {
901*4882a593Smuzhiyun  * 		RS_ERR("open file error:%s\n", BT_ADDR_FILE);
902*4882a593Smuzhiyun  * 	}
903*4882a593Smuzhiyun  * }
904*4882a593Smuzhiyun  */
905*4882a593Smuzhiyun 
906*4882a593Smuzhiyun /*
907*4882a593Smuzhiyun  * Read and parse Realtek Bluetooth Config file.
908*4882a593Smuzhiyun  */
rtb_read_config(const char * file,int * cfg_len,uint8_t chip_type)909*4882a593Smuzhiyun uint8_t *rtb_read_config(const char *file, int *cfg_len, uint8_t chip_type)
910*4882a593Smuzhiyun {
911*4882a593Smuzhiyun 	char *file_name;
912*4882a593Smuzhiyun 	struct stat st;
913*4882a593Smuzhiyun 	ssize_t file_len;
914*4882a593Smuzhiyun 	int fd;
915*4882a593Smuzhiyun 	uint8_t *buf;
916*4882a593Smuzhiyun #ifndef BT_ADDR_FROM_VENDOR_STORAGE
917*4882a593Smuzhiyun 	size_t size;
918*4882a593Smuzhiyun #endif
919*4882a593Smuzhiyun 	size_t result;
920*4882a593Smuzhiyun 	struct list_head *pos, *next;
921*4882a593Smuzhiyun 	struct cfg_list_item *n;
922*4882a593Smuzhiyun 	uint16_t config_len;
923*4882a593Smuzhiyun 	uint16_t dlen;
924*4882a593Smuzhiyun 	uint32_t baudrate;
925*4882a593Smuzhiyun 	uint8_t *p;
926*4882a593Smuzhiyun 
927*4882a593Smuzhiyun 	if (!file || !cfg_len) {
928*4882a593Smuzhiyun 		RS_ERR("%s: Invalid parameter", __func__);
929*4882a593Smuzhiyun 		return NULL;
930*4882a593Smuzhiyun 	}
931*4882a593Smuzhiyun 
932*4882a593Smuzhiyun 	config_lists_init();
933*4882a593Smuzhiyun 
934*4882a593Smuzhiyun 	/* All extra configs will be added to list_extracfgs */
935*4882a593Smuzhiyun 	parse_extra_config(EXTRA_CONFIG_FILE);
936*4882a593Smuzhiyun 	list_for_each_safe(pos, next, &list_extracfgs) {
937*4882a593Smuzhiyun 		n = list_entry(pos, struct cfg_list_item, list);
938*4882a593Smuzhiyun 		RS_INFO("extra cfg: ofs %04x, len %u", n->offset, n->len);
939*4882a593Smuzhiyun 	}
940*4882a593Smuzhiyun 
941*4882a593Smuzhiyun #ifdef BT_ADDR_FROM_VENDOR_STORAGE
942*4882a593Smuzhiyun 	int ret = 0;
943*4882a593Smuzhiyun 	uint8_t bdaddr[6];
944*4882a593Smuzhiyun 	uint16_t ofs;
945*4882a593Smuzhiyun 	char tbuf[BDADDR_STRING_LEN + 1];
946*4882a593Smuzhiyun 	char *str;
947*4882a593Smuzhiyun 	int i = 0;
948*4882a593Smuzhiyun 
949*4882a593Smuzhiyun 	ret = vendor_storage_read_bt_addr(tbuf);
950*4882a593Smuzhiyun 	if (ret >= 0 && bachk(tbuf) < 0) {
951*4882a593Smuzhiyun 		ret = -1;
952*4882a593Smuzhiyun 		RS_ERR("vendor_storage bt addr chechk failed");
953*4882a593Smuzhiyun 	}
954*4882a593Smuzhiyun 	if (ret < 0) {
955*4882a593Smuzhiyun 		RS_ERR("vendor storage read bt addr failed, generate one");
956*4882a593Smuzhiyun 
957*4882a593Smuzhiyun 		rknand_get_randeom_btaddr(bdaddr);
958*4882a593Smuzhiyun 		vendor_storage_write_bt_addr(bdaddr);
959*4882a593Smuzhiyun 	} else {
960*4882a593Smuzhiyun 		str = tbuf;
961*4882a593Smuzhiyun 		for (i = 5; i >= 0; i--) {
962*4882a593Smuzhiyun 			bdaddr[i] = (uint8_t)strtoul(str, NULL, 16);
963*4882a593Smuzhiyun 			str += 3;
964*4882a593Smuzhiyun 		}
965*4882a593Smuzhiyun 
966*4882a593Smuzhiyun 		/*reserve LAP addr from 0x9e8b00 to 0x9e8b3f, change to 0x008b** */
967*4882a593Smuzhiyun 		if (0x9e == bdaddr[3] && 0x8b == bdaddr[4]
968*4882a593Smuzhiyun 			&& (bdaddr[5] <= 0x3f)) {
969*4882a593Smuzhiyun 			bdaddr[3] = 0x00;
970*4882a593Smuzhiyun 		}
971*4882a593Smuzhiyun 
972*4882a593Smuzhiyun 		RS_DBG("BT MAC is %02x:%02x:%02x:%02x:%02x:%02x",
973*4882a593Smuzhiyun 			   bdaddr[5], bdaddr[4],
974*4882a593Smuzhiyun 			   bdaddr[3], bdaddr[2],
975*4882a593Smuzhiyun 			   bdaddr[1], bdaddr[0]);
976*4882a593Smuzhiyun 		ofs = get_mac_offset(chip_type);
977*4882a593Smuzhiyun 		n = malloc(sizeof(*n) + 6);
978*4882a593Smuzhiyun 		if (n) {
979*4882a593Smuzhiyun 			n->offset = ofs;
980*4882a593Smuzhiyun 			n->len = 6;
981*4882a593Smuzhiyun 			memcpy(n->data, bdaddr, 6);
982*4882a593Smuzhiyun 			list_add_tail(&n->list, &list_extracfgs);
983*4882a593Smuzhiyun 		} else {
984*4882a593Smuzhiyun 			RS_ERR("Couldn't alloc cfg item for bdaddr");
985*4882a593Smuzhiyun 		}
986*4882a593Smuzhiyun 	}
987*4882a593Smuzhiyun #else
988*4882a593Smuzhiyun 	if (stat(BT_ADDR_FILE, &st) < 0) {
989*4882a593Smuzhiyun 		RS_INFO("Couldnt access customer BT MAC file %s",
990*4882a593Smuzhiyun 		        BT_ADDR_FILE);
991*4882a593Smuzhiyun 
992*4882a593Smuzhiyun 		goto read_cfg;
993*4882a593Smuzhiyun 	}
994*4882a593Smuzhiyun 
995*4882a593Smuzhiyun 	size = st.st_size;
996*4882a593Smuzhiyun 	/* Only read the first 17-byte if the file length is larger */
997*4882a593Smuzhiyun 	if (size > BDADDR_STRING_LEN)
998*4882a593Smuzhiyun 		size = BDADDR_STRING_LEN;
999*4882a593Smuzhiyun 
1000*4882a593Smuzhiyun 	fd = open(BT_ADDR_FILE, O_RDONLY);
1001*4882a593Smuzhiyun 	if (fd == -1) {
1002*4882a593Smuzhiyun 		RS_INFO("Couldnt open BT MAC file %s, %s", BT_ADDR_FILE,
1003*4882a593Smuzhiyun 			strerror(errno));
1004*4882a593Smuzhiyun 	} else {
1005*4882a593Smuzhiyun 		uint16_t ofs;
1006*4882a593Smuzhiyun 		char *str;
1007*4882a593Smuzhiyun 		int i = 0;
1008*4882a593Smuzhiyun 		uint8_t bdaddr[6];
1009*4882a593Smuzhiyun 		uint8_t tbuf[BDADDR_STRING_LEN + 1];
1010*4882a593Smuzhiyun 
1011*4882a593Smuzhiyun 		memset(tbuf, 0, sizeof(tbuf));
1012*4882a593Smuzhiyun 		result = read(fd, tbuf, size);
1013*4882a593Smuzhiyun 		close(fd);
1014*4882a593Smuzhiyun 		if (result == -1) {
1015*4882a593Smuzhiyun 			RS_ERR("Couldnt read BT MAC file %s, err %s",
1016*4882a593Smuzhiyun 			       BT_ADDR_FILE, strerror(errno));
1017*4882a593Smuzhiyun 			goto read_cfg;
1018*4882a593Smuzhiyun 		}
1019*4882a593Smuzhiyun 
1020*4882a593Smuzhiyun 		if (bachk((const char *)tbuf) < 0) {
1021*4882a593Smuzhiyun 			goto read_cfg;
1022*4882a593Smuzhiyun 		}
1023*4882a593Smuzhiyun 
1024*4882a593Smuzhiyun 		str = (char *)tbuf;
1025*4882a593Smuzhiyun 		for (i = 5; i >= 0; i--) {
1026*4882a593Smuzhiyun 			bdaddr[i] = (uint8_t)strtoul(str, NULL, 16);
1027*4882a593Smuzhiyun 			str += 3;
1028*4882a593Smuzhiyun 		}
1029*4882a593Smuzhiyun 
1030*4882a593Smuzhiyun 		/* Reserve LAP addr from 0x9e8b00 to 0x9e8b3f,
1031*4882a593Smuzhiyun 		 * Change to 0x008bXX */
1032*4882a593Smuzhiyun 		if (0x9e == bdaddr[3] && 0x8b == bdaddr[4] &&
1033*4882a593Smuzhiyun 		    bdaddr[5] <= 0x3f)
1034*4882a593Smuzhiyun 			bdaddr[3] = 0x00;
1035*4882a593Smuzhiyun 
1036*4882a593Smuzhiyun 		RS_DBG("BT MAC %02x:%02x:%02x:%02x:%02x:%02x",
1037*4882a593Smuzhiyun 		       bdaddr[5], bdaddr[4], bdaddr[3], bdaddr[2],
1038*4882a593Smuzhiyun 		       bdaddr[1], bdaddr[0]);
1039*4882a593Smuzhiyun 		ofs = get_mac_offset(chip_type);
1040*4882a593Smuzhiyun 		n = malloc(sizeof(*n) + 6);
1041*4882a593Smuzhiyun 		if (n) {
1042*4882a593Smuzhiyun 			n->offset = ofs;
1043*4882a593Smuzhiyun 			n->len = 6;
1044*4882a593Smuzhiyun 			memcpy(n->data, bdaddr, 6);
1045*4882a593Smuzhiyun 			list_add_tail(&n->list, &list_extracfgs);
1046*4882a593Smuzhiyun 		} else {
1047*4882a593Smuzhiyun 			RS_ERR("Couldn't alloc cfg item for bdaddr");
1048*4882a593Smuzhiyun 		}
1049*4882a593Smuzhiyun 	}
1050*4882a593Smuzhiyun 
1051*4882a593Smuzhiyun read_cfg:
1052*4882a593Smuzhiyun #endif
1053*4882a593Smuzhiyun 
1054*4882a593Smuzhiyun 	*cfg_len = 0;
1055*4882a593Smuzhiyun 	file_name = malloc(PATH_MAX);
1056*4882a593Smuzhiyun 	if (!file_name) {
1057*4882a593Smuzhiyun 		RS_ERR("Can't allocate memory for Config file name");
1058*4882a593Smuzhiyun 		return NULL;
1059*4882a593Smuzhiyun 	}
1060*4882a593Smuzhiyun 	memset(file_name, 0, PATH_MAX);
1061*4882a593Smuzhiyun 	snprintf(file_name, PATH_MAX, "%s%s", BT_CONFIG_DIRECTORY, file);
1062*4882a593Smuzhiyun 	if (stat(file_name, &st) < 0) {
1063*4882a593Smuzhiyun 		RS_ERR("Can't access Config file: %s, %s",
1064*4882a593Smuzhiyun 		       file_name, strerror(errno));
1065*4882a593Smuzhiyun 		goto err_stat;
1066*4882a593Smuzhiyun 	}
1067*4882a593Smuzhiyun 
1068*4882a593Smuzhiyun 	file_len = st.st_size;
1069*4882a593Smuzhiyun 
1070*4882a593Smuzhiyun 	if ((fd = open(file_name, O_RDONLY)) < 0) {
1071*4882a593Smuzhiyun 		perror("Can't open Config file");
1072*4882a593Smuzhiyun 		goto err_open;
1073*4882a593Smuzhiyun 	}
1074*4882a593Smuzhiyun 
1075*4882a593Smuzhiyun 	buf = malloc(file_len);
1076*4882a593Smuzhiyun 	if (!buf) {
1077*4882a593Smuzhiyun 		RS_ERR("Couldnt malloc buffer for Config %zd", file_len);
1078*4882a593Smuzhiyun 		goto err_malloc;
1079*4882a593Smuzhiyun 	}
1080*4882a593Smuzhiyun 
1081*4882a593Smuzhiyun 	result = read(fd, buf, file_len);
1082*4882a593Smuzhiyun 	if (result < (ssize_t)file_len) {
1083*4882a593Smuzhiyun 		perror("Can't read Config file");
1084*4882a593Smuzhiyun 		goto err_read;
1085*4882a593Smuzhiyun 	}
1086*4882a593Smuzhiyun 	close(fd);
1087*4882a593Smuzhiyun 	free(file_name);
1088*4882a593Smuzhiyun 
1089*4882a593Smuzhiyun 	result = rtb_parse_config(buf, file_len);
1090*4882a593Smuzhiyun 	if (result < 0) {
1091*4882a593Smuzhiyun 		RS_ERR("Invalid Config content");
1092*4882a593Smuzhiyun 		close(fd);
1093*4882a593Smuzhiyun 		free(buf);
1094*4882a593Smuzhiyun 		config_lists_free();
1095*4882a593Smuzhiyun 		exit(EXIT_FAILURE);
1096*4882a593Smuzhiyun 	}
1097*4882a593Smuzhiyun 	RS_INFO("Origin cfg len %u", (uint16_t)file_len);
1098*4882a593Smuzhiyun 	util_hexdump((const uint8_t *)buf, file_len);
1099*4882a593Smuzhiyun 
1100*4882a593Smuzhiyun 	merge_configs(&list_configs, &list_extracfgs);
1101*4882a593Smuzhiyun 
1102*4882a593Smuzhiyun 	config_len = 4; /* magic word length */
1103*4882a593Smuzhiyun 	config_len += 2; /* data length field */
1104*4882a593Smuzhiyun 	/* Calculate the config_len */
1105*4882a593Smuzhiyun 	dlen = 0;
1106*4882a593Smuzhiyun 	list_for_each_safe(pos, next, &list_configs) {
1107*4882a593Smuzhiyun 		n = list_entry(pos, struct cfg_list_item, list);
1108*4882a593Smuzhiyun 		switch (n->offset) {
1109*4882a593Smuzhiyun 		case 0x003c:
1110*4882a593Smuzhiyun 		case 0x0030:
1111*4882a593Smuzhiyun 		case 0x0044:
1112*4882a593Smuzhiyun 			if (is_mac(chip_type, n->offset) && n->len == 6) {
1113*4882a593Smuzhiyun 				char s[18];
1114*4882a593Smuzhiyun 				sprintf(s, "%2.2X:%2.2X:%2.2X:%2.2X:%2.2X:%2.2X",
1115*4882a593Smuzhiyun 					n->data[5], n->data[4],
1116*4882a593Smuzhiyun 					n->data[3], n->data[2],
1117*4882a593Smuzhiyun 					n->data[1], n->data[0]);
1118*4882a593Smuzhiyun 				RS_INFO("bdaddr ofs %04x, %s", n->offset, s);
1119*4882a593Smuzhiyun 			}
1120*4882a593Smuzhiyun 			break;
1121*4882a593Smuzhiyun 		case 0x000c:
1122*4882a593Smuzhiyun #ifdef BAUDRATE_4BYTES
1123*4882a593Smuzhiyun 			baudrate = get_unaligned_le32(n->data);
1124*4882a593Smuzhiyun #else
1125*4882a593Smuzhiyun 			baudrate = get_unaligned_le16(n->data);
1126*4882a593Smuzhiyun #endif
1127*4882a593Smuzhiyun 			rtb_cfg.vendor_baud = baudrate;
1128*4882a593Smuzhiyun 			RS_INFO("Config baudrate: %08x", baudrate);
1129*4882a593Smuzhiyun 
1130*4882a593Smuzhiyun 			if (n->len > 12) {
1131*4882a593Smuzhiyun 				uint8_t d = n->data[12];
1132*4882a593Smuzhiyun 				rtb_cfg.uart_flow_ctrl = (d & 0x4) ? 1 : 0;
1133*4882a593Smuzhiyun 				RS_INFO("uart flow ctrl: %u",
1134*4882a593Smuzhiyun 					rtb_cfg.uart_flow_ctrl);
1135*4882a593Smuzhiyun 			}
1136*4882a593Smuzhiyun 			break;
1137*4882a593Smuzhiyun #ifdef RTL8723DSH4_UART_HWFLOWC
1138*4882a593Smuzhiyun 		case 0x0018:
1139*4882a593Smuzhiyun 			if (chip_type == CHIP_8723DS &&
1140*4882a593Smuzhiyun 			    rtb_cfg.proto == HCI_UART_H4) {
1141*4882a593Smuzhiyun 				if (n->data[0] & (1 << 2))
1142*4882a593Smuzhiyun 					rtb_cfg.uart_flow_ctrl = 1;
1143*4882a593Smuzhiyun 				RS_INFO("8723DSH4: hw flow control %u",
1144*4882a593Smuzhiyun 					rtb_cfg.uart_flow_ctrl);
1145*4882a593Smuzhiyun 				if (n->data[0] & 0x01) {
1146*4882a593Smuzhiyun 					rtb_cfg.parenb = 1;
1147*4882a593Smuzhiyun 					if (n->data[0] & 0x02)
1148*4882a593Smuzhiyun 						rtb_cfg.pareven = 1;
1149*4882a593Smuzhiyun 					else
1150*4882a593Smuzhiyun 						rtb_cfg.pareven = 0;
1151*4882a593Smuzhiyun 				}
1152*4882a593Smuzhiyun 				RS_INFO("8723DSH4: parity %u, even %u",
1153*4882a593Smuzhiyun 					rtb_cfg.parenb,
1154*4882a593Smuzhiyun 					rtb_cfg.pareven);
1155*4882a593Smuzhiyun 			}
1156*4882a593Smuzhiyun 			break;
1157*4882a593Smuzhiyun #endif
1158*4882a593Smuzhiyun 		default:
1159*4882a593Smuzhiyun 			break;
1160*4882a593Smuzhiyun 		}
1161*4882a593Smuzhiyun 
1162*4882a593Smuzhiyun 		config_len += (3 + n->len);
1163*4882a593Smuzhiyun 		dlen += (3 + n->len);
1164*4882a593Smuzhiyun 
1165*4882a593Smuzhiyun 	}
1166*4882a593Smuzhiyun 	p = realloc(buf, config_len);
1167*4882a593Smuzhiyun 	if (!p) {
1168*4882a593Smuzhiyun 		/* block is left untouched; it is not freed or moved */
1169*4882a593Smuzhiyun 		RS_ERR("Couldn't realloc buf for configs");
1170*4882a593Smuzhiyun 		free(buf);
1171*4882a593Smuzhiyun 		buf = NULL;
1172*4882a593Smuzhiyun 		*cfg_len = 0;
1173*4882a593Smuzhiyun 		goto done;
1174*4882a593Smuzhiyun 	}
1175*4882a593Smuzhiyun 	buf = p;
1176*4882a593Smuzhiyun 
1177*4882a593Smuzhiyun 	memcpy(buf, cfg_magic, 4);
1178*4882a593Smuzhiyun 	buf[4] = dlen & 0xff;
1179*4882a593Smuzhiyun 	buf[5] = (dlen >> 8) & 0xff;
1180*4882a593Smuzhiyun 	p = buf + 6;
1181*4882a593Smuzhiyun 	list_for_each_safe(pos, next, &list_configs) {
1182*4882a593Smuzhiyun 		n = list_entry(pos, struct cfg_list_item, list);
1183*4882a593Smuzhiyun 		p[0] = n->offset & 0xff;
1184*4882a593Smuzhiyun 		p[1] = (n->offset >> 8) & 0xff;
1185*4882a593Smuzhiyun 		p[2] = n->len;
1186*4882a593Smuzhiyun 		memcpy(p + 3, n->data, n->len);
1187*4882a593Smuzhiyun 		p += (3 + n->len);
1188*4882a593Smuzhiyun 	}
1189*4882a593Smuzhiyun 
1190*4882a593Smuzhiyun 	RS_INFO("Vendor baud from Config file: %08x", rtb_cfg.vendor_baud);
1191*4882a593Smuzhiyun 
1192*4882a593Smuzhiyun 	RS_INFO("New cfg len %u", config_len);
1193*4882a593Smuzhiyun 	util_hexdump((const uint8_t *)buf, config_len);
1194*4882a593Smuzhiyun 
1195*4882a593Smuzhiyun 	*cfg_len = config_len;
1196*4882a593Smuzhiyun 
1197*4882a593Smuzhiyun done:
1198*4882a593Smuzhiyun 	config_lists_free();
1199*4882a593Smuzhiyun 
1200*4882a593Smuzhiyun 	return buf;
1201*4882a593Smuzhiyun 
1202*4882a593Smuzhiyun err_read:
1203*4882a593Smuzhiyun 	free(buf);
1204*4882a593Smuzhiyun err_malloc:
1205*4882a593Smuzhiyun 	config_lists_free();
1206*4882a593Smuzhiyun 	close(fd);
1207*4882a593Smuzhiyun err_open:
1208*4882a593Smuzhiyun err_stat:
1209*4882a593Smuzhiyun 	free(file_name);
1210*4882a593Smuzhiyun 	return NULL;
1211*4882a593Smuzhiyun }
1212*4882a593Smuzhiyun 
1213*4882a593Smuzhiyun /*
1214*4882a593Smuzhiyun  * Read Realtek Bluetooth firmaware file.
1215*4882a593Smuzhiyun  */
rtb_read_firmware(struct rtb_struct * btrtl,int * fw_len)1216*4882a593Smuzhiyun uint8_t *rtb_read_firmware(struct rtb_struct *btrtl, int *fw_len)
1217*4882a593Smuzhiyun {
1218*4882a593Smuzhiyun 	char *filename;
1219*4882a593Smuzhiyun 	struct stat st;
1220*4882a593Smuzhiyun 	int fd = -1;
1221*4882a593Smuzhiyun 	size_t fwsize;
1222*4882a593Smuzhiyun 	uint8_t *fw_buf;
1223*4882a593Smuzhiyun 	ssize_t result;
1224*4882a593Smuzhiyun 
1225*4882a593Smuzhiyun 	if (!btrtl || !fw_len) {
1226*4882a593Smuzhiyun 		RS_ERR("%s: Invalid parameter", __func__);
1227*4882a593Smuzhiyun 		return NULL;
1228*4882a593Smuzhiyun 	}
1229*4882a593Smuzhiyun 
1230*4882a593Smuzhiyun 	filename = malloc(PATH_MAX);
1231*4882a593Smuzhiyun 	if (!filename) {
1232*4882a593Smuzhiyun 		RS_ERR("Can't allocate memory for fw name");
1233*4882a593Smuzhiyun 		return NULL;
1234*4882a593Smuzhiyun 	}
1235*4882a593Smuzhiyun 
1236*4882a593Smuzhiyun 	snprintf(filename, PATH_MAX, "%s%s", FIRMWARE_DIRECTORY,
1237*4882a593Smuzhiyun 		 btrtl->patch_ent->patch_file);
1238*4882a593Smuzhiyun 
1239*4882a593Smuzhiyun 	if (stat(filename, &st) < 0) {
1240*4882a593Smuzhiyun 		RS_ERR("Can't access firmware %s, %s", filename,
1241*4882a593Smuzhiyun 		       strerror(errno));
1242*4882a593Smuzhiyun 		goto err_stat;
1243*4882a593Smuzhiyun 	}
1244*4882a593Smuzhiyun 
1245*4882a593Smuzhiyun 	fwsize = st.st_size;
1246*4882a593Smuzhiyun 
1247*4882a593Smuzhiyun 	if ((fd = open(filename, O_RDONLY)) < 0) {
1248*4882a593Smuzhiyun 		RS_ERR("Can't open firmware, %s", strerror(errno));
1249*4882a593Smuzhiyun 		goto err_open;
1250*4882a593Smuzhiyun 	}
1251*4882a593Smuzhiyun 
1252*4882a593Smuzhiyun 	fw_buf = malloc(fwsize);
1253*4882a593Smuzhiyun 	if (!fw_buf) {
1254*4882a593Smuzhiyun 		RS_ERR("Can't allocate memory for fw, %s", strerror(errno));
1255*4882a593Smuzhiyun 		goto err_malloc;
1256*4882a593Smuzhiyun 	}
1257*4882a593Smuzhiyun 
1258*4882a593Smuzhiyun 	result = read(fd, fw_buf, fwsize);
1259*4882a593Smuzhiyun 	if (result != (ssize_t) fwsize) {
1260*4882a593Smuzhiyun 		RS_ERR("Read FW %s error, %s", filename, strerror(errno));
1261*4882a593Smuzhiyun 		goto err_read;
1262*4882a593Smuzhiyun 	}
1263*4882a593Smuzhiyun 
1264*4882a593Smuzhiyun 	*fw_len = (int)result;
1265*4882a593Smuzhiyun 	RS_INFO("Load FW %s OK, size %zd", filename, result);
1266*4882a593Smuzhiyun 
1267*4882a593Smuzhiyun 	close(fd);
1268*4882a593Smuzhiyun 	free(filename);
1269*4882a593Smuzhiyun 
1270*4882a593Smuzhiyun 	return fw_buf;
1271*4882a593Smuzhiyun 
1272*4882a593Smuzhiyun err_read:
1273*4882a593Smuzhiyun 	free(fw_buf);
1274*4882a593Smuzhiyun 	*fw_len = 0;
1275*4882a593Smuzhiyun err_malloc:
1276*4882a593Smuzhiyun 	close(fd);
1277*4882a593Smuzhiyun err_open:
1278*4882a593Smuzhiyun err_stat:
1279*4882a593Smuzhiyun 	free(filename);
1280*4882a593Smuzhiyun 	return NULL;
1281*4882a593Smuzhiyun }
1282*4882a593Smuzhiyun 
rtb_get_fw_project_id(uint8_t * p_buf)1283*4882a593Smuzhiyun static uint8_t rtb_get_fw_project_id(uint8_t *p_buf)
1284*4882a593Smuzhiyun {
1285*4882a593Smuzhiyun 	uint8_t opcode;
1286*4882a593Smuzhiyun 	uint8_t len;
1287*4882a593Smuzhiyun 	uint8_t data = 0;
1288*4882a593Smuzhiyun 
1289*4882a593Smuzhiyun 	do {
1290*4882a593Smuzhiyun 		opcode = *p_buf;
1291*4882a593Smuzhiyun 		len = *(p_buf - 1);
1292*4882a593Smuzhiyun 		if (opcode == 0x00) {
1293*4882a593Smuzhiyun 			if (len == 1) {
1294*4882a593Smuzhiyun 				data = *(p_buf - 2);
1295*4882a593Smuzhiyun 				RS_INFO("%s: opcode %u, len %u, data %u",
1296*4882a593Smuzhiyun 					__func__, opcode, len, data);
1297*4882a593Smuzhiyun 				break;
1298*4882a593Smuzhiyun 			} else {
1299*4882a593Smuzhiyun 				RS_ERR("%s: Invalid len %u", __func__, len);
1300*4882a593Smuzhiyun 			}
1301*4882a593Smuzhiyun 		}
1302*4882a593Smuzhiyun 		p_buf -= len + 2;
1303*4882a593Smuzhiyun 	} while (*p_buf != 0xFF);
1304*4882a593Smuzhiyun 
1305*4882a593Smuzhiyun 	return data;
1306*4882a593Smuzhiyun }
1307*4882a593Smuzhiyun 
rtb_get_patch_entry(void)1308*4882a593Smuzhiyun struct rtb_patch_entry *rtb_get_patch_entry(void)
1309*4882a593Smuzhiyun {
1310*4882a593Smuzhiyun 	uint16_t i;
1311*4882a593Smuzhiyun 	struct rtb_patch_hdr *patch;
1312*4882a593Smuzhiyun 	struct rtb_patch_entry *entry;
1313*4882a593Smuzhiyun 	uint32_t tmp;
1314*4882a593Smuzhiyun 	uint8_t *ci_base; /* Chip id base */
1315*4882a593Smuzhiyun 	uint8_t *pl_base; /* Patch length base */
1316*4882a593Smuzhiyun 	uint8_t *so_base; /* Start offset base */
1317*4882a593Smuzhiyun 
1318*4882a593Smuzhiyun 	patch = (struct rtb_patch_hdr *)rtb_cfg.fw_buf;
1319*4882a593Smuzhiyun 	entry = (struct rtb_patch_entry *)malloc(sizeof(*entry));
1320*4882a593Smuzhiyun 	if (!entry) {
1321*4882a593Smuzhiyun 		RS_ERR("Failed to allocate mem for patch entry");
1322*4882a593Smuzhiyun 		return NULL;
1323*4882a593Smuzhiyun 	}
1324*4882a593Smuzhiyun 
1325*4882a593Smuzhiyun 	patch->number_of_patch = le16_to_cpu(patch->number_of_patch);
1326*4882a593Smuzhiyun 
1327*4882a593Smuzhiyun 	RS_DBG("FW version 0x%08x, Patch num %u",
1328*4882a593Smuzhiyun 	       le32_to_cpu(patch->fw_version), patch->number_of_patch);
1329*4882a593Smuzhiyun 
1330*4882a593Smuzhiyun 	ci_base = rtb_cfg.fw_buf + 14;
1331*4882a593Smuzhiyun 	pl_base = ci_base + 2 * patch->number_of_patch;
1332*4882a593Smuzhiyun 	so_base = pl_base + 2 * patch->number_of_patch;
1333*4882a593Smuzhiyun 	for (i = 0; i < patch->number_of_patch; i++) {
1334*4882a593Smuzhiyun 		uint16_t chip_id = get_unaligned_le16(ci_base + 2 * i);
1335*4882a593Smuzhiyun 
1336*4882a593Smuzhiyun 		RS_INFO("Chip id 0x%04x", chip_id);
1337*4882a593Smuzhiyun 		if (chip_id == rtb_cfg.eversion + 1) {
1338*4882a593Smuzhiyun 			entry->chip_id = rtb_cfg.eversion + 1;
1339*4882a593Smuzhiyun 			entry->patch_len = get_unaligned_le16(pl_base + 2 * i);
1340*4882a593Smuzhiyun 			entry->soffset = get_unaligned_le32(so_base + 4 * i);
1341*4882a593Smuzhiyun 			RS_DBG("Patch length 0x%04x", entry->patch_len);
1342*4882a593Smuzhiyun 			RS_DBG("Start offset 0x%08x", entry->soffset);
1343*4882a593Smuzhiyun 
1344*4882a593Smuzhiyun 			entry->svn_ver = get_unaligned_le32(rtb_cfg.fw_buf +
1345*4882a593Smuzhiyun 						entry->soffset +
1346*4882a593Smuzhiyun 						entry->patch_len - 8);
1347*4882a593Smuzhiyun 			entry->coex_ver = get_unaligned_le32(rtb_cfg.fw_buf +
1348*4882a593Smuzhiyun 						entry->soffset +
1349*4882a593Smuzhiyun 						entry->patch_len - 12);
1350*4882a593Smuzhiyun 
1351*4882a593Smuzhiyun 			RS_INFO("Svn version: %8u", entry->svn_ver);
1352*4882a593Smuzhiyun 			tmp = ((entry->coex_ver >> 16) & 0x7ff) +
1353*4882a593Smuzhiyun 			      (entry->coex_ver >> 27) * 10000;
1354*4882a593Smuzhiyun 			RS_INFO("Coexistence: BTCOEX_20%06u-%04x\n", tmp,
1355*4882a593Smuzhiyun 				(entry->coex_ver & 0xffff));
1356*4882a593Smuzhiyun 
1357*4882a593Smuzhiyun 			break;
1358*4882a593Smuzhiyun 		}
1359*4882a593Smuzhiyun 	}
1360*4882a593Smuzhiyun 
1361*4882a593Smuzhiyun 	if (i == patch->number_of_patch) {
1362*4882a593Smuzhiyun 		RS_ERR("Failed to get entry");
1363*4882a593Smuzhiyun 		free(entry);
1364*4882a593Smuzhiyun 		entry = NULL;
1365*4882a593Smuzhiyun 	}
1366*4882a593Smuzhiyun 
1367*4882a593Smuzhiyun 	return entry;
1368*4882a593Smuzhiyun }
1369*4882a593Smuzhiyun 
rtb_get_final_patch(int fd,int proto,int * rlen)1370*4882a593Smuzhiyun uint8_t *rtb_get_final_patch(int fd, int proto, int *rlen)
1371*4882a593Smuzhiyun {
1372*4882a593Smuzhiyun 	struct rtb_struct *rtl = &rtb_cfg;
1373*4882a593Smuzhiyun 	uint8_t proj_id = 0;
1374*4882a593Smuzhiyun 	struct rtb_patch_entry *entry = NULL;
1375*4882a593Smuzhiyun 	struct rtb_patch_hdr *patch = (struct rtb_patch_hdr *)rtl->fw_buf;
1376*4882a593Smuzhiyun 	uint32_t svn_ver, coex_ver, tmp;
1377*4882a593Smuzhiyun 	const uint8_t rtb_patch_smagic[8] = {
1378*4882a593Smuzhiyun 		0x52, 0x65, 0x61, 0x6C, 0x74, 0x65, 0x63, 0x68
1379*4882a593Smuzhiyun 	};
1380*4882a593Smuzhiyun 	const uint8_t rtb_patch_emagic[4] = { 0x51, 0x04, 0xFD, 0x77 };
1381*4882a593Smuzhiyun 	uint8_t *buf;
1382*4882a593Smuzhiyun 	int len;
1383*4882a593Smuzhiyun 
1384*4882a593Smuzhiyun 	if (!rlen) {
1385*4882a593Smuzhiyun 		RS_ERR("%s: Invalid parameter", __func__);
1386*4882a593Smuzhiyun 		return NULL;
1387*4882a593Smuzhiyun 	}
1388*4882a593Smuzhiyun 
1389*4882a593Smuzhiyun 	/* Use single patch for 3wire && 8723a */
1390*4882a593Smuzhiyun 	if (proto == HCI_UART_3WIRE && rtl->lmp_subver == ROM_LMP_8723a) {
1391*4882a593Smuzhiyun 		if (!memcmp(rtl->fw_buf, rtb_patch_smagic, 8)) {
1392*4882a593Smuzhiyun 			RS_ERR("Unexpected signature");
1393*4882a593Smuzhiyun 			goto err;
1394*4882a593Smuzhiyun 		}
1395*4882a593Smuzhiyun 
1396*4882a593Smuzhiyun 		len = rtl->config_len + rtl->fw_len;
1397*4882a593Smuzhiyun 		buf = malloc(len);
1398*4882a593Smuzhiyun 		if (!buf) {
1399*4882a593Smuzhiyun 			RS_ERR("Can't alloc mem for fwc, %s", strerror(errno));
1400*4882a593Smuzhiyun 			goto err;
1401*4882a593Smuzhiyun 		} else {
1402*4882a593Smuzhiyun 			uint8_t *b;
1403*4882a593Smuzhiyun 
1404*4882a593Smuzhiyun 			RS_INFO("FWC copy directly");
1405*4882a593Smuzhiyun 
1406*4882a593Smuzhiyun 			b = rtl->fw_buf + rtl->fw_len;
1407*4882a593Smuzhiyun 			svn_ver = get_unaligned_le32(b - 8);
1408*4882a593Smuzhiyun 			coex_ver = get_unaligned_le32(b - 12);
1409*4882a593Smuzhiyun 
1410*4882a593Smuzhiyun 			RS_INFO("Svn version: %u\n", svn_ver);
1411*4882a593Smuzhiyun 			tmp = ((coex_ver >> 16) & 0x7ff) +
1412*4882a593Smuzhiyun 			      (coex_ver >> 27) * 10000;
1413*4882a593Smuzhiyun 			RS_INFO("Coexistence: BTCOEX_20%06u-%04x\n", tmp,
1414*4882a593Smuzhiyun 				(coex_ver & 0xffff));
1415*4882a593Smuzhiyun 
1416*4882a593Smuzhiyun 			/* Copy Patch and Config */
1417*4882a593Smuzhiyun 			memcpy(buf, rtl->fw_buf, rtl->fw_len);
1418*4882a593Smuzhiyun 			if (rtl->config_len)
1419*4882a593Smuzhiyun 				memcpy(buf + rtl->fw_len,
1420*4882a593Smuzhiyun 				       rtl->config_buf, rtl->config_len);
1421*4882a593Smuzhiyun 			rtl->dl_fw_flag = 1;
1422*4882a593Smuzhiyun 			*rlen = len;
1423*4882a593Smuzhiyun 			return buf;
1424*4882a593Smuzhiyun 		}
1425*4882a593Smuzhiyun 	}
1426*4882a593Smuzhiyun 
1427*4882a593Smuzhiyun 	if (memcmp(rtl->fw_buf, rtb_patch_smagic, 8)) {
1428*4882a593Smuzhiyun 		RS_ERR("Signature error");
1429*4882a593Smuzhiyun 		goto err;
1430*4882a593Smuzhiyun 	}
1431*4882a593Smuzhiyun 
1432*4882a593Smuzhiyun 	if (memcmp(rtl->fw_buf + rtl->fw_len - 4, rtb_patch_emagic, 4)) {
1433*4882a593Smuzhiyun 		RS_ERR("Extension section signature error");
1434*4882a593Smuzhiyun 		goto err;
1435*4882a593Smuzhiyun 	}
1436*4882a593Smuzhiyun 
1437*4882a593Smuzhiyun 	proj_id = rtb_get_fw_project_id(rtl->fw_buf + rtl->fw_len - 5);
1438*4882a593Smuzhiyun 
1439*4882a593Smuzhiyun #ifdef RTL_8703A_SUPPORT
1440*4882a593Smuzhiyun 	if (rtl->hci_ver == 0x4 && rtl->lmp_subver == ROM_LMP_8723b) {
1441*4882a593Smuzhiyun 		RS_INFO("HCI version = 0x4, IC is 8703A.");
1442*4882a593Smuzhiyun 	} else {
1443*4882a593Smuzhiyun 		RS_ERR("error: lmp_version %x, hci_version %x, project_id %x",
1444*4882a593Smuzhiyun 		       rtl->lmp_subver, rtl->hci_ver, project_id[proj_id]);
1445*4882a593Smuzhiyun 		goto err;
1446*4882a593Smuzhiyun 	}
1447*4882a593Smuzhiyun #else
1448*4882a593Smuzhiyun 	if (rtl->lmp_subver != ROM_LMP_8703b) {
1449*4882a593Smuzhiyun 		if (rtl->lmp_subver != project_id[proj_id]) {
1450*4882a593Smuzhiyun 			RS_ERR("lmp_subver %04x, project id %04x, mismatch\n",
1451*4882a593Smuzhiyun 			       rtl->lmp_subver, project_id[proj_id]);
1452*4882a593Smuzhiyun 			goto err;
1453*4882a593Smuzhiyun 		}
1454*4882a593Smuzhiyun 	} else {
1455*4882a593Smuzhiyun 		if (rtb_cfg.patch_ent->proj_id != project_id[proj_id]) {
1456*4882a593Smuzhiyun 			RS_ERR("proj_id %04x, version %04x from firmware "
1457*4882a593Smuzhiyun 			       "project_id[%u], mismatch",
1458*4882a593Smuzhiyun 			       rtb_cfg.patch_ent->proj_id,
1459*4882a593Smuzhiyun 			       project_id[proj_id], proj_id);
1460*4882a593Smuzhiyun 			goto err;
1461*4882a593Smuzhiyun 		}
1462*4882a593Smuzhiyun 	}
1463*4882a593Smuzhiyun #endif
1464*4882a593Smuzhiyun 
1465*4882a593Smuzhiyun 	/* Entry is allocated dynamically. It should be freed later in the
1466*4882a593Smuzhiyun 	 * function.
1467*4882a593Smuzhiyun 	 */
1468*4882a593Smuzhiyun 	entry = rtb_get_patch_entry();
1469*4882a593Smuzhiyun 
1470*4882a593Smuzhiyun 	if (entry) {
1471*4882a593Smuzhiyun 		len = entry->patch_len + rtl->config_len;
1472*4882a593Smuzhiyun 	} else {
1473*4882a593Smuzhiyun 		RS_ERR("Can't find the patch entry");
1474*4882a593Smuzhiyun 		goto err;
1475*4882a593Smuzhiyun 	}
1476*4882a593Smuzhiyun 
1477*4882a593Smuzhiyun 	buf = malloc(len);
1478*4882a593Smuzhiyun 	if (!buf) {
1479*4882a593Smuzhiyun 		RS_ERR("%s: Can't alloc memory for fwc, %s", __func__,
1480*4882a593Smuzhiyun 		       strerror(errno));
1481*4882a593Smuzhiyun 		free(entry);
1482*4882a593Smuzhiyun 		goto err;
1483*4882a593Smuzhiyun 	} else {
1484*4882a593Smuzhiyun 		memcpy(buf, rtl->fw_buf + entry->soffset, entry->patch_len);
1485*4882a593Smuzhiyun 		memcpy(buf + entry->patch_len - 4, &patch->fw_version, 4);
1486*4882a593Smuzhiyun 
1487*4882a593Smuzhiyun 		if (rtl->config_len)
1488*4882a593Smuzhiyun 			memcpy(buf + entry->patch_len, rtl->config_buf,
1489*4882a593Smuzhiyun 			       rtl->config_len);
1490*4882a593Smuzhiyun 		rtl->dl_fw_flag = 1;
1491*4882a593Smuzhiyun 		*rlen = len;
1492*4882a593Smuzhiyun 	}
1493*4882a593Smuzhiyun 
1494*4882a593Smuzhiyun 	RS_INFO("FW %s exists, Config file %s exists",
1495*4882a593Smuzhiyun 		(rtl->fw_len > 0) ? "" : "not",
1496*4882a593Smuzhiyun 		(rtl->config_len > 0) ? "" : "not");
1497*4882a593Smuzhiyun 
1498*4882a593Smuzhiyun 	free(entry);
1499*4882a593Smuzhiyun 	return buf;
1500*4882a593Smuzhiyun 
1501*4882a593Smuzhiyun err:
1502*4882a593Smuzhiyun 	rtl->dl_fw_flag = 0;
1503*4882a593Smuzhiyun 	*rlen = 0;
1504*4882a593Smuzhiyun 	return NULL;
1505*4882a593Smuzhiyun }
1506*4882a593Smuzhiyun 
1507