1
0

wlan_cfg.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2018-08-06 tyx the first version
  9. */
  10. #include <rtthread.h>
  11. #include <wlan_cfg.h>
  12. #define DBG_TAG "WLAN.cfg"
  13. #ifdef RT_WLAN_CFG_DEBUG
  14. #define DBG_LVL DBG_LOG
  15. #else
  16. #define DBG_LVL DBG_INFO
  17. #endif /* RT_WLAN_CFG_DEBUG */
  18. #include <rtdbg.h>
  19. #ifdef RT_WLAN_CFG_ENABLE
  20. #define WLAN_CFG_LOCK() (rt_mutex_take(&cfg_mutex, RT_WAITING_FOREVER))
  21. #define WLAN_CFG_UNLOCK() (rt_mutex_release(&cfg_mutex))
  22. #if RT_WLAN_CFG_INFO_MAX < 1
  23. #error "The minimum configuration is 1"
  24. #endif
  25. struct cfg_save_info_head
  26. {
  27. rt_uint32_t magic;
  28. rt_uint32_t len;
  29. rt_uint32_t num;
  30. rt_uint32_t crc;
  31. };
  32. struct rt_wlan_cfg_des
  33. {
  34. rt_uint32_t num;
  35. struct rt_wlan_cfg_info *cfg_info;
  36. };
  37. static struct rt_wlan_cfg_des *cfg_cache;
  38. static const struct rt_wlan_cfg_ops *cfg_ops;
  39. static struct rt_mutex cfg_mutex;
  40. /*
  41. * CRC16_CCITT
  42. */
  43. static rt_uint16_t rt_wlan_cal_crc(rt_uint8_t *buff, int len)
  44. {
  45. rt_uint16_t wCRCin = 0x0000;
  46. rt_uint16_t wCPoly = 0x1021;
  47. rt_uint8_t wChar = 0;
  48. while (len--)
  49. {
  50. int i;
  51. wChar = *(buff++);
  52. wCRCin ^= (wChar << 8);
  53. for (i = 0; i < 8; i++)
  54. {
  55. if (wCRCin & 0x8000)
  56. wCRCin = (wCRCin << 1) ^ wCPoly;
  57. else
  58. wCRCin = wCRCin << 1;
  59. }
  60. }
  61. return wCRCin;
  62. }
  63. void rt_wlan_cfg_init(void)
  64. {
  65. /* init cache memory */
  66. if (cfg_cache == RT_NULL)
  67. {
  68. cfg_cache = rt_malloc(sizeof(struct rt_wlan_cfg_des));
  69. if (cfg_cache != RT_NULL)
  70. {
  71. rt_memset(cfg_cache, 0, sizeof(struct rt_wlan_cfg_des));
  72. }
  73. /* init mutex lock */
  74. rt_mutex_init(&cfg_mutex, "wlan_cfg", RT_IPC_FLAG_PRIO);
  75. }
  76. }
  77. void rt_wlan_cfg_set_ops(const struct rt_wlan_cfg_ops *ops)
  78. {
  79. rt_wlan_cfg_init();
  80. WLAN_CFG_LOCK();
  81. /* save ops pointer */
  82. cfg_ops = ops;
  83. WLAN_CFG_UNLOCK();
  84. }
  85. /* save config data */
  86. rt_err_t rt_wlan_cfg_cache_save(void)
  87. {
  88. rt_err_t err = RT_EOK;
  89. struct cfg_save_info_head *info_pkg;
  90. int len = 0;
  91. if ((cfg_ops == RT_NULL) || (cfg_ops->write_cfg == RT_NULL))
  92. return RT_EOK;
  93. WLAN_CFG_LOCK();
  94. len = sizeof(struct cfg_save_info_head) + sizeof(struct rt_wlan_cfg_info) * cfg_cache->num;
  95. info_pkg = rt_malloc(len);
  96. if (info_pkg == RT_NULL)
  97. {
  98. WLAN_CFG_UNLOCK();
  99. return -RT_ENOMEM;
  100. }
  101. info_pkg->magic = RT_WLAN_CFG_MAGIC;
  102. info_pkg->len = len;
  103. info_pkg->num = cfg_cache->num;
  104. /* CRC */
  105. info_pkg->crc = rt_wlan_cal_crc((rt_uint8_t *)cfg_cache->cfg_info, sizeof(struct rt_wlan_cfg_info) * cfg_cache->num);
  106. rt_memcpy(((rt_uint8_t *)info_pkg) + sizeof(struct cfg_save_info_head),
  107. cfg_cache->cfg_info, sizeof(struct rt_wlan_cfg_info) * cfg_cache->num);
  108. if (cfg_ops->write_cfg(info_pkg, len) != len)
  109. err = -RT_ERROR;
  110. rt_free(info_pkg);
  111. WLAN_CFG_UNLOCK();
  112. return err;
  113. }
  114. rt_err_t rt_wlan_cfg_cache_refresh(void)
  115. {
  116. int len = 0, i, j;
  117. struct cfg_save_info_head *head;
  118. void *data;
  119. struct rt_wlan_cfg_info *t_info, *cfg_info;
  120. rt_uint32_t crc;
  121. rt_bool_t equal_flag;
  122. /* cache is full! exit */
  123. if (cfg_cache == RT_NULL || cfg_cache->num >= RT_WLAN_CFG_INFO_MAX)
  124. return -RT_ERROR;
  125. /* check callback */
  126. if ((cfg_ops == RT_NULL) ||
  127. (cfg_ops->get_len == RT_NULL) ||
  128. (cfg_ops->read_cfg == RT_NULL))
  129. return -RT_ERROR;
  130. WLAN_CFG_LOCK();
  131. /* get data len */
  132. if ((len = cfg_ops->get_len()) <= 0)
  133. {
  134. WLAN_CFG_UNLOCK();
  135. return -RT_ERROR;
  136. }
  137. head = rt_malloc(len);
  138. if (head == RT_NULL)
  139. {
  140. WLAN_CFG_UNLOCK();
  141. return -RT_ERROR;
  142. }
  143. /* get data */
  144. if (cfg_ops->read_cfg(head, len) != len)
  145. {
  146. rt_free(head);
  147. WLAN_CFG_UNLOCK();
  148. return -RT_ERROR;
  149. }
  150. /* get config */
  151. data = ((rt_uint8_t *)head) + sizeof(struct cfg_save_info_head);
  152. crc = rt_wlan_cal_crc((rt_uint8_t *)data, len - sizeof(struct cfg_save_info_head));
  153. LOG_D("head->magic:0x%08x RT_WLAN_CFG_MAGIC:0x%08x", head->magic, RT_WLAN_CFG_MAGIC);
  154. LOG_D("head->len:%d len:%d", head->len, len);
  155. LOG_D("head->num:%d num:%d", head->num, (len - sizeof(struct cfg_save_info_head)) / sizeof(struct rt_wlan_cfg_info));
  156. LOG_D("hred->crc:0x%04x crc:0x%04x", head->crc, crc);
  157. /* check */
  158. if ((head->magic != RT_WLAN_CFG_MAGIC) ||
  159. (head->len != len) ||
  160. (head->num != (len - sizeof(struct cfg_save_info_head)) / sizeof(struct rt_wlan_cfg_info)) ||
  161. (head->crc != crc))
  162. {
  163. rt_free(head);
  164. WLAN_CFG_UNLOCK();
  165. return -RT_ERROR;
  166. }
  167. /* remove duplicate config */
  168. cfg_info = (struct rt_wlan_cfg_info *)data;
  169. for (i = 0; i < head->num; i++)
  170. {
  171. equal_flag = RT_FALSE;
  172. for (j = 0; j < cfg_cache->num; j++)
  173. {
  174. if ((cfg_cache->cfg_info[j].info.ssid.len == cfg_info[i].info.ssid.len) &&
  175. (rt_memcmp(&cfg_cache->cfg_info[j].info.ssid.val[0], &cfg_info[i].info.ssid.val[0],
  176. cfg_cache->cfg_info[j].info.ssid.len) == 0) &&
  177. (rt_memcmp(&cfg_cache->cfg_info[j].info.bssid[0], &cfg_info[i].info.bssid[0], RT_WLAN_BSSID_MAX_LENGTH) == 0))
  178. {
  179. equal_flag = RT_TRUE;
  180. break;
  181. }
  182. }
  183. if (cfg_cache->num >= RT_WLAN_CFG_INFO_MAX)
  184. {
  185. break;
  186. }
  187. if (equal_flag == RT_FALSE)
  188. {
  189. t_info = rt_realloc(cfg_cache->cfg_info, sizeof(struct rt_wlan_cfg_info) * (cfg_cache->num + 1));
  190. if (t_info == RT_NULL)
  191. {
  192. rt_free(head);
  193. WLAN_CFG_UNLOCK();
  194. return -RT_ERROR;
  195. }
  196. cfg_cache->cfg_info = t_info;
  197. cfg_cache->cfg_info[cfg_cache->num] = cfg_info[i];
  198. cfg_cache->num ++;
  199. }
  200. }
  201. rt_free(head);
  202. WLAN_CFG_UNLOCK();
  203. return RT_EOK;
  204. }
  205. int rt_wlan_cfg_get_num(void)
  206. {
  207. rt_wlan_cfg_init();
  208. return cfg_cache->num;
  209. }
  210. int rt_wlan_cfg_read(struct rt_wlan_cfg_info *cfg_info, int num)
  211. {
  212. rt_wlan_cfg_init();
  213. if ((cfg_info == RT_NULL) || (num <= 0))
  214. return 0;
  215. /* copy data */
  216. WLAN_CFG_LOCK();
  217. num = cfg_cache->num > num ? num : cfg_cache->num;
  218. rt_memcpy(&cfg_cache->cfg_info[0], cfg_info, sizeof(struct rt_wlan_cfg_info) * num);
  219. WLAN_CFG_UNLOCK();
  220. return num;
  221. }
  222. rt_err_t rt_wlan_cfg_save(struct rt_wlan_cfg_info *cfg_info)
  223. {
  224. rt_err_t err = RT_EOK;
  225. struct rt_wlan_cfg_info *t_info;
  226. int idx = -1, i = 0;
  227. rt_wlan_cfg_init();
  228. /* parameter check */
  229. if ((cfg_info == RT_NULL) || (cfg_info->info.ssid.len == 0))
  230. {
  231. return -RT_EINVAL;
  232. }
  233. /* if (iteam == cache) exit */
  234. WLAN_CFG_LOCK();
  235. for (i = 0; i < cfg_cache->num; i++)
  236. {
  237. if ((cfg_cache->cfg_info[i].info.ssid.len == cfg_info->info.ssid.len) &&
  238. (rt_memcmp(&cfg_cache->cfg_info[i].info.ssid.val[0], &cfg_info->info.ssid.val[0],
  239. cfg_cache->cfg_info[i].info.ssid.len) == 0) &&
  240. (rt_memcmp(&cfg_cache->cfg_info[i].info.bssid[0], &cfg_info->info.bssid[0], RT_WLAN_BSSID_MAX_LENGTH) == 0))
  241. {
  242. idx = i;
  243. break;
  244. }
  245. }
  246. if ((idx == 0) && (cfg_cache->cfg_info[i].key.len == cfg_info->key.len) &&
  247. (rt_memcmp(&cfg_cache->cfg_info[i].key.val[0], &cfg_info->key.val[0], cfg_info->key.len) == 0))
  248. {
  249. WLAN_CFG_UNLOCK();
  250. return RT_EOK;
  251. }
  252. /* not find iteam with cache, Add iteam to the head */
  253. if ((idx == -1) && (cfg_cache->num < RT_WLAN_CFG_INFO_MAX))
  254. {
  255. t_info = rt_realloc(cfg_cache->cfg_info, sizeof(struct rt_wlan_cfg_info) * (cfg_cache->num + 1));
  256. if (t_info == RT_NULL)
  257. {
  258. WLAN_CFG_UNLOCK();
  259. return -RT_ENOMEM;
  260. }
  261. cfg_cache->cfg_info = t_info;
  262. cfg_cache->num ++;
  263. }
  264. /* move cache info */
  265. i = (i >= RT_WLAN_CFG_INFO_MAX ? RT_WLAN_CFG_INFO_MAX - 1 : i);
  266. for (; i; i--)
  267. {
  268. cfg_cache->cfg_info[i] = cfg_cache->cfg_info[i - 1];
  269. }
  270. /* add iteam to head */
  271. cfg_cache->cfg_info[i] = *cfg_info;
  272. WLAN_CFG_UNLOCK();
  273. /* save info to flash */
  274. err = rt_wlan_cfg_cache_save();
  275. return err;
  276. }
  277. int rt_wlan_cfg_read_index(struct rt_wlan_cfg_info *cfg_info, int index)
  278. {
  279. rt_wlan_cfg_init();
  280. if ((cfg_info == RT_NULL) || (index < 0))
  281. return 0;
  282. WLAN_CFG_LOCK();
  283. if (index >= cfg_cache->num)
  284. {
  285. WLAN_CFG_UNLOCK();
  286. return 0;
  287. }
  288. /* copy data */
  289. *cfg_info = cfg_cache->cfg_info[index];
  290. WLAN_CFG_UNLOCK();
  291. return 1;
  292. }
  293. int rt_wlan_cfg_delete_index(int index)
  294. {
  295. struct rt_wlan_cfg_info *cfg_info;
  296. int i;
  297. rt_wlan_cfg_init();
  298. if (index < 0)
  299. return -1;
  300. WLAN_CFG_LOCK();
  301. if (index >= cfg_cache->num)
  302. {
  303. WLAN_CFG_UNLOCK();
  304. return -1;
  305. }
  306. /* malloc new mem */
  307. cfg_info = rt_malloc(sizeof(struct rt_wlan_cfg_info) * (cfg_cache->num - 1));
  308. if (cfg_info == RT_NULL)
  309. {
  310. WLAN_CFG_UNLOCK();
  311. return -1;
  312. }
  313. /* copy data to new mem */
  314. for (i = 0; i < cfg_cache->num; i++)
  315. {
  316. if (i < index)
  317. {
  318. cfg_info[i] = cfg_cache->cfg_info[i];
  319. }
  320. else if (i > index)
  321. {
  322. cfg_info[i - 1] = cfg_cache->cfg_info[i];
  323. }
  324. }
  325. rt_free(cfg_cache->cfg_info);
  326. cfg_cache->cfg_info = cfg_info;
  327. cfg_cache->num --;
  328. WLAN_CFG_UNLOCK();
  329. return 0;
  330. }
  331. void rt_wlan_cfg_delete_all(void)
  332. {
  333. rt_wlan_cfg_init();
  334. /* delete all iteam */
  335. WLAN_CFG_LOCK();
  336. cfg_cache->num = 0;
  337. rt_free(cfg_cache->cfg_info);
  338. cfg_cache->cfg_info = RT_NULL;
  339. WLAN_CFG_UNLOCK();
  340. }
  341. void rt_wlan_cfg_dump(void)
  342. {
  343. int index = 0;
  344. struct rt_wlan_info *info;
  345. struct rt_wlan_key *key;
  346. char *security;
  347. rt_wlan_cfg_init();
  348. rt_kprintf(" SSID PASSWORD MAC security chn\n");
  349. rt_kprintf("------------------------------- ------------------------------- ----------------- -------------- ---\n");
  350. for (index = 0; index < cfg_cache->num; index ++)
  351. {
  352. info = &cfg_cache->cfg_info[index].info;
  353. key = &cfg_cache->cfg_info[index].key;
  354. if (info->ssid.len)
  355. rt_kprintf("%-32.32s", &info->ssid.val[0]);
  356. else
  357. rt_kprintf("%-32.32s", " ");
  358. if (key->len)
  359. rt_kprintf("%-32.32s", &key->val[0]);
  360. else
  361. rt_kprintf("%-32.32s", " ");
  362. rt_kprintf("%02x:%02x:%02x:%02x:%02x:%02x ",
  363. info->bssid[0],
  364. info->bssid[1],
  365. info->bssid[2],
  366. info->bssid[3],
  367. info->bssid[4],
  368. info->bssid[5]
  369. );
  370. switch (info->security)
  371. {
  372. case SECURITY_OPEN:
  373. security = "OPEN";
  374. break;
  375. case SECURITY_WEP_PSK:
  376. security = "WEP_PSK";
  377. break;
  378. case SECURITY_WEP_SHARED:
  379. security = "WEP_SHARED";
  380. break;
  381. case SECURITY_WPA_TKIP_PSK:
  382. security = "WPA_TKIP_PSK";
  383. break;
  384. case SECURITY_WPA_AES_PSK:
  385. security = "WPA_AES_PSK";
  386. break;
  387. case SECURITY_WPA2_AES_PSK:
  388. security = "WPA2_AES_PSK";
  389. break;
  390. case SECURITY_WPA2_TKIP_PSK:
  391. security = "WPA2_TKIP_PSK";
  392. break;
  393. case SECURITY_WPA2_MIXED_PSK:
  394. security = "WPA2_MIXED_PSK";
  395. break;
  396. case SECURITY_WPS_OPEN:
  397. security = "WPS_OPEN";
  398. break;
  399. case SECURITY_WPS_SECURE:
  400. security = "WPS_SECURE";
  401. break;
  402. default:
  403. security = "UNKNOWN";
  404. break;
  405. }
  406. rt_kprintf("%-14.14s ", security);
  407. rt_kprintf("%3d \n", info->channel);
  408. }
  409. }
  410. #endif