wlan_cfg.c 12 KB

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