Non puoi selezionare più di 25 argomenti Gli argomenti devono iniziare con una lettera o un numero, possono includere trattini ('-') e possono essere lunghi fino a 35 caratteri.
 
 
 
 
 

456 righe
15 KiB

  1. #include "usmart.h"
  2. #include "usart.h"
  3. #include "sys.h"
  4. //////////////////////////////////////////////////////////////////////////////////
  5. //本程序只供学习使用,未经作者许可,不得用于其它任何用途
  6. //ALIENTEK STM32开发板
  7. //正点原子@ALIENTEK
  8. //技术论坛:www.openedv.com
  9. //版本:V3.1
  10. //版权所有,盗版必究。
  11. //Copyright(C) 正点原子 2011-2021
  12. //All rights reserved
  13. //********************************************************************************
  14. //升级说明
  15. //V1.4
  16. //增加了对参数为string类型的函数的支持.适用范围大大提高.
  17. //优化了内存占用,静态内存占用为79个字节@10个参数.动态适应数字及字符串长度
  18. //V2.0
  19. //1,修改了list指令,打印函数的完整表达式.
  20. //2,增加了id指令,打印每个函数的入口地址.
  21. //3,修改了参数匹配,支持函数参数的调用(输入入口地址).
  22. //4,增加了函数名长度宏定义.
  23. //V2.1 20110707
  24. //1,增加dec,hex两个指令,用于设置参数显示进制,及执行进制转换.
  25. //注:当dec,hex不带参数的时候,即设定显示参数进制.当后跟参数的时候,即执行进制转换.
  26. //如:"dec 0XFF" 则会将0XFF转为255,由串口返回.
  27. //如:"hex 100" 则会将100转为0X64,由串口返回
  28. //2,新增usmart_get_cmdname函数,用于获取指令名字.
  29. //V2.2 20110726
  30. //1,修正了void类型参数的参数统计错误.
  31. //2,修改数据显示格式默认为16进制.
  32. //V2.3 20110815
  33. //1,去掉了函数名后必须跟"("的限制.
  34. //2,修正了字符串参数中不能有"("的bug.
  35. //3,修改了函数默认显示参数格式的修改方式.
  36. //V2.4 20110905
  37. //1,修改了usmart_get_cmdname函数,增加最大参数长度限制.避免了输入错误参数时的死机现象.
  38. //2,增加USMART_ENTIM2_SCAN宏定义,用于配置是否使用TIM2定时执行scan函数.
  39. //V2.5 20110930
  40. //1,修改usmart_init函数为void usmart_init(u8 sysclk),可以根据系统频率自动设定扫描时间.(固定100ms)
  41. //2,去掉了usmart_init函数中的uart_init函数,串口初始化必须在外部初始化,方便用户自行管理.
  42. //V2.6 20111009
  43. //1,增加了read_addr和write_addr两个函数.可以利用这两个函数读写内部任意地址(必须是有效地址).更加方便调试.
  44. //2,read_addr和write_addr两个函数可以通过设置USMART_USE_WRFUNS为来使能和关闭.
  45. //3,修改了usmart_strcmp,使其规范化.
  46. //V2.7 20111024
  47. //1,修正了返回值16进制显示时不换行的bug.
  48. //2,增加了函数是否有返回值的判断,如果没有返回值,则不会显示.有返回值时才显示其返回值.
  49. //V2.8 20111116
  50. //1,修正了list等不带参数的指令发送后可能导致死机的bug.
  51. //V2.9 20120917
  52. //1,修改了形如:void*xxx(void)类型函数不能识别的bug。
  53. //V3.0 20130425
  54. //1,新增了字符串参数对转义符的支持。
  55. //V3.1 20131120
  56. //1,增加runtime系统指令,可以用于统计函数执行时间.
  57. //用法:
  58. //发送:runtime 1 ,则开启函数执行时间统计功能
  59. //发送:runtime 0 ,则关闭函数执行时间统计功能
  60. ///runtime统计功能,必须设置:USMART_ENTIMX_SCAN 为1,才可以使用!!
  61. //V3.2 20140828
  62. //1,修改usmart_get_aparm函数,加入+/-符号的支持
  63. //2,修改usmart_str2num函数,支持负数转换
  64. //V3.3 20160506
  65. //修正usmart_exe函数在USMART_ENTIMX_SCAN为0的时候,报错的bug
  66. /////////////////////////////////////////////////////////////////////////////////////
  67. TIM_HandleTypeDef TIM4_Handler; //定时器句柄
  68. //系统命令
  69. u8 *sys_cmd_tab[]=
  70. {
  71. "?",
  72. "help",
  73. "list",
  74. "id",
  75. "hex",
  76. "dec",
  77. "runtime",
  78. };
  79. //处理系统指令
  80. //0,成功处理;其他,错误代码;
  81. u8 usmart_sys_cmd_exe(u8 *str)
  82. {
  83. u8 i;
  84. u8 sfname[MAX_FNAME_LEN];//存放本地函数名
  85. u8 pnum;
  86. u8 rval;
  87. u32 res;
  88. res=usmart_get_cmdname(str,sfname,&i,MAX_FNAME_LEN);//得到指令及指令长度
  89. if(res)return USMART_FUNCERR;//错误的指令
  90. str+=i;
  91. for(i=0;i<sizeof(sys_cmd_tab)/4;i++)//支持的系统指令
  92. {
  93. if(usmart_strcmp(sfname,sys_cmd_tab[i])==0)break;
  94. }
  95. switch(i)
  96. {
  97. case 0:
  98. case 1://帮助指令
  99. printf("\r\n");
  100. #if USMART_USE_HELP
  101. printf("------------------------USMART V3.3------------------------ \r\n");
  102. printf(" USMART是由ALIENTEK开发的一个灵巧的串口调试互交组件,通过 \r\n");
  103. printf("它,你可以通过串口助手调用程序里面的任何函数,并执行.因此,你可\r\n");
  104. printf("以随意更改函数的输入参数(支持数字(10/16进制,支持负数)、字符串\r\n"),
  105. printf("、函数入口地址等作为参数),单个函数最多支持10个输入参数,并支持\r\n"),
  106. printf("函数返回值显示.支持参数显示进制设置功能,支持进制转换功能.\r\n");
  107. printf("技术支持:www.openedv.com\r\n");
  108. printf("USMART有7个系统命令(必须小写):\r\n");
  109. printf("?: 获取帮助信息\r\n");
  110. printf("help: 获取帮助信息\r\n");
  111. printf("list: 可用的函数列表\r\n\n");
  112. printf("id: 可用函数的ID列表\r\n\n");
  113. printf("hex: 参数16进制显示,后跟空格+数字即执行进制转换\r\n\n");
  114. printf("dec: 参数10进制显示,后跟空格+数字即执行进制转换\r\n\n");
  115. printf("runtime:1,开启函数运行计时;0,关闭函数运行计时;\r\n\n");
  116. printf("请按照程序编写格式输入函数名及参数并以回车键结束.\r\n");
  117. printf("--------------------------ALIENTEK------------------------- \r\n");
  118. #else
  119. printf("指令失效\r\n");
  120. #endif
  121. break;
  122. case 2://查询指令
  123. printf("\r\n");
  124. printf("-------------------------函数清单--------------------------- \r\n");
  125. for(i=0;i<usmart_dev.fnum;i++)printf("%s\r\n",usmart_dev.funs[i].name);
  126. printf("\r\n");
  127. break;
  128. case 3://查询ID
  129. printf("\r\n");
  130. printf("-------------------------函数 ID --------------------------- \r\n");
  131. for(i=0;i<usmart_dev.fnum;i++)
  132. {
  133. usmart_get_fname((u8*)usmart_dev.funs[i].name,sfname,&pnum,&rval);//得到本地函数名
  134. printf("%s id is:\r\n0X%08X\r\n",sfname,usmart_dev.funs[i].func); //显示ID
  135. }
  136. printf("\r\n");
  137. break;
  138. case 4://hex指令
  139. printf("\r\n");
  140. usmart_get_aparm(str,sfname,&i);
  141. if(i==0)//参数正常
  142. {
  143. i=usmart_str2num(sfname,&res); //记录该参数
  144. if(i==0) //进制转换功能
  145. {
  146. printf("HEX:0X%X\r\n",res); //转为16进制
  147. }else if(i!=4)return USMART_PARMERR;//参数错误.
  148. else //参数显示设定功能
  149. {
  150. printf("16进制参数显示!\r\n");
  151. usmart_dev.sptype=SP_TYPE_HEX;
  152. }
  153. }else return USMART_PARMERR; //参数错误.
  154. printf("\r\n");
  155. break;
  156. case 5://dec指令
  157. printf("\r\n");
  158. usmart_get_aparm(str,sfname,&i);
  159. if(i==0)//参数正常
  160. {
  161. i=usmart_str2num(sfname,&res); //记录该参数
  162. if(i==0) //进制转换功能
  163. {
  164. printf("DEC:%lu\r\n",res); //转为10进制
  165. }else if(i!=4)return USMART_PARMERR;//参数错误.
  166. else //参数显示设定功能
  167. {
  168. printf("10进制参数显示!\r\n");
  169. usmart_dev.sptype=SP_TYPE_DEC;
  170. }
  171. }else return USMART_PARMERR; //参数错误.
  172. printf("\r\n");
  173. break;
  174. case 6://runtime指令,设置是否显示函数执行时间
  175. printf("\r\n");
  176. usmart_get_aparm(str,sfname,&i);
  177. if(i==0)//参数正常
  178. {
  179. i=usmart_str2num(sfname,&res); //记录该参数
  180. if(i==0) //读取指定地址数据功能
  181. {
  182. if(USMART_ENTIMX_SCAN==0)printf("\r\nError! \r\nTo EN RunTime function,Please set USMART_ENTIMX_SCAN = 1 first!\r\n");//报错
  183. else
  184. {
  185. usmart_dev.runtimeflag=res;
  186. if(usmart_dev.runtimeflag)printf("Run Time Calculation ON\r\n");
  187. else printf("Run Time Calculation OFF\r\n");
  188. }
  189. }else return USMART_PARMERR; //未带参数,或者参数错误
  190. }else return USMART_PARMERR; //参数错误.
  191. printf("\r\n");
  192. break;
  193. default://非法指令
  194. return USMART_FUNCERR;
  195. }
  196. return 0;
  197. }
  198. ////////////////////////////////////////////////////////////////////////////////////////
  199. //移植注意:本例是以stm32为例,如果要移植到其他mcu,请做相应修改.
  200. //usmart_reset_runtime,清除函数运行时间,连同定时器的计数寄存器以及标志位一起清零.并设置重装载值为最大,以最大限度的延长计时时间.
  201. //usmart_get_runtime,获取函数运行时间,通过读取CNT值获取,由于usmart是通过中断调用的函数,所以定时器中断不再有效,此时最大限度
  202. //只能统计2次CNT的值,也就是清零后+溢出一次,当溢出超过2次,没法处理,所以最大延时,控制在:2*计数器CNT*0.1ms.对STM32来说,是:13.1s左右
  203. //其他的:TIM4_IRQHandler和Timer4_Init,需要根据MCU特点自行修改.确保计数器计数频率为:10Khz即可.另外,定时器不要开启自动重装载功能!!
  204. #if USMART_ENTIMX_SCAN==1
  205. //复位runtime
  206. //需要根据所移植到的MCU的定时器参数进行修改
  207. void usmart_reset_runtime(void)
  208. {
  209. __HAL_TIM_CLEAR_FLAG(&TIM4_Handler,TIM_FLAG_UPDATE);//清除中断标志位
  210. __HAL_TIM_SET_AUTORELOAD(&TIM4_Handler,0XFFFF); //将重装载值设置到最大
  211. __HAL_TIM_SET_COUNTER(&TIM4_Handler,0); //清空定时器的CNT
  212. usmart_dev.runtime=0;
  213. }
  214. //获得runtime时间
  215. //返回值:执行时间,单位:0.1ms,最大延时时间为定时器CNT值的2倍*0.1ms
  216. //需要根据所移植到的MCU的定时器参数进行修改
  217. u32 usmart_get_runtime(void)
  218. {
  219. if(__HAL_TIM_GET_FLAG(&TIM4_Handler,TIM_FLAG_UPDATE)==SET)//在运行期间,产生了定时器溢出
  220. {
  221. usmart_dev.runtime+=0XFFFF;
  222. }
  223. usmart_dev.runtime+=__HAL_TIM_GET_COUNTER(&TIM4_Handler);
  224. return usmart_dev.runtime; //返回计数值
  225. }
  226. //下面这两个函数,非USMART函数,放到这里,仅仅方便移植.
  227. //定时器4中断服务程序
  228. void TIM4_IRQHandler(void)
  229. {
  230. if(__HAL_TIM_GET_IT_SOURCE(&TIM4_Handler,TIM_IT_UPDATE)==SET)//溢出中断
  231. {
  232. usmart_dev.scan(); //执行usmart扫描
  233. __HAL_TIM_SET_COUNTER(&TIM4_Handler,0);; //清空定时器的CNT
  234. __HAL_TIM_SET_AUTORELOAD(&TIM4_Handler,100);//恢复原来的设置
  235. }
  236. __HAL_TIM_CLEAR_IT(&TIM4_Handler, TIM_IT_UPDATE);//清除中断标志位
  237. }
  238. //使能定时器4,使能中断.
  239. void Timer4_Init(u16 arr,u16 psc)
  240. {
  241. //定时器4
  242. __HAL_RCC_TIM4_CLK_ENABLE();
  243. HAL_NVIC_SetPriority(TIM4_IRQn,3,3); //设置中断优先级,抢占优先级3,子优先级3
  244. HAL_NVIC_EnableIRQ(TIM4_IRQn); //开启ITM4中断
  245. TIM4_Handler.Instance=TIM4; //通用定时器4
  246. TIM4_Handler.Init.Prescaler=psc; //分频
  247. TIM4_Handler.Init.CounterMode=TIM_COUNTERMODE_UP; //向上计数器
  248. TIM4_Handler.Init.Period=arr; //自动装载值
  249. TIM4_Handler.Init.ClockDivision=TIM_CLOCKDIVISION_DIV1;
  250. HAL_TIM_Base_Init(&TIM4_Handler);
  251. HAL_TIM_Base_Start_IT(&TIM4_Handler); //使能定时器4和定时器4中断
  252. }
  253. #endif
  254. ////////////////////////////////////////////////////////////////////////////////////////
  255. //初始化串口控制器
  256. //sysclk:系统时钟(Mhz)
  257. void usmart_init(u8 sysclk)
  258. {
  259. #if USMART_ENTIMX_SCAN==1
  260. Timer4_Init(1000,(u32)sysclk*100-1);//分频,时钟为10K ,100ms中断一次,注意,计数频率必须为10Khz,以和runtime单位(0.1ms)同步.
  261. #endif
  262. usmart_dev.sptype=1; //十六进制显示参数
  263. }
  264. //从str中获取函数名,id,及参数信息
  265. //*str:字符串指针.
  266. //返回值:0,识别成功;其他,错误代码.
  267. u8 usmart_cmd_rec(u8*str)
  268. {
  269. u8 sta,i,rval;//状态
  270. u8 rpnum,spnum;
  271. u8 rfname[MAX_FNAME_LEN];//暂存空间,用于存放接收到的函数名
  272. u8 sfname[MAX_FNAME_LEN];//存放本地函数名
  273. sta=usmart_get_fname(str,rfname,&rpnum,&rval);//得到接收到的数据的函数名及参数个数
  274. if(sta)return sta;//错误
  275. for(i=0;i<usmart_dev.fnum;i++)
  276. {
  277. sta=usmart_get_fname((u8*)usmart_dev.funs[i].name,sfname,&spnum,&rval);//得到本地函数名及参数个数
  278. if(sta)return sta;//本地解析有误
  279. if(usmart_strcmp(sfname,rfname)==0)//相等
  280. {
  281. if(spnum>rpnum)return USMART_PARMERR;//参数错误(输入参数比源函数参数少)
  282. usmart_dev.id=i;//记录函数ID.
  283. break;//跳出.
  284. }
  285. }
  286. if(i==usmart_dev.fnum)return USMART_NOFUNCFIND; //未找到匹配的函数
  287. sta=usmart_get_fparam(str,&i); //得到函数参数个数
  288. if(sta)return sta; //返回错误
  289. usmart_dev.pnum=i; //参数个数记录
  290. return USMART_OK;
  291. }
  292. //usamrt执行函数
  293. //该函数用于最终执行从串口收到的有效函数.
  294. //最多支持10个参数的函数,更多的参数支持也很容易实现.不过用的很少.一般5个左右的参数的函数已经很少见了.
  295. //该函数会在串口打印执行情况.以:"函数名(参数1,参数2...参数N)=返回值".的形式打印.
  296. //当所执行的函数没有返回值的时候,所打印的返回值是一个无意义的数据.
  297. void usmart_exe(void)
  298. {
  299. u8 id,i;
  300. u32 res;
  301. u32 temp[MAX_PARM];//参数转换,使之支持了字符串
  302. u8 sfname[MAX_FNAME_LEN];//存放本地函数名
  303. u8 pnum,rval;
  304. id=usmart_dev.id;
  305. if(id>=usmart_dev.fnum)return;//不执行.
  306. usmart_get_fname((u8*)usmart_dev.funs[id].name,sfname,&pnum,&rval);//得到本地函数名,及参数个数
  307. printf("\r\n%s(",sfname);//输出正要执行的函数名
  308. for(i=0;i<pnum;i++)//输出参数
  309. {
  310. if(usmart_dev.parmtype&(1<<i))//参数是字符串
  311. {
  312. printf("%c",'"');
  313. printf("%s",usmart_dev.parm+usmart_get_parmpos(i));
  314. printf("%c",'"');
  315. temp[i]=(u32)&(usmart_dev.parm[usmart_get_parmpos(i)]);
  316. }else //参数是数字
  317. {
  318. temp[i]=*(u32*)(usmart_dev.parm+usmart_get_parmpos(i));
  319. if(usmart_dev.sptype==SP_TYPE_DEC)printf("%ld",temp[i]);//10进制参数显示
  320. else printf("0X%X",temp[i]);//16进制参数显示
  321. }
  322. if(i!=pnum-1)printf(",");
  323. }
  324. printf(")");
  325. #if USMART_ENTIMX_SCAN==1
  326. usmart_reset_runtime(); //计时器清零,开始计时
  327. #endif
  328. switch(usmart_dev.pnum)
  329. {
  330. case 0://无参数(void类型)
  331. res=(*(u32(*)())usmart_dev.funs[id].func)();
  332. break;
  333. case 1://有1个参数
  334. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0]);
  335. break;
  336. case 2://有2个参数
  337. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1]);
  338. break;
  339. case 3://有3个参数
  340. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2]);
  341. break;
  342. case 4://有4个参数
  343. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3]);
  344. break;
  345. case 5://有5个参数
  346. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3],temp[4]);
  347. break;
  348. case 6://有6个参数
  349. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3],temp[4],\
  350. temp[5]);
  351. break;
  352. case 7://有7个参数
  353. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3],temp[4],\
  354. temp[5],temp[6]);
  355. break;
  356. case 8://有8个参数
  357. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3],temp[4],\
  358. temp[5],temp[6],temp[7]);
  359. break;
  360. case 9://有9个参数
  361. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3],temp[4],\
  362. temp[5],temp[6],temp[7],temp[8]);
  363. break;
  364. case 10://有10个参数
  365. res=(*(u32(*)())usmart_dev.funs[id].func)(temp[0],temp[1],temp[2],temp[3],temp[4],\
  366. temp[5],temp[6],temp[7],temp[8],temp[9]);
  367. break;
  368. }
  369. #if USMART_ENTIMX_SCAN==1
  370. usmart_get_runtime();//获取函数执行时间
  371. #endif
  372. if(rval==1)//需要返回值.
  373. {
  374. if(usmart_dev.sptype==SP_TYPE_DEC)printf("=%lu;\r\n",res);//输出执行结果(10进制参数显示)
  375. else printf("=0X%X;\r\n",res);//输出执行结果(16进制参数显示)
  376. }else printf(";\r\n"); //不需要返回值,直接输出结束
  377. if(usmart_dev.runtimeflag) //需要显示函数执行时间
  378. {
  379. printf("Function Run Time:%d.%1dms\r\n",usmart_dev.runtime/10,usmart_dev.runtime%10);//打印函数执行时间
  380. }
  381. }
  382. //usmart扫描函数
  383. //通过调用该函数,实现usmart的各个控制.该函数需要每隔一定时间被调用一次
  384. //以及时执行从串口发过来的各个函数.
  385. //本函数可以在中断里面调用,从而实现自动管理.
  386. //如果非ALIENTEK用户,则USART_RX_STA和USART_RX_BUF[]需要用户自己实现
  387. void usmart_scan(void)
  388. {
  389. u8 sta,len;
  390. if(USART_RX_STA&0x8000)//串口接收完成?
  391. {
  392. len=USART_RX_STA&0x3fff; //得到此次接收到的数据长度
  393. USART_RX_BUF[len]='\0'; //在末尾加入结束符.
  394. sta=usmart_dev.cmd_rec(USART_RX_BUF);//得到函数各个信息
  395. if(sta==0)usmart_dev.exe(); //执行函数
  396. else
  397. {
  398. len=usmart_sys_cmd_exe(USART_RX_BUF);
  399. if(len!=USMART_FUNCERR)sta=len;
  400. if(sta)
  401. {
  402. switch(sta)
  403. {
  404. case USMART_FUNCERR:
  405. printf("函数错误!\r\n");
  406. break;
  407. case USMART_PARMERR:
  408. printf("参数错误!\r\n");
  409. break;
  410. case USMART_PARMOVER:
  411. printf("参数太多!\r\n");
  412. break;
  413. case USMART_NOFUNCFIND:
  414. printf("未找到匹配的函数!\r\n");
  415. break;
  416. }
  417. }
  418. }
  419. USART_RX_STA=0;//状态寄存器清空
  420. }
  421. }
  422. #if USMART_USE_WRFUNS==1 //如果使能了读写操作
  423. //读取指定地址的值
  424. u32 read_addr(u32 addr)
  425. {
  426. return *(u32*)addr;//
  427. }
  428. //在指定地址写入指定的值
  429. void write_addr(u32 addr,u32 val)
  430. {
  431. *(u32*)addr=val;
  432. }
  433. #endif