You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

441 line
11 KiB

  1. #include "usmart_str.h"
  2. #include "usmart.h"
  3. //////////////////////////////////////////////////////////////////////////////////
  4. //本程序只供学习使用,未经作者许可,不得用于其它任何用途
  5. //ALIENTEK STM32开发板
  6. //正点原子@ALIENTEK
  7. //技术论坛:www.openedv.com
  8. //版本:V3.3
  9. //版权所有,盗版必究。
  10. //Copyright(C) 正点原子 2011-2021
  11. //All rights reserved
  12. //********************************************************************************
  13. //升级说明
  14. //V1.4
  15. //增加了对参数为string类型的函数的支持.适用范围大大提高.
  16. //优化了内存占用,静态内存占用为79个字节@10个参数.动态适应数字及字符串长度
  17. //V2.0
  18. //1,修改了list指令,打印函数的完整表达式.
  19. //2,增加了id指令,打印每个函数的入口地址.
  20. //3,修改了参数匹配,支持函数参数的调用(输入入口地址).
  21. //4,增加了函数名长度宏定义.
  22. //V2.1 20110707
  23. //1,增加dec,hex两个指令,用于设置参数显示进制,及执行进制转换.
  24. //注:当dec,hex不带参数的时候,即设定显示参数进制.当后跟参数的时候,即执行进制转换.
  25. //如:"dec 0XFF" 则会将0XFF转为255,由串口返回.
  26. //如:"hex 100" 则会将100转为0X64,由串口返回
  27. //2,新增usmart_get_cmdname函数,用于获取指令名字.
  28. //V2.2 20110726
  29. //1,修正了void类型参数的参数统计错误.
  30. //2,修改数据显示格式默认为16进制.
  31. //V2.3 20110815
  32. //1,去掉了函数名后必须跟"("的限制.
  33. //2,修正了字符串参数中不能有"("的bug.
  34. //3,修改了函数默认显示参数格式的修改方式.
  35. //V2.4 20110905
  36. //1,修改了usmart_get_cmdname函数,增加最大参数长度限制.避免了输入错误参数时的死机现象.
  37. //2,增加USMART_ENTIM2_SCAN宏定义,用于配置是否使用TIM2定时执行scan函数.
  38. //V2.5 20110930
  39. //1,修改usmart_init函数为void usmart_init(u8 sysclk),可以根据系统频率自动设定扫描时间.(固定100ms)
  40. //2,去掉了usmart_init函数中的uart_init函数,串口初始化必须在外部初始化,方便用户自行管理.
  41. //V2.6 20111009
  42. //1,增加了read_addr和write_addr两个函数.可以利用这两个函数读写内部任意地址(必须是有效地址).更加方便调试.
  43. //2,read_addr和write_addr两个函数可以通过设置USMART_USE_WRFUNS为来使能和关闭.
  44. //3,修改了usmart_strcmp,使其规范化.
  45. //V2.7 20111024
  46. //1,修正了返回值16进制显示时不换行的bug.
  47. //2,增加了函数是否有返回值的判断,如果没有返回值,则不会显示.有返回值时才显示其返回值.
  48. //V2.8 20111116
  49. //1,修正了list等不带参数的指令发送后可能导致死机的bug.
  50. //V2.9 20120917
  51. //1,修改了形如:void*xxx(void)类型函数不能识别的bug。
  52. //V3.0 20130425
  53. //1,新增了字符串参数对转义符的支持。
  54. //V3.1 20131120
  55. //1,增加runtime系统指令,可以用于统计函数执行时间.
  56. //用法:
  57. //发送:runtime 1 ,则开启函数执行时间统计功能
  58. //发送:runtime 0 ,则关闭函数执行时间统计功能
  59. ///runtime统计功能,必须设置:USMART_ENTIMX_SCAN 为1,才可以使用!!
  60. //V3.2 20140828
  61. //1,修改usmart_get_aparm函数,加入+/-符号的支持
  62. //2,修改usmart_str2num函数,支持负数转换
  63. //V3.3 20160506
  64. //修正usmart_exe函数在USMART_ENTIMX_SCAN为0的时候,报错的bug
  65. /////////////////////////////////////////////////////////////////////////////////////
  66. //对比字符串str1和str2
  67. //*str1:字符串1指针
  68. //*str2:字符串2指针
  69. //返回值:0,相等;1,不相等;
  70. u8 usmart_strcmp(u8 *str1,u8 *str2)
  71. {
  72. while(1)
  73. {
  74. if(*str1!=*str2)return 1;//不相等
  75. if(*str1=='\0')break;//对比完成了.
  76. str1++;
  77. str2++;
  78. }
  79. return 0;//两个字符串相等
  80. }
  81. //把str1的内容copy到str2
  82. //*str1:字符串1指针
  83. //*str2:字符串2指针
  84. void usmart_strcopy(u8*str1,u8 *str2)
  85. {
  86. while(1)
  87. {
  88. *str2=*str1; //拷贝
  89. if(*str1=='\0')break;//拷贝完成了.
  90. str1++;
  91. str2++;
  92. }
  93. }
  94. //得到字符串的长度(字节)
  95. //*str:字符串指针
  96. //返回值:字符串的长度
  97. u8 usmart_strlen(u8*str)
  98. {
  99. u8 len=0;
  100. while(1)
  101. {
  102. if(*str=='\0')break;//拷贝完成了.
  103. len++;
  104. str++;
  105. }
  106. return len;
  107. }
  108. //m^n函数
  109. //返回值:m^n次方
  110. u32 usmart_pow(u8 m,u8 n)
  111. {
  112. u32 result=1;
  113. while(n--)result*=m;
  114. return result;
  115. }
  116. //把字符串转为数字
  117. //支持16进制转换,但是16进制字母必须是大写的,且格式为以0X开头的.
  118. //支持负数
  119. //*str:数字字符串指针
  120. //*res:转换完的结果存放地址.
  121. //返回值:0,成功转换完成.其他,错误代码.
  122. //1,数据格式错误.2,16进制位数为0.3,起始格式错误.4,十进制位数为0.
  123. u8 usmart_str2num(u8*str,u32 *res)
  124. {
  125. u32 t;
  126. int tnum;
  127. u8 bnum=0; //数字的位数
  128. u8 *p;
  129. u8 hexdec=10; //默认为十进制数据
  130. u8 flag=0; //0,没有符号标记;1,表示正数;2,表示负数.
  131. p=str;
  132. *res=0;//清零.
  133. while(1)
  134. {
  135. if((*p<='9'&&*p>='0')||((*str=='-'||*str=='+')&&bnum==0)||(*p<='F'&&*p>='A')||(*p=='X'&&bnum==1))//参数合法
  136. {
  137. if(*p>='A')hexdec=16; //字符串中存在字母,为16进制格式.
  138. if(*str=='-'){flag=2;str+=1;}//偏移掉符号
  139. else if(*str=='+'){flag=1;str+=1;}//偏移掉符号
  140. else bnum++; //位数增加.
  141. }else if(*p=='\0')break; //碰到结束符,退出.
  142. else return 1; //不全是十进制或者16进制数据.
  143. p++;
  144. }
  145. p=str; //重新定位到字符串开始的地址.
  146. if(hexdec==16) //16进制数据
  147. {
  148. if(bnum<3)return 2; //位数小于3,直接退出.因为0X就占了2个,如果0X后面不跟数据,则该数据非法.
  149. if(*p=='0' && (*(p+1)=='X'))//必须以'0X'开头.
  150. {
  151. p+=2; //偏移到数据起始地址.
  152. bnum-=2;//减去偏移量
  153. }else return 3;//起始头的格式不对
  154. }else if(bnum==0)return 4;//位数为0,直接退出.
  155. while(1)
  156. {
  157. if(bnum)bnum--;
  158. if(*p<='9'&&*p>='0')t=*p-'0'; //得到数字的值
  159. else t=*p-'A'+10; //得到A~F对应的值
  160. *res+=t*usmart_pow(hexdec,bnum);
  161. p++;
  162. if(*p=='\0')break;//数据都查完了.
  163. }
  164. if(flag==2)//是负数?
  165. {
  166. tnum=-*res;
  167. *res=tnum;
  168. }
  169. return 0;//成功转换
  170. }
  171. //得到指令名
  172. //*str:源字符串
  173. //*cmdname:指令名
  174. //*nlen:指令名长度
  175. //maxlen:最大长度(做限制,指令不可能太长的)
  176. //返回值:0,成功;其他,失败.
  177. u8 usmart_get_cmdname(u8*str,u8*cmdname,u8 *nlen,u8 maxlen)
  178. {
  179. *nlen=0;
  180. while(*str!=' '&&*str!='\0') //找到空格或者结束符则认为结束了
  181. {
  182. *cmdname=*str;
  183. str++;
  184. cmdname++;
  185. (*nlen)++;//统计命令长度
  186. if(*nlen>=maxlen)return 1;//错误的指令
  187. }
  188. *cmdname='\0';//加入结束符
  189. return 0;//正常返回
  190. }
  191. //获取下一个字符(当中间有很多空格的时候,此函数直接忽略空格,找到空格之后的第一个字符)
  192. //str:字符串指针
  193. //返回值:下一个字符
  194. u8 usmart_search_nextc(u8* str)
  195. {
  196. str++;
  197. while(*str==' '&&str!='\0')str++;
  198. return *str;
  199. }
  200. //从str中得到函数名
  201. //*str:源字符串指针
  202. //*fname:获取到的函数名字指针
  203. //*pnum:函数的参数个数
  204. //*rval:是否需要显示返回值(0,不需要;1,需要)
  205. //返回值:0,成功;其他,错误代码.
  206. u8 usmart_get_fname(u8*str,u8*fname,u8 *pnum,u8 *rval)
  207. {
  208. u8 res;
  209. u8 fover=0; //括号深度
  210. u8 *strtemp;
  211. u8 offset=0;
  212. u8 parmnum=0;
  213. u8 temp=1;
  214. u8 fpname[6];//void+X+'/0'
  215. u8 fplcnt=0; //第一个参数的长度计数器
  216. u8 pcnt=0; //参数计数器
  217. u8 nchar;
  218. //判断函数是否有返回值
  219. strtemp=str;
  220. while(*strtemp!='\0')//没有结束
  221. {
  222. if(*strtemp!=' '&&(pcnt&0X7F)<5)//最多记录5个字符
  223. {
  224. if(pcnt==0)pcnt|=0X80;//置位最高位,标记开始接收返回值类型
  225. if(((pcnt&0x7f)==4)&&(*strtemp!='*'))break;//最后一个字符,必须是*
  226. fpname[pcnt&0x7f]=*strtemp;//记录函数的返回值类型
  227. pcnt++;
  228. }else if(pcnt==0X85)break;
  229. strtemp++;
  230. }
  231. if(pcnt)//接收完了
  232. {
  233. fpname[pcnt&0x7f]='\0';//加入结束符
  234. if(usmart_strcmp(fpname,"void")==0)*rval=0;//不需要返回值
  235. else *rval=1; //需要返回值
  236. pcnt=0;
  237. }
  238. res=0;
  239. strtemp=str;
  240. while(*strtemp!='('&&*strtemp!='\0') //此代码找到函数名的真正起始位置
  241. {
  242. strtemp++;
  243. res++;
  244. if(*strtemp==' '||*strtemp=='*')
  245. {
  246. nchar=usmart_search_nextc(strtemp); //获取下一个字符
  247. if(nchar!='('&&nchar!='*')offset=res; //跳过空格和*号
  248. }
  249. }
  250. strtemp=str;
  251. if(offset)strtemp+=offset+1;//跳到函数名开始的地方
  252. res=0;
  253. nchar=0;//是否正在字符串里面的标志,0,不在字符串;1,在字符串;
  254. while(1)
  255. {
  256. if(*strtemp==0)
  257. {
  258. res=USMART_FUNCERR;//函数错误
  259. break;
  260. }else if(*strtemp=='('&&nchar==0)fover++;//括号深度增加一级
  261. else if(*strtemp==')'&&nchar==0)
  262. {
  263. if(fover)fover--;
  264. else res=USMART_FUNCERR;//错误结束,没收到'('
  265. if(fover==0)break;//到末尾了,退出
  266. }else if(*strtemp=='"')nchar=!nchar;
  267. if(fover==0)//函数名还没接收完
  268. {
  269. if(*strtemp!=' ')//空格不属于函数名
  270. {
  271. *fname=*strtemp;//得到函数名
  272. fname++;
  273. }
  274. }else //已经接受完了函数名了.
  275. {
  276. if(*strtemp==',')
  277. {
  278. temp=1; //使能增加一个参数
  279. pcnt++;
  280. }else if(*strtemp!=' '&&*strtemp!='(')
  281. {
  282. if(pcnt==0&&fplcnt<5) //当第一个参数来时,为了避免统计void类型的参数,必须做判断.
  283. {
  284. fpname[fplcnt]=*strtemp;//记录参数特征.
  285. fplcnt++;
  286. }
  287. temp++; //得到有效参数(非空格)
  288. }
  289. if(fover==1&&temp==2)
  290. {
  291. temp++; //防止重复增加
  292. parmnum++; //参数增加一个
  293. }
  294. }
  295. strtemp++;
  296. }
  297. if(parmnum==1)//只有1个参数.
  298. {
  299. fpname[fplcnt]='\0';//加入结束符
  300. if(usmart_strcmp(fpname,"void")==0)parmnum=0;//参数为void,表示没有参数.
  301. }
  302. *pnum=parmnum; //记录参数个数
  303. *fname='\0'; //加入结束符
  304. return res; //返回执行结果
  305. }
  306. //从str中得到一个函数的参数
  307. //*str:源字符串指针
  308. //*fparm:参数字符串指针
  309. //*ptype:参数类型 0,数字;1,字符串;0XFF,参数错误
  310. //返回值:0,已经无参数了;其他,下一个参数的偏移量.
  311. u8 usmart_get_aparm(u8 *str,u8 *fparm,u8 *ptype)
  312. {
  313. u8 i=0;
  314. u8 enout=0;
  315. u8 type=0;//默认是数字
  316. u8 string=0; //标记str是否正在读
  317. while(1)
  318. {
  319. if(*str==','&& string==0)enout=1; //暂缓立即退出,目的是寻找下一个参数的起始地址
  320. if((*str==')'||*str=='\0')&&string==0)break;//立即退出标识符
  321. if(type==0)//默认是数字的
  322. {
  323. if((*str>='0' && *str<='9')||*str=='-'||*str=='+'||(*str>='a' && *str<='f')||(*str>='A' && *str<='F')||*str=='X'||*str=='x')//数字串检测
  324. {
  325. if(enout)break; //找到了下一个参数,直接退出.
  326. if(*str>='a')*fparm=*str-0X20; //小写转换为大写
  327. else *fparm=*str; //小写或者数字保持不变
  328. fparm++;
  329. }else if(*str=='"')//找到字符串的开始标志
  330. {
  331. if(enout)break;//找到,后才找到",认为结束了.
  332. type=1;
  333. string=1;//登记STRING 正在读了
  334. }else if(*str!=' '&&*str!=',')//发现非法字符,参数错误
  335. {
  336. type=0XFF;
  337. break;
  338. }
  339. }else//string类
  340. {
  341. if(*str=='"')string=0;
  342. if(enout)break; //找到了下一个参数,直接退出.
  343. if(string) //字符串正在读
  344. {
  345. if(*str=='\\') //遇到转义符(不复制转义符)
  346. {
  347. str++; //偏移到转义符后面的字符,不管什么字符,直接COPY
  348. i++;
  349. }
  350. *fparm=*str; //小写或者数字保持不变
  351. fparm++;
  352. }
  353. }
  354. i++;//偏移量增加
  355. str++;
  356. }
  357. *fparm='\0'; //加入结束符
  358. *ptype=type; //返回参数类型
  359. return i; //返回参数长度
  360. }
  361. //得到指定参数的起始地址
  362. //num:第num个参数,范围0~9.
  363. //返回值:该参数的起始地址
  364. u8 usmart_get_parmpos(u8 num)
  365. {
  366. u8 temp=0;
  367. u8 i;
  368. for(i=0;i<num;i++)temp+=usmart_dev.plentbl[i];
  369. return temp;
  370. }
  371. //从str中得到函数参数
  372. //str:源字符串;
  373. //parn:参数的多少.0表示无参数 void类型
  374. //返回值:0,成功;其他,错误代码.
  375. u8 usmart_get_fparam(u8*str,u8 *parn)
  376. {
  377. u8 i,type;
  378. u32 res;
  379. u8 n=0;
  380. u8 len;
  381. u8 tstr[PARM_LEN+1];//字节长度的缓存,最多可以存放PARM_LEN个字符的字符串
  382. for(i=0;i<MAX_PARM;i++)usmart_dev.plentbl[i]=0;//清空参数长度表
  383. while(*str!='(')//偏移到参数开始的地方
  384. {
  385. str++;
  386. if(*str=='\0')return USMART_FUNCERR;//遇到结束符了
  387. }
  388. str++;//偏移到"("之后的第一个字节
  389. while(1)
  390. {
  391. i=usmart_get_aparm(str,tstr,&type); //得到第一个参数
  392. str+=i; //偏移
  393. switch(type)
  394. {
  395. case 0: //数字
  396. if(tstr[0]!='\0') //接收到的参数有效
  397. {
  398. i=usmart_str2num(tstr,&res); //记录该参数
  399. if(i)return USMART_PARMERR; //参数错误.
  400. *(u32*)(usmart_dev.parm+usmart_get_parmpos(n))=res;//记录转换成功的结果.
  401. usmart_dev.parmtype&=~(1<<n); //标记数字
  402. usmart_dev.plentbl[n]=4; //该参数的长度为4
  403. n++; //参数增加
  404. if(n>MAX_PARM)return USMART_PARMOVER;//参数太多
  405. }
  406. break;
  407. case 1://字符串
  408. len=usmart_strlen(tstr)+1; //包含了结束符'\0'
  409. usmart_strcopy(tstr,&usmart_dev.parm[usmart_get_parmpos(n)]);//拷贝tstr数据到usmart_dev.parm[n]
  410. usmart_dev.parmtype|=1<<n; //标记字符串
  411. usmart_dev.plentbl[n]=len; //该参数的长度为len
  412. n++;
  413. if(n>MAX_PARM)return USMART_PARMOVER;//参数太多
  414. break;
  415. case 0XFF://错误
  416. return USMART_PARMERR;//参数错误
  417. }
  418. if(*str==')'||*str=='\0')break;//查到结束标志了.
  419. }
  420. *parn=n; //记录参数的个数
  421. return USMART_OK;//正确得到了参数
  422. }