当前位置:   article > 正文

毕设 基于STM32F103ZET6的出租车计价器心得_基于stm32的出租车计价器设计

基于stm32的出租车计价器设计
    • 设计前准备

1、材料清单

  1. PCB

使用嘉立创PCB来绘图

优点:新手上手更快、拥有丰富的分装库、操作简单、可直接下单、每月可免费打样两次(10*10cm)

主控PCB用的模块所以都用的排母作为引脚的引出

电机PCB

3、环境配置

使用KEIL5进行程序编写

ST_Link设置

    • 模块程序编写

1、syn6288.c

使用淘宝卖家提供的SYN6288程序驱动,只需在SYN_FrameInfo函数中修改值就可以进行简单的使用

  1. #include "syn6288.h"
  2. #include "usart.h"
  3. #include "string.h"
  4. #include "delay.h"
  5. //Music:选择背景音乐。0:无背景音乐,1~15:选择背景音乐
  6. void SYN_FrameInfo(u8 Music, u8 *HZdata)
  7. {
  8. /****************需要发送的文本**********************************/
  9. unsigned char Frame_Info[50];
  10. unsigned char HZ_Length;
  11. unsigned char ecc = 0; //定义校验字节
  12. unsigned int i = 0;
  13. HZ_Length = strlen((char*)HZdata); //需要发送文本的长度
  14. /*****************帧固定配置信息**************************************/
  15. Frame_Info[0] = 0xFD ; //构造帧头FD
  16. Frame_Info[1] = 0x00 ; //构造数据区长度的高字节
  17. Frame_Info[2] = HZ_Length + 3; //构造数据区长度的低字节
  18. Frame_Info[3] = 0x01 ; //构造命令字:合成播放命令
  19. Frame_Info[4] = 0x01 | Music << 4 ; //构造命令参数:背景音乐设定
  20. /*******************校验码计算***************************************/
  21. for(i = 0; i < 5; i++) //依次发送构造好的5个帧头字节
  22. {
  23. ecc = ecc ^ (Frame_Info[i]); //对发送的字节进行异或校验
  24. }
  25. for(i = 0; i < HZ_Length; i++) //依次发送待合成的文本数据
  26. {
  27. ecc = ecc ^ (HZdata[i]); //对发送的字节进行异或校验
  28. }
  29. /*******************发送帧信息***************************************/
  30. memcpy(&Frame_Info[5], HZdata, HZ_Length);
  31. Frame_Info[5 + HZ_Length] = ecc;
  32. USART3_SendString(Frame_Info, 5 + HZ_Length + 1);
  33. }
  34. /***********************************************************
  35. * 名 称: YS_SYN_Set(u8 *Info_data)
  36. * 功 能: 主函数 程序入口
  37. * 入口参数: *Info_data:固定的配置信息变量
  38. * 出口参数:
  39. * 说 明:本函数用于配置,停止合成、暂停合成等设置 ,默认波特率9600bps。
  40. * 调用方法:通过调用已经定义的相关数组进行配置。
  41. **********************************************************/
  42. void YS_SYN_Set(u8 *Info_data)
  43. {
  44. u8 Com_Len;
  45. Com_Len = strlen((char*)Info_data);
  46. USART3_SendString(Info_data, Com_Len);
  47. }

2、霍尔传感器驱动

霍尔传感器的测速使用DO端口进行计数,在DO模式下当磁铁靠近会产生低电平型号,使用中断进行触发并使用一个变量进行计数

  1. #include "stm32f10x.h"
  2. #include "tim_cesu.h"
  3. #include "delay.h"
  4. #include "tim_cesu.h"
  5. #include "delay.h"
  6. u8 quan=0;
  7. void CESU_Init(void)
  8. { GPIO_InitTypeDef gpio;
  9. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA+RCC_APB2Periph_AFIO,ENABLE);//使能GPIOA
  10. gpio.GPIO_Mode=GPIO_Mode_IPD;
  11. gpio.GPIO_Pin=GPIO_Pin_7;
  12. gpio.GPIO_Speed=GPIO_Speed_10MHz;
  13. GPIO_Init(GPIOA,&gpio);
  14. EXTI_InitTypeDef EXTI_InitStructure; //中断方式初始化
  15. GPIO_EXTILineConfig(GPIO_PortSourceGPIOA,GPIO_PinSource7);
  16. EXTI_ClearITPendingBit(EXTI_Line7);
  17. EXTI_InitStructure.EXTI_Line=EXTI_Line7; //指定中断线3
  18. EXTI_InitStructure.EXTI_LineCmd=ENABLE; //使能中断线
  19. EXTI_InitStructure.EXTI_Mode=EXTI_Mode_Interrupt; //中断模式
  20. EXTI_InitStructure.EXTI_Trigger=EXTI_Trigger_Falling;
  21. EXTI_Init(&EXTI_InitStructure);
  22. }
  23. void NVICinit(void)
  24. {
  25. NVIC_InitTypeDef nvic;//中断优先级初始化结构体
  26. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
  27. nvic.NVIC_IRQChannel =EXTI9_5_IRQn; ;//使能按键外部中断通道
  28. nvic.NVIC_IRQChannelPreemptionPriority =2; //抢占优先级主优先级0
  29. nvic.NVIC_IRQChannelSubPriority =0x01;//子优先级0-15
  30. nvic.NVIC_IRQChannelCmd = ENABLE;
  31. NVIC_Init(&nvic);//初始化中断优先级
  32. }
  33. void EXTI9_5_IRQHandler(void)
  34. {
  35. if(EXTI_GetFlagStatus(EXTI_Line7)!=RESET)
  36. {
  37. quan=quan+1;
  38. delay_us(100);
  39. //清空中断标志位,防止持续进入中断
  40. }
  41. EXTI_ClearITPendingBit(EXTI_Line7);
  42. }

3、OLED驱动

使用的是正点原子opendv论坛上提供的0.96OLED驱动,使用I2C进行数据的传输。是使用取模软件进行对汉字的取模

  1. #include "oled.h"
  2. #include "stdlib.h"
  3. #include "oledfont.h"
  4. #include "delay.h"
  5. u8 OLED_GRAM[144][8];
  6. //反显函数
  7. void OLED_ColorTurn(u8 i)
  8. {
  9. if(i==0)
  10. {
  11. OLED_WR_Byte(0xA6,OLED_CMD);//正常显示
  12. }
  13. if(i==1)
  14. {
  15. OLED_WR_Byte(0xA7,OLED_CMD);//反色显示
  16. }
  17. }
  18. //屏幕旋转180度
  19. void OLED_DisplayTurn(u8 i)
  20. {
  21. if(i==0)
  22. {
  23. OLED_WR_Byte(0xC8,OLED_CMD);//正常显示
  24. OLED_WR_Byte(0xA1,OLED_CMD);
  25. }
  26. if(i==1)
  27. {
  28. OLED_WR_Byte(0xC0,OLED_CMD);//反转显示
  29. OLED_WR_Byte(0xA0,OLED_CMD);
  30. }
  31. }
  32. //延时
  33. void IIC_delay(void)
  34. {
  35. u8 t=3;
  36. while(t--);
  37. }
  38. //起始信号
  39. void I2C_Start(void)
  40. {
  41. OLED_SDA_Set();
  42. OLED_SCL_Set();
  43. IIC_delay();
  44. OLED_SDA_Clr();
  45. IIC_delay();
  46. OLED_SCL_Clr();
  47. IIC_delay();
  48. }
  49. //结束信号
  50. void I2C_Stop(void)
  51. {
  52. OLED_SDA_Clr();
  53. OLED_SCL_Set();
  54. IIC_delay();
  55. OLED_SDA_Set();
  56. }
  57. //等待信号响应
  58. void I2C_WaitAck(void) //测数据信号的电平
  59. {
  60. OLED_SDA_Set();
  61. IIC_delay();
  62. OLED_SCL_Set();
  63. IIC_delay();
  64. OLED_SCL_Clr();
  65. IIC_delay();
  66. }
  67. //写入一个字节
  68. void Send_Byte(u8 dat)
  69. {
  70. u8 i;
  71. for(i=0;i<8;i++)
  72. {
  73. if(dat&0x80)//将dat的8位从最高位依次写入
  74. {
  75. OLED_SDA_Set();
  76. }
  77. else
  78. {
  79. OLED_SDA_Clr();
  80. }
  81. IIC_delay();
  82. OLED_SCL_Set();
  83. IIC_delay();
  84. OLED_SCL_Clr();//将时钟信号设置为低电平
  85. dat<<=1;
  86. }
  87. }
  88. //发送一个字节
  89. //mode:数据/命令标志 0,表示命令;1,表示数据;
  90. void OLED_WR_Byte(u8 dat,u8 mode)
  91. {
  92. I2C_Start();
  93. Send_Byte(0x78);
  94. I2C_WaitAck();
  95. if(mode){Send_Byte(0x40);}
  96. else{Send_Byte(0x00);}
  97. I2C_WaitAck();
  98. Send_Byte(dat);
  99. I2C_WaitAck();
  100. I2C_Stop();
  101. }
  102. //开启OLED显示
  103. void OLED_DisPlay_On(void)
  104. {
  105. OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  106. OLED_WR_Byte(0x14,OLED_CMD);//开启电荷泵
  107. OLED_WR_Byte(0xAF,OLED_CMD);//点亮屏幕
  108. }
  109. //关闭OLED显示
  110. void OLED_DisPlay_Off(void)
  111. {
  112. OLED_WR_Byte(0x8D,OLED_CMD);//电荷泵使能
  113. OLED_WR_Byte(0x10,OLED_CMD);//关闭电荷泵
  114. OLED_WR_Byte(0xAE,OLED_CMD);//关闭屏幕
  115. }
  116. //更新显存到OLED
  117. void OLED_Refresh(void)
  118. {
  119. u8 i,n;
  120. for(i=0;i<8;i++)
  121. {
  122. OLED_WR_Byte(0xb0+i,OLED_CMD); //设置行起始地址
  123. OLED_WR_Byte(0x00,OLED_CMD); //设置低列起始地址
  124. OLED_WR_Byte(0x10,OLED_CMD); //设置高列起始地址
  125. I2C_Start();
  126. Send_Byte(0x78);
  127. I2C_WaitAck();
  128. Send_Byte(0x40);
  129. I2C_WaitAck();
  130. for(n=0;n<128;n++)
  131. {
  132. Send_Byte(OLED_GRAM[n][i]);
  133. I2C_WaitAck();
  134. }
  135. I2C_Stop();
  136. }
  137. }
  138. //清屏函数
  139. void OLED_Clear(void)
  140. {
  141. u8 i,n;
  142. for(i=0;i<8;i++)
  143. {
  144. for(n=0;n<128;n++)
  145. {
  146. OLED_GRAM[n][i]=0;//清除所有数据
  147. }
  148. }
  149. OLED_Refresh();//更新显示
  150. }
  151. //画点
  152. //x:0~127
  153. //y:0~63
  154. //t:1 填充 0,清空
  155. void OLED_DrawPoint(u8 x,u8 y,u8 t)
  156. {
  157. u8 i,m,n;
  158. i=y/8;
  159. m=y%8;
  160. n=1<<m;
  161. if(t){OLED_GRAM[x][i]|=n;}
  162. else
  163. {
  164. OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  165. OLED_GRAM[x][i]|=n;
  166. OLED_GRAM[x][i]=~OLED_GRAM[x][i];
  167. }
  168. }
  169. //画线
  170. //x1,y1:起点坐标
  171. //x2,y2:结束坐标
  172. void OLED_DrawLine(u8 x1,u8 y1,u8 x2,u8 y2,u8 mode)
  173. {
  174. u16 t;
  175. int xerr=0,yerr=0,delta_x,delta_y,distance;
  176. int incx,incy,uRow,uCol;
  177. delta_x=x2-x1; //计算坐标增量
  178. delta_y=y2-y1;
  179. uRow=x1;//画线起点坐标
  180. uCol=y1;
  181. if(delta_x>0)incx=1; //设置单步方向
  182. else if (delta_x==0)incx=0;//垂直线
  183. else {incx=-1;delta_x=-delta_x;}
  184. if(delta_y>0)incy=1;
  185. else if (delta_y==0)incy=0;//水平线
  186. else {incy=-1;delta_y=-delta_x;}
  187. if(delta_x>delta_y)distance=delta_x; //选取基本增量坐标轴
  188. else distance=delta_y;
  189. for(t=0;t<distance+1;t++)
  190. {
  191. OLED_DrawPoint(uRow,uCol,mode);//画点
  192. xerr+=delta_x;
  193. yerr+=delta_y;
  194. if(xerr>distance)
  195. {
  196. xerr-=distance;
  197. uRow+=incx;
  198. }
  199. if(yerr>distance)
  200. {
  201. yerr-=distance;
  202. uCol+=incy;
  203. }
  204. }
  205. }
  206. //x,y:圆心坐标
  207. //r:圆的半径
  208. void OLED_DrawCircle(u8 x,u8 y,u8 r)
  209. {
  210. int a, b,num;
  211. a = 0;
  212. b = r;
  213. while(2 * b * b >= r * r)
  214. {
  215. OLED_DrawPoint(x + a, y - b,1);
  216. OLED_DrawPoint(x - a, y - b,1);
  217. OLED_DrawPoint(x - a, y + b,1);
  218. OLED_DrawPoint(x + a, y + b,1);
  219. OLED_DrawPoint(x + b, y + a,1);
  220. OLED_DrawPoint(x + b, y - a,1);
  221. OLED_DrawPoint(x - b, y - a,1);
  222. OLED_DrawPoint(x - b, y + a,1);
  223. a++;
  224. num = (a * a + b * b) - r*r;//计算画的点离圆心的距离
  225. if(num > 0)
  226. {
  227. b--;
  228. a--;
  229. }
  230. }
  231. }
  232. //在指定位置显示一个字符,包括部分字符
  233. //x:0~127
  234. //y:0~63
  235. //size1:选择字体 6x8/6x12/8x16/12x24
  236. //mode:0,反色显示;1,正常显示
  237. void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size1,u8 mode)
  238. {
  239. u8 i,m,temp,size2,chr1;
  240. u8 x0=x,y0=y;
  241. if(size1==8)size2=6;
  242. else size2=(size1/8+((size1%8)?1:0))*(size1/2); //得到字体一个字符对应点阵集所占的字节数
  243. chr1=chr-' '; //计算偏移后的值
  244. for(i=0;i<size2;i++)
  245. {
  246. if(size1==8)
  247. {temp=asc2_0806[chr1][i];} //调用0806字体
  248. else if(size1==12)
  249. {temp=asc2_1206[chr1][i];} //调用1206字体
  250. else if(size1==16)
  251. {temp=asc2_1608[chr1][i];} //调用1608字体
  252. else if(size1==24)
  253. {temp=asc2_2412[chr1][i];} //调用2412字体
  254. else return;
  255. for(m=0;m<8;m++)
  256. {
  257. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  258. else OLED_DrawPoint(x,y,!mode);
  259. temp>>=1;
  260. y++;
  261. }
  262. x++;
  263. if((size1!=8)&&((x-x0)==size1/2))
  264. {x=x0;y0=y0+8;}
  265. y=y0;
  266. }
  267. }
  268. //显示字符串
  269. //x,y:起点坐标
  270. //size1:字体大小
  271. //*chr:字符串起始地址
  272. //mode:0,反色显示;1,正常显示
  273. void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 size1,u8 mode)
  274. {
  275. while((*chr>=' ')&&(*chr<='~'))//判断是不是非法字符!
  276. {
  277. OLED_ShowChar(x,y,*chr,size1,mode);
  278. if(size1==8)x+=6;
  279. else x+=size1/2;
  280. chr++;
  281. }
  282. }
  283. //m^n
  284. u32 OLED_Pow(u8 m,u8 n)
  285. {
  286. u32 result=1;
  287. while(n--)
  288. {
  289. result*=m;
  290. }
  291. return result;
  292. }
  293. //显示数字
  294. //x,y :起点坐标
  295. //num :要显示的数字
  296. //len :数字的位数
  297. //size:字体大小
  298. //mode:0,反色显示;1,正常显示
  299. void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size1,u8 mode)
  300. {
  301. u8 t,temp,m=0;
  302. if(size1==8)m=2;
  303. for(t=0;t<len;t++)
  304. {
  305. temp=(num/OLED_Pow(10,len-t-1))%10;
  306. if(temp==0)
  307. {
  308. OLED_ShowChar(x+(size1/2+m)*t,y,'0',size1,mode);
  309. }
  310. else
  311. {
  312. OLED_ShowChar(x+(size1/2+m)*t,y,temp+'0',size1,mode);
  313. }
  314. }
  315. }
  316. //显示汉字
  317. //x,y:起点坐标
  318. //num:汉字对应的序号
  319. //mode:0,反色显示;1,正常显示
  320. void OLED_ShowChinese(u8 x,u8 y,u8 num,u8 size1,u8 mode)
  321. {
  322. u8 m,temp;
  323. u8 x0=x,y0=y;
  324. u16 i,size3=(size1/8+((size1%8)?1:0))*size1; //得到字体一个字符对应点阵集所占的字节数
  325. for(i=0;i<size3;i++)
  326. {
  327. if(size1==16)
  328. {temp=Hzk1[num][i];}//调用16*16字体
  329. else if(size1==24)
  330. {temp=Hzk2[num][i];}//调用24*24字体
  331. else if(size1==32)
  332. {temp=Hzk3[num][i];}//调用32*32字体
  333. else if(size1==64)
  334. {temp=Hzk4[num][i];}//调用64*64字体
  335. else return;
  336. for(m=0;m<8;m++)
  337. {
  338. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  339. else OLED_DrawPoint(x,y,!mode);
  340. temp>>=1;
  341. y++;
  342. }
  343. x++;
  344. if((x-x0)==size1)
  345. {x=x0;y0=y0+8;}
  346. y=y0;
  347. }
  348. }
  349. //num 显示汉字的个数
  350. //space 每一遍显示的间隔
  351. //mode:0,反色显示;1,正常显示
  352. void OLED_ScrollDisplay(u8 num,u8 space,u8 mode)
  353. {
  354. u8 i,n,t=0,m=0,r;
  355. while(1)
  356. {
  357. if(m==0)
  358. {
  359. OLED_ShowChinese(128,24,t,16,mode); //写入一个汉字保存在OLED_GRAM[][]数组中
  360. t++;
  361. }
  362. if(t==num)
  363. {
  364. for(r=0;r<16*space;r++) //显示间隔
  365. {
  366. for(i=1;i<144;i++)
  367. {
  368. for(n=0;n<8;n++)
  369. {
  370. OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
  371. }
  372. }
  373. OLED_Refresh();
  374. }
  375. t=0;
  376. }
  377. m++;
  378. if(m==16){m=0;}
  379. for(i=1;i<144;i++) //实现左移
  380. {
  381. for(n=0;n<8;n++)
  382. {
  383. OLED_GRAM[i-1][n]=OLED_GRAM[i][n];
  384. }
  385. }
  386. OLED_Refresh();
  387. }
  388. }
  389. //x,y:起点坐标
  390. //sizex,sizey,图片长宽
  391. //BMP[]:要写入的图片数组
  392. //mode:0,反色显示;1,正常显示
  393. void OLED_ShowPicture(u8 x,u8 y,u8 sizex,u8 sizey,u8 BMP[],u8 mode)
  394. {
  395. u16 j=0;
  396. u8 i,n,temp,m;
  397. u8 x0=x,y0=y;
  398. sizey=sizey/8+((sizey%8)?1:0);
  399. for(n=0;n<sizey;n++)
  400. {
  401. for(i=0;i<sizex;i++)
  402. {
  403. temp=BMP[j];
  404. j++;
  405. for(m=0;m<8;m++)
  406. {
  407. if(temp&0x01)OLED_DrawPoint(x,y,mode);
  408. else OLED_DrawPoint(x,y,!mode);
  409. temp>>=1;
  410. y++;
  411. }
  412. x++;
  413. if((x-x0)==sizex)
  414. {
  415. x=x0;
  416. y0=y0+8;
  417. }
  418. y=y0;
  419. }
  420. }
  421. }
  422. //OLED的初始化
  423. void OLED_Init(void)
  424. {
  425. GPIO_InitTypeDef GPIO_InitStructure;
  426. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOD, ENABLE); //使能A端口时钟
  427. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7;
  428. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
  429. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz
  430. GPIO_Init(GPIOB, &GPIO_InitStructure);
  431. GPIO_SetBits(GPIOB,GPIO_Pin_6|GPIO_Pin_7);
  432. OLED_RES_Clr();
  433. delay_ms(200);
  434. OLED_RES_Set();
  435. OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel
  436. OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
  437. OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
  438. OLED_WR_Byte(0x40,OLED_CMD);//--set start line address Set Mapping RAM Display Start Line (0x00~0x3F)
  439. OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register
  440. OLED_WR_Byte(0xCF,OLED_CMD);// Set SEG Output Current Brightness
  441. OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping 0xa0左右反置 0xa1正常
  442. OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常
  443. OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display
  444. OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
  445. OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty
  446. OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F)
  447. OLED_WR_Byte(0x00,OLED_CMD);//-not offset
  448. OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency
  449. OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec
  450. OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period
  451. OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
  452. OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration
  453. OLED_WR_Byte(0x12,OLED_CMD);
  454. OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh
  455. OLED_WR_Byte(0x40,OLED_CMD);//Set VCOM Deselect Level
  456. OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02)
  457. OLED_WR_Byte(0x02,OLED_CMD);//
  458. OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable
  459. OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable
  460. OLED_WR_Byte(0xA4,OLED_CMD);// Disable Entire Display On (0xa4/0xa5)
  461. OLED_WR_Byte(0xA6,OLED_CMD);// Disable Inverse Display On (0xa6/a7)
  462. OLED_Clear();
  463. OLED_WR_Byte(0xAF,OLED_CMD);
  464. }

4、按键模块

拉高引脚输出高电平,当按键按下按键接地端会将端口电平拉低,调用内置函数读取应交状态

  1. #include "stm32f10x.h"
  2. #include "key.h"
  3. #include "sys.h"
  4. #include "delay.h"
  5. //按键初始化函数
  6. void KEY_Init(void) //IO初始化
  7. {
  8. GPIO_InitTypeDef GPIO_InitStructure;
  9. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC,ENABLE);//使能PORTA,PORTC时钟
  10. GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE);//关闭jtag,使能SWD,可以用SWD模式调试
  11. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15;//PA15
  12. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //设置成上拉输入
  13. GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA15
  14. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;//PC5
  15. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //设置成上拉输入
  16. GPIO_Init(GPIOC, &GPIO_InitStructure);//初始化GPIOC5
  17. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;//PA0
  18. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PA0设置成输入,默认下拉
  19. GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.0
  20. }
  21. u8 KEY_Scan(u8 mode)
  22. {
  23. static u8 key_up=1;//按键按松开标志
  24. if(mode)key_up=1; //支持连按
  25. if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))
  26. {
  27. delay_ms(10);//去抖动
  28. key_up=0;
  29. if(KEY0==0)return KEY0_PRES;
  30. else if(KEY1==0)return KEY1_PRES;
  31. else if(WK_UP==1)return WKUP_PRES;
  32. }else if(KEY0==1&&KEY1==1&&WK_UP==0)key_up=1;
  33. return 0;// 无按键按下
  34. }

5、红外人体感应

HC_SR505当有人靠近并移动时输出高电平,当人离开大概8S后输出低电平,在8S内会一直输出高电平且在期间有检测到移动,会将高电平时间进行顺言。

  1. #include "stm32f10x.h"
  2. #include "red_wire.h"
  3. void red_wire_Init(void)
  4. {
  5. GPIO_InitTypeDef gpio;
  6. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA+RCC_APB2Periph_AFIO,ENABLE);//使能GPIOA
  7. gpio.GPIO_Mode=GPIO_Mode_IPD;
  8. gpio.GPIO_Pin=GPIO_Pin_6;
  9. gpio.GPIO_Speed=GPIO_Speed_10MHz;
  10. GPIO_Init(GPIOA,&gpio);
  11. }

6、主函数

流程为当有人触发红外,播报价格提醒,结束后KEY1控制启动,KEY2暂停,KEY3结束行程清零所有参数并向乘客播报。其中使用变量使红外在本轮中只能被执行一次,当结束时在对其清零。在测试过程中发现,霍尔传感器在被出发时会计两次数,所以在调用参数时对其进行除2的操作。

  1. #include "sys.h"
  2. #include "delay.h"
  3. #include "stdio.h"
  4. #include "usart.h"
  5. #include "syn6288.h"
  6. #include "oled.h"
  7. #include "bmp.h"
  8. #include "tim_cesu.h"
  9. #include "key.h"
  10. #include "tb6612.h"
  11. #include "led.h"
  12. #include "red_wire.h"
  13. #include "stm32f10x.h"
  14. extern u8 quan;//tim_cesu中记录圈数的zhi
  15. /**************芯片设置命令*********************/
  16. u8 SYN_StopCom[] = {0xFD, 0X00, 0X02, 0X02, 0XFD}; //停止合成
  17. u8 SYN_SuspendCom[] = {0XFD, 0X00, 0X02, 0X03, 0XFC}; //暂停合成
  18. u8 SYN_RecoverCom[] = {0XFD, 0X00, 0X02, 0X04, 0XFB}; //恢复合成
  19. u8 SYN_ChackCom[] = {0XFD, 0X00, 0X02, 0X21, 0XDE}; //状态查询
  20. u8 SYN_PowerDownCom[] = {0XFD, 0X00, 0X02, 0X88, 0X77}; //进入POWER DOWN 状态命令
  21. void run()
  22. {
  23. TIM_SetCompare2(TIM1,1000);//设置TIMx捕获比较1寄存器值 此处为TIM1通道1 占空比为50%
  24. PBout(12)=1;
  25. PBout(13)=0;
  26. PBout(0)=1;
  27. PBout(1)=0;
  28. }
  29. void stop()
  30. {
  31. PBout(12)=0;
  32. PBout(13)=0;
  33. PBout(0)=0;
  34. PBout(1)=0;
  35. }
  36. void shwo_Foat_to_u8(u8 x,u8 y,float a)
  37. {
  38. u8 qian,hou;
  39. unsigned char b[7];
  40. qian=(u8)a;//提取将整数部分
  41. b[0]=qian/100+'0';
  42. b[1]=qian/10%10+'0';
  43. b[2]=qian%10+'0';
  44. b[3]='.';
  45. hou=(a-(u8)a)*100;
  46. b[4]=hou/10%10+'0';
  47. b[5]=hou%10+'0';
  48. OLED_ShowString(x,y,b,16,1);
  49. }
  50. int main(void)
  51. {
  52. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级
  53. delay_init();
  54. USART3_Init(9600);//设置语音播报的波特率
  55. CESU_Init();
  56. NVICinit();
  57. KEY_Init();
  58. LED_Init();
  59. TB6612_Ctrl_Init();
  60. TB6612_Config();
  61. delay_init();
  62. OLED_Init();
  63. red_wire_Init();
  64. float mile=0;
  65. float money=0;
  66. char date[20];
  67. u8 startvalue=10;
  68. u8 t=0;
  69. u8 a=0;
  70. u8 c=0;
  71. OLED_ColorTurn(0);//0正常显示,1 反色显示
  72. OLED_DisplayTurn(0);//0正常显示 1 屏幕翻转显示
  73. OLED_ShowChinese(0,0,0,16,1);
  74. OLED_ShowChinese(16,0,1,16,1);
  75. OLED_ShowChinese(48,0,4,16,1);//每
  76. OLED_ShowChinese(64,0,5,16,1);//公
  77. OLED_ShowChinese(80,0,6,16,1);//理
  78. OLED_ShowString(96,0,"2.5",16,1);
  79. OLED_ShowChinese(0,48,11,16,1);//里
  80. OLED_ShowChinese(16,48,12,16,1);//程
  81. OLED_ShowChinese(32,48,15,16,1);//:
  82. OLED_ShowChinese(0,16,13,16,1);//价
  83. OLED_ShowChinese(16,16,14,16,1);//格
  84. OLED_ShowChinese(32,16,15,16,1);
  85. while(1)
  86. {
  87. mile=(quan/2)*0.196;
  88. money=startvalue+mile*2.5;
  89. t=KEY_Scan(0); //得到键值
  90. switch(t)
  91. {
  92. case KEY0_PRES: LED0=1;
  93. run();
  94. break;
  95. case KEY1_PRES:
  96. LED1=0;
  97. stop();
  98. break;
  99. case WKUP_PRES:
  100. LED1=1; stop();
  101. sprintf(date,"%s%.2f%s%s%.2f%s","本次行程结束共计行驶",mile,"公里","共消费",money,"元");
  102. SYN_FrameInfo(6,(uint8_t *)date);
  103. delay_ms(9600);
  104. quan=0;a=0;c=0;
  105. break;
  106. default:
  107. delay_ms(10);
  108. }
  109. shwo_Foat_to_u8(48,48,mile);
  110. OLED_ShowString(108,48,"KM",16,1);
  111. shwo_Foat_to_u8(48,16,money);
  112. OLED_ShowChinese(108,16,7,16,1);
  113. OLED_Refresh();
  114. if(a==0)
  115. {
  116. if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_6)==1)
  117. {
  118. a++;
  119. delay_ms(10);
  120. if(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_6)==SET)
  121. {
  122. LED0=!LED0;
  123. SYN_FrameInfo(0,"欢迎乘坐本次出租车本车起步价十元白天2.5元每公里夜晚3.5元每公里");
  124. }
  125. }
  126. }
  127. }
  128. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/385689
推荐阅读
相关标签
  

闽ICP备14008679号