uparam.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819
  1. #include "uparam.h"
  2. #include <fal.h>
  3. #include <finsh.h>
  4. #define UPARAM_DEBUG
  5. #define UPARAM_FINSH
  6. #define LOG_TAG "uparam"
  7. #ifdef UPARAM_DEBUG
  8. #define LOG_LVL LOG_LVL_DBG
  9. #else
  10. #define LOG_LVL LOG_LVL_WARNING
  11. #endif
  12. #include <ulog.h>
  13. #define DEFAULT_PRA_PART "param"
  14. static char *praram_partition = DEFAULT_PRA_PART;
  15. static const struct fal_partition *par_part = RT_NULL;
  16. /* 写数据覆盖保护,必须等读取后才能写入 */
  17. static int8_t write_protect = 0;
  18. /* 参数表 */
  19. static param_struct *ls = 0;
  20. /* 参数表在内存的索引 */
  21. static uint32_t param_index = 0;
  22. /* 保存到flash的结构头部信息 */
  23. static param_header_struct param_header;
  24. /**
  25. * @brief uparam_add_list
  26. * @note 添加参数表
  27. * @param *list_address: 参数表地址
  28. * @param list_size: 包含参数的个数
  29. * @retval
  30. */
  31. rt_err_t uparam_add_list(param_list *list_address, uint16_t list_size)
  32. {
  33. param_list *pa_this;
  34. //先遍历一下参数表是否已经添加过
  35. for (int i = 0; i < param_index; i++)
  36. {
  37. pa_this = (param_list *)ls[i].par_list_add;
  38. if (pa_this == list_address)
  39. {
  40. LOG_E("param list is exist, address: 0x%X", (uint32_t)list_address);
  41. return RT_ERROR;
  42. }
  43. }
  44. //分配内存
  45. param_struct *new_ls = (param_struct *)rt_realloc(ls, (param_index + 1) * sizeof(param_struct));
  46. if (new_ls != RT_NULL)
  47. {
  48. ls = new_ls;
  49. ls[param_index].par_list_add = (uint32_t)list_address;
  50. ls[param_index].par_list_size = list_size;
  51. //按位标记参数是否有效
  52. uint16_t bit_num = (list_size % 8 == 0) ? (list_size / 8) : (list_size / 8 + 1);
  53. ls[param_index].read_valid = (uint8_t *)rt_malloc(bit_num);
  54. memset(ls[param_index].read_valid, 0, bit_num);
  55. param_index++;
  56. for (int i = 0; i < list_size; i++)
  57. {
  58. pa_this = list_address + i;
  59. param_header.size.u32 += pa_this->size;
  60. LOG_D("add param suc,name: %s , address: 0x%X, size: %d", pa_this->name, pa_this->address, pa_this->size);
  61. }
  62. param_header.cnt.u32 += list_size;
  63. LOG_D("add param list success, list size: %d, data size total: %d", list_size, param_header.size.u32);
  64. return RT_EOK;
  65. }
  66. else
  67. {
  68. rt_free(ls);
  69. LOG_E("uparam realloc memory failed");
  70. }
  71. return RT_ERROR;
  72. }
  73. /**
  74. * @brief cal_crc
  75. * @note 计算校验值
  76. * @param start: 起始值
  77. * @param *buff:
  78. * @param size:
  79. * @retval
  80. */
  81. static uint8_t cal_crc(uint8_t start, uint8_t *buff, uint16_t size)
  82. {
  83. uint8_t check = start;
  84. for (int i = 0; i < size; i++)
  85. {
  86. check ^= buff[i];
  87. }
  88. return check;
  89. }
  90. /**
  91. * @brief uparam_readall
  92. * @note 从flash读取所有参数到内存
  93. * @retval
  94. */
  95. static uint16_t uparam_readall()
  96. {
  97. uint32_t offset = 0;
  98. param_header_struct header;
  99. uint8_t temp[260];
  100. uint16_t read_num = 0; //读成功的数量
  101. param_p pa_this; //当前参数信息
  102. param_p *pa_next; //指向下一组参数的信息
  103. uint8_t rsize = sizeof(param_header_struct) + sizeof(param_p);
  104. write_protect++;
  105. //read header
  106. if (fal_partition_read(par_part, 0, temp, rsize) != rsize)
  107. {
  108. LOG_E("Uparam read header failed!");
  109. return 0;
  110. }
  111. offset += rsize;
  112. memcpy(&header, temp, sizeof(param_header_struct));
  113. pa_next = (param_p *)(temp + sizeof(param_header_struct));
  114. //检查header是否有效
  115. uint8_t check = cal_crc(0x55, header.cnt.u8, 4);
  116. check = cal_crc(check, header.size.u8, 4);
  117. if (header.header != 0x55 || check != header.crc)
  118. {
  119. LOG_E("Uparam header invalid!");
  120. return 0;
  121. }
  122. LOG_D("read param number: %d", header.cnt.u32);
  123. if (header.cnt.u32 != param_header.cnt.u32)
  124. {
  125. LOG_W("param number is changed, the exist is %d", param_header.cnt.u32);
  126. }
  127. //循环读取每组参数
  128. for (int i = 0; i < header.cnt.u32; i++)
  129. {
  130. //读数据段,长度为数据长度 + 1CRC + 下一组参数信息
  131. rsize = pa_next->size + 1 + sizeof(param_p);
  132. //这里读取了数据后pa指向的size是下一组数据的size,需要先记录一下当前数据的size不然crc会错误
  133. pa_this.size = pa_next->size;
  134. pa_this.address = pa_next->address;
  135. if (fal_partition_read(par_part, offset, temp, rsize) != rsize)
  136. {
  137. LOG_E("Uparam read data failed!");
  138. return 0;
  139. }
  140. offset += rsize;
  141. //检查CRC
  142. check = cal_crc(0x55, temp, pa_this.size);
  143. LOG_D("read param address 0x%X, size:%d, crc:%X, calc:%X", pa_this.address, pa_this.size, check, temp[pa_this.size]);
  144. if (check != temp[pa_this.size])
  145. {
  146. LOG_E("Uparam check data failed! Index:%d, Address:%X", i, (uint32_t)pa_this.address);
  147. return 0;
  148. }
  149. //读成功了,对内存赋值
  150. //不要直接赋值,先检查一下是否在参数表里面存在,防止数据地址被修改后写入未知地址
  151. param_list *pa_list_t;
  152. param_p *pa_t;
  153. int li;
  154. int idx;
  155. for (li = 0; li < param_index; li++)
  156. {
  157. pa_list_t = (param_list *)ls[li].par_list_add;
  158. for (idx = 0; idx < ls[li].par_list_size; idx++)
  159. {
  160. pa_t = (param_p *)&pa_list_t[idx];
  161. //是否存在 并且 size相同
  162. if (pa_t->address == pa_this.address)
  163. {
  164. if (pa_t->size == pa_this.size)
  165. {
  166. //对成功读出的数据标记一下
  167. ls[li].read_valid[idx / 8] |= 1 << idx;
  168. //对数据赋读出的值
  169. memcpy((void *)(pa_this.address), temp, pa_this.size);
  170. read_num++;
  171. break;
  172. }
  173. else
  174. {
  175. LOG_W("target size is different, size: %d", pa_t->size);
  176. }
  177. }
  178. }
  179. //找到了
  180. if (idx != ls[li].par_list_size)
  181. {
  182. break;
  183. }
  184. }
  185. //未找到此参数
  186. if (li == param_index)
  187. {
  188. LOG_W("param is not exist, address: 0x%X ,read size: %d", pa_this.address, pa_this.size);
  189. }
  190. pa_next = (param_p *)(temp + pa_this.size + 1);
  191. }
  192. LOG_D("read param success count: %d", read_num);
  193. return read_num;
  194. }
  195. /**
  196. * @brief uparam_writeall
  197. * @note 将参数表里面所有参数写入到flash
  198. * @retval
  199. */
  200. static uint16_t uparam_writeall()
  201. {
  202. uint32_t offset = 0;
  203. uint8_t temp[270];
  204. param_list *pa_list;
  205. param_p *pa;
  206. uint8_t wsize = sizeof(param_header_struct);
  207. if (write_protect < 1)
  208. {
  209. LOG_E("Uparam should read once before write!");
  210. return 0;
  211. }
  212. //计算要写入的总字节数 header+ 数据头+数据+校验
  213. int allsize = wsize + sizeof(param_p) * param_header.cnt.u32 + param_header.size.u32 + param_header.cnt.u32;
  214. LOG_D("Uparam write, cnt: %d, data size: %d, all size: %d!", param_header.cnt.u32, param_header.size.u32, allsize);
  215. //擦除flash
  216. fal_partition_erase(par_part, 0, allsize);
  217. offset = wsize;
  218. //循环写入所有参数
  219. for (int li = 0; li < param_index; li++)
  220. {
  221. pa_list = (param_list *)ls[li].par_list_add;
  222. for (int i = 0; i < ls[li].par_list_size; i++)
  223. {
  224. pa = (param_p *)&pa_list[i];
  225. memcpy(temp, pa, sizeof(param_p));
  226. //准备数据
  227. memcpy(temp + sizeof(param_p), (void *)(pa->address), pa->size);
  228. temp[sizeof(param_p) + pa->size] = cal_crc(0x55, (uint8_t *)(pa->address), pa->size);
  229. LOG_D("write [%-16s], size:%d, crc:%X", pa_list[i].name, pa->size, temp[sizeof(param_p) + pa->size]);
  230. //写入 数据加一字节校验
  231. wsize = sizeof(param_p) + pa->size + 1;
  232. if (fal_partition_write(par_part, offset, temp, wsize) != wsize)
  233. {
  234. LOG_E("Uparam write data failed!");
  235. return 0;
  236. }
  237. offset += wsize;
  238. }
  239. }
  240. //最后写入header
  241. wsize = sizeof(param_header_struct);
  242. param_header.header = 0x55;
  243. param_header.crc = cal_crc(0x55, param_header.cnt.u8, 4);
  244. param_header.crc = cal_crc(param_header.crc, param_header.size.u8, 4);
  245. if (fal_partition_write(par_part, 0, (uint8_t *)&param_header, wsize) != wsize)
  246. {
  247. LOG_E("Uparam write header failed!");
  248. return 0;
  249. }
  250. LOG_D("Uparam write param, cnt: %d, write size: %d!", param_header.cnt.u32, offset - wsize);
  251. return param_header.cnt.u32;
  252. }
  253. /**
  254. * @brief
  255. * @note
  256. * @retval
  257. */
  258. uint16_t uparam_flush()
  259. {
  260. return uparam_writeall();
  261. }
  262. /**
  263. * @brief uparam_default
  264. * @note 还原参数到默认值
  265. * @retval None
  266. */
  267. static void uparam_default()
  268. {
  269. //对新加入的数据执行默认操作
  270. LOG_D("Reset changed param to default!");
  271. //遍历没有读出的数据
  272. param_list *pa_list_t;
  273. for (int li = 0; li < param_index; li++)
  274. {
  275. pa_list_t = (param_list *)ls[li].par_list_add;
  276. for (int i = 0; i < ls[li].par_list_size; i++)
  277. {
  278. if ((ls[li].read_valid[i / 8] & (1 << i)) == 0)
  279. {
  280. //
  281. LOG_D("reset param [%-16s], address: 0x%X, size:%d", pa_list_t[i].name, pa_list_t[i].address, pa_list_t[i].size);
  282. if (pa_list_t[i].default_fun != RT_NULL)
  283. {
  284. pa_list_t[i].default_fun(pa_list_t[i].address, pa_list_t[i].size);
  285. }
  286. else
  287. {
  288. LOG_E("reset error [%-16s], address: 0x%X, default_fun is null", pa_list_t[i].name, pa_list_t[i].address);
  289. }
  290. }
  291. }
  292. }
  293. }
  294. /**
  295. * @brief 打印参数列表的描述信息
  296. * @note
  297. * @retval None
  298. */
  299. static void print_list_header()
  300. {
  301. rt_kprintf("\nFormat show as: f=float,d=int,u=uint,v*=vector(hex or float),s=str\r\n");
  302. rt_kprintf("Index Param Address Size Format Value\r\n");
  303. rt_kprintf("----- ---------- ---------- ---- ------ -----\r\n");
  304. }
  305. /**
  306. * @brief 打印单个参数
  307. * @note
  308. * @param *pa:
  309. * @param index:
  310. * @param offset: 数组类型的打印起始偏移
  311. * @retval None
  312. */
  313. static void print_element(param_p *pa, uint32_t index, uint32_t offset)
  314. {
  315. uint16_t len;
  316. char buff[64];
  317. char value[8];
  318. param_list *pa_list = (param_list *)pa;
  319. //打印信息
  320. rt_kprintf("%-5d %-16s 0x%-8X %-4d ", index, (const char *)pa_list->name,
  321. pa->address, pa->size);
  322. memset(buff, 0, sizeof(buff));
  323. memset(value, 0, sizeof(value));
  324. //打印数据
  325. if (pa_list->type[0] == 'f')
  326. {
  327. memcpy(value, (uint8_t *)pa->address, pa->size);
  328. len = sprintf(buff, "Float %.3f\r\n", *(float *)(value));
  329. }
  330. else if (pa_list->type[0] == 's')
  331. {
  332. len = sprintf(buff, "String %s\r\n", (char *)pa->address);
  333. }
  334. else if (pa_list->type[0] == 'd')
  335. {
  336. int64_t convert = 0;
  337. if (pa->size == 1)
  338. {
  339. convert = (int64_t)(*(int8_t *)(pa->address));
  340. }
  341. if (pa->size == 2)
  342. {
  343. convert = (int64_t)(*(int16_t *)(pa->address));
  344. }
  345. if (pa->size == 4)
  346. {
  347. convert = (int64_t)(*(int32_t *)(pa->address));
  348. }
  349. if (pa->size == 8)
  350. {
  351. convert = (int64_t)(*(int64_t *)(pa->address));
  352. }
  353. len = sprintf(buff, "Intger %lld\r\n", convert);
  354. }
  355. else if (pa_list->type[0] == 'u')
  356. {
  357. memcpy(value, (uint8_t *)pa->address, pa->size);
  358. len = sprintf(buff, "UIntger %lld\r\n", *(uint64_t *)(value));
  359. }
  360. else if (pa_list->type[0] == 'v')
  361. {
  362. //vector 格式,判断下输出形式
  363. if (pa_list->type[1] == 'b')
  364. {
  365. /**按单字节打印输出 */
  366. len = sprintf(buff, "V Byte ");
  367. //最长只打印5个数字
  368. for (int s = 0; s < pa->size && s < 5; s++)
  369. {
  370. len += sprintf(buff + len, "%02X ", *((uint8_t *)(pa->address) + offset + s));
  371. }
  372. }
  373. else if (pa_list->type[1] == 'w')
  374. {
  375. /**按双字节打印输出 */
  376. len = sprintf(buff, "V Word ");
  377. //最长只打印5个数字
  378. for (int s = 0; s < (pa->size / 2 - offset) && s < 5; s++)
  379. {
  380. len += sprintf(buff + len, "%04X ", *((uint16_t *)(pa->address) + offset + s));
  381. }
  382. }
  383. else if (pa_list->type[1] == 'd')
  384. {
  385. /**按四字节打印输出 */
  386. len = sprintf(buff, "V Dword ");
  387. //最长只打印5个数字
  388. for (int s = 0; s < (pa->size / 4 - offset) && s < 5; s++)
  389. {
  390. len += sprintf(buff + len, "%08X ", *((uint32_t *)(pa->address) + offset + s));
  391. }
  392. }
  393. else if (pa_list->type[1] == 'f')
  394. {
  395. /**按float打印输出 */
  396. len = sprintf(buff, "V Float ");
  397. //最长只打印5个数字
  398. for (int s = 0; s < (pa->size / 4 - offset) && s < 5; s++)
  399. {
  400. len += sprintf(buff + len, "%.3f ", *((float *)(pa->address) + offset + s));
  401. }
  402. }
  403. len += sprintf(buff + len, "\r\n");
  404. }
  405. rt_kprintf("%s", buff);
  406. }
  407. /**
  408. * @brief uparam_list
  409. * @note 打印参数
  410. * @retval None
  411. */
  412. static void uparam_list()
  413. {
  414. param_list *pa_list;
  415. param_p *pa;
  416. uint32_t index = 0;
  417. print_list_header();
  418. for (int li = 0; li < param_index; li++)
  419. {
  420. pa_list = (param_list *)ls[li].par_list_add;
  421. for (int i = 0; i < ls[li].par_list_size; i++)
  422. {
  423. pa = (param_p *)&pa_list[i];
  424. print_element(pa, index++, 0);
  425. }
  426. }
  427. }
  428. /**
  429. * @brief find_param_by_index
  430. * @note 通过索引找到参数
  431. * @param index:
  432. * @retval
  433. */
  434. static param_list *find_param_by_index(uint32_t index)
  435. {
  436. uint32_t pre_index = 0;
  437. param_list *pa_list;
  438. for (int li = 0; li < param_index; li++)
  439. {
  440. pa_list = (param_list *)ls[li].par_list_add;
  441. if (index - pre_index < ls[li].par_list_size)
  442. {
  443. return (pa_list + (index - pre_index));
  444. }
  445. pre_index += ls[li].par_list_size;
  446. }
  447. return NULL;
  448. }
  449. /**
  450. * @brief reset_param_by_index
  451. * @note 通过索引还原参数到默认值
  452. * @param index:
  453. * @retval None
  454. */
  455. static void reset_param_by_index(uint32_t index)
  456. {
  457. param_list *pa_list;
  458. pa_list = find_param_by_index(index);
  459. if (pa_list->default_fun != RT_NULL)
  460. {
  461. pa_list->default_fun(pa_list->address, pa_list->size);
  462. }
  463. }
  464. /**
  465. * @brief erase_all_param
  466. * @note 擦除所有参数。都会被还原到默认值
  467. * @retval None
  468. */
  469. static void erase_all_param(void)
  470. {
  471. //只需要擦除保存的header即可
  472. fal_partition_erase(par_part, 0, sizeof(param_header_struct));
  473. //清除参数读取标志
  474. for (int li = 0; li < param_index; li++)
  475. {
  476. uint16_t bit_num = (ls[li].par_list_size % 8 == 0) ? (ls[li].par_list_size / 8) : (ls[li].par_list_size / 8 + 1);
  477. memset(ls[li].read_valid, 0, bit_num);
  478. }
  479. //初始化参数到默认值
  480. uparam_default();
  481. }
  482. /**
  483. * @brief uparam_init
  484. * @note 初始化参数
  485. * @retval
  486. */
  487. static int uparam_init(void)
  488. {
  489. /* 寻找参数分区是否存在 */
  490. if ((par_part = fal_partition_find(praram_partition)) == RT_NULL)
  491. {
  492. LOG_E("Uparam init failed! Partition (%s) find error!", praram_partition);
  493. RT_ASSERT(RT_ERROR);
  494. return RT_ERROR;
  495. }
  496. if (param_index < 1)
  497. {
  498. LOG_W("params num is zero, nothing to be done!");
  499. return RT_EOK;
  500. }
  501. //加载参数失败
  502. if (uparam_readall() < param_header.cnt.u32)
  503. {
  504. LOG_W("Uparam read failed, reset to default!");
  505. //初始化参数到默认值
  506. uparam_default();
  507. //重新写入参数
  508. uparam_writeall();
  509. }
  510. return RT_EOK;
  511. }
  512. INIT_COMPONENT_EXPORT(uparam_init);
  513. //INIT_ENV_EXPORT
  514. #ifdef UPARAM_FINSH
  515. #include <finsh.h>
  516. static void par(uint8_t argc, char **argv)
  517. {
  518. #define CMD_LIST_INDEX 0
  519. #define CMD_RESET_INDEX 1
  520. #define CMD_SET_INDEX 2
  521. #define CMD_ERASE_INDEX 3
  522. #define CMD_FLUSH_INDEX 4
  523. #define CMD_RELOAD_INDEX 5
  524. const char *help_info[] =
  525. {
  526. "par list [*/index] [offset] - list all param",
  527. "par reset [index] - reset 'index' param to default",
  528. "par set [index] [offset] data1 ... dataN - set index data[offset] to param with the format",
  529. "par erase [yes] - erase all param and reset to default",
  530. "par flush - save all param to flash",
  531. "par reload - read all param to ram",
  532. };
  533. if (argc < 2)
  534. {
  535. rt_kprintf("Usage:\n");
  536. for (int i = 0; i < sizeof(help_info) / sizeof(char *); i++)
  537. {
  538. rt_kprintf("%s\n", help_info[i]);
  539. }
  540. rt_kprintf("\n");
  541. }
  542. else
  543. {
  544. const char *cmd = argv[1];
  545. uint32_t index;
  546. if (!strcmp(cmd, "list"))
  547. {
  548. if (argc == 2)
  549. {
  550. uparam_list();
  551. }
  552. else if (argc == 3)
  553. {
  554. index = strtoul(argv[2], NULL, 0);
  555. param_p *pa = (param_p *)find_param_by_index(index);
  556. print_element(pa, index, 0);
  557. }
  558. else if (argc == 4)
  559. {
  560. index = strtoul(argv[2], NULL, 0);
  561. int offset = strtoul(argv[3], NULL, 0);
  562. param_p *pa = (param_p *)find_param_by_index(index);
  563. print_element(pa, index, offset);
  564. }
  565. }
  566. else if (!strcmp(cmd, "reset"))
  567. {
  568. index = strtoul(argv[2], NULL, 0);
  569. if (index > param_header.cnt.u32)
  570. {
  571. rt_kprintf("index is over range\r\n");
  572. return;
  573. }
  574. rt_kprintf("reset param , index: %d\r\n", index);
  575. reset_param_by_index(index);
  576. }
  577. else if (!strcmp(cmd, "set"))
  578. {
  579. if (argc < 5)
  580. {
  581. rt_kprintf("Usage: %s.\n", help_info[CMD_SET_INDEX]);
  582. return;
  583. }
  584. //设置的数据不是vector的时候offset无效
  585. param_list *pa_list;
  586. uint32_t offset;
  587. index = strtoul(argv[2], NULL, 0);
  588. offset = strtoul(argv[3], NULL, 0);
  589. if (index > param_header.cnt.u32)
  590. {
  591. rt_kprintf("index is over range\r\n");
  592. return;
  593. }
  594. pa_list = find_param_by_index(index);
  595. if (pa_list->type[0] == 'f')
  596. {
  597. float value_f = (float)atof(argv[4]);
  598. *(float *)(pa_list->address) = value_f;
  599. char buff[32];
  600. memset(buff, 0, 32);
  601. sprintf(buff, "set index: %d, to value:%.5f \r\n", index, value_f);
  602. rt_kprintf("%s \r\n", buff);
  603. }
  604. else if (pa_list->type[0] == 'd')
  605. {
  606. long long value_d = atoll(argv[4]);
  607. unsigned long long value_ud = (long long)1<<63;
  608. value_ud = value_d & ~value_ud;
  609. memcpy((uint8_t *)pa_list->address, &value_ud, pa_list->size);
  610. if (value_d < 0)
  611. {
  612. *((uint8_t *)pa_list->address + pa_list->size - 1) |= 0x80;
  613. }
  614. char buff[32];
  615. memset(buff, 0, 32);
  616. sprintf(buff, "set index: %d, to value:%lld \r\n", index, value_d);
  617. rt_kprintf("%s \r\n", buff);
  618. }
  619. else if (pa_list->type[0] == 'u')
  620. {
  621. long long value_u = atoll(argv[4]);
  622. memcpy((uint8_t *)pa_list->address, &value_u, pa_list->size);
  623. char buff[32];
  624. memset(buff, 0, 32);
  625. sprintf(buff, "set index: %d, to value:%lld \r\n", index, value_u);
  626. rt_kprintf("%s \r\n", buff);
  627. }
  628. else if (pa_list->type[0] == 's')
  629. {
  630. char *value_s = argv[4];
  631. if (strlen(value_s) > pa_list->size)
  632. {
  633. rt_kprintf("input value is too long\n");
  634. return;
  635. }
  636. memset(pa_list->address, 0, pa_list->size);
  637. memcpy((uint8_t *)pa_list->address, value_s, strlen(value_s));
  638. rt_kprintf("set index: %d, to value:%s\r\n", index, value_s);
  639. }
  640. else if (pa_list->type[0] == 'v')
  641. {
  642. uint8_t input_size = argc - 4;
  643. int v;
  644. //参数是数据的情况下,需要判断offset是否超出地址范围
  645. if (pa_list->type[1] == 'b')
  646. {
  647. //按字节输入
  648. if (offset >= pa_list->size)
  649. {
  650. rt_kprintf("offset error,data size: %d\r\n", pa_list->size);
  651. return;
  652. }
  653. rt_kprintf("set index: %d, offset: %d, to value:", index, offset);
  654. for (uint8_t i = 0; i < input_size && i < pa_list->size; i++)
  655. {
  656. v = atoi(argv[4 + i]);
  657. *((uint8_t *)pa_list->address + offset + i) = (uint8_t)(v & 0xff);
  658. rt_kprintf("%d ", (uint8_t)v);
  659. }
  660. }
  661. else if (pa_list->type[1] == 'w')
  662. {
  663. //按2字节输入
  664. if (offset >= pa_list->size / 2)
  665. {
  666. rt_kprintf("offset error,data size: %d\r\n", pa_list->size);
  667. return;
  668. }
  669. rt_kprintf("set index: %d, offset: %d, to value:", index, offset);
  670. for (uint8_t i = 0; i < input_size && i < (pa_list->size / 2); i++)
  671. {
  672. v = atoi(argv[4 + i]);
  673. *((uint16_t *)pa_list->address + offset + i) = (uint16_t)(v & 0xffff);
  674. rt_kprintf("%d ", (uint16_t)v);
  675. }
  676. }
  677. else if (pa_list->type[1] == 'd')
  678. {
  679. //按4字节输入
  680. if (offset >= pa_list->size / 4)
  681. {
  682. rt_kprintf("offset error,data size: %d\r\n", pa_list->size);
  683. return;
  684. }
  685. rt_kprintf("set index: %d, offset: %d, to value:", index, offset);
  686. for (uint8_t i = 0; i < input_size && i < (pa_list->size / 4); i++)
  687. {
  688. v = atoi(argv[4 + i]);
  689. *((uint32_t *)pa_list->address + offset + i) = (uint32_t)v;
  690. rt_kprintf("%d ", (uint32_t)v);
  691. }
  692. }
  693. else if (pa_list->type[1] == 'f')
  694. {
  695. //按float输入
  696. if (offset >= pa_list->size / 4)
  697. {
  698. rt_kprintf("offset error,data size: %d\r\n", pa_list->size);
  699. return;
  700. }
  701. rt_kprintf("set index: %d, offset: %d, to value:", index, offset);
  702. char sbuff[128] = {0};
  703. char slen = 0;
  704. for (uint8_t i = 0; i < input_size && i < (pa_list->size / 4); i++)
  705. {
  706. double dv = atof(argv[4 + i]);
  707. *((float *)pa_list->address + offset + i) = (float)dv;
  708. slen += sprintf(sbuff + slen, "%.3f ", (float)dv);
  709. }
  710. rt_kprintf("%s", sbuff);
  711. }
  712. rt_kprintf("\r\n");
  713. }
  714. }
  715. else if (!strcmp(cmd, "erase"))
  716. {
  717. if (argc < 3)
  718. {
  719. rt_kprintf("Usage: %s.\n", help_info[CMD_ERASE_INDEX]);
  720. return;
  721. }
  722. if (!strcmp((const char *)argv[2], "yes"))
  723. {
  724. rt_kprintf("erase all param\r\n");
  725. erase_all_param();
  726. }
  727. else
  728. {
  729. rt_kprintf("input yes to erase\r\n");
  730. }
  731. }
  732. else if (!strcmp(cmd, "flush"))
  733. {
  734. uparam_flush();
  735. }
  736. else if (!strcmp(cmd, "reload"))
  737. {
  738. uparam_readall();
  739. }
  740. }
  741. }
  742. MSH_CMD_EXPORT(par, uparam operate);
  743. #endif