找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5334|回复: 5
收起左侧

程序更新!STM32+OV7670+IL9325找球(黑底红球)带2路PWM(PID)程序

  [复制链接]
ID:225718 发表于 2017-8-12 14:54 | 显示全部楼层 |阅读模式
本帖最后由 zhangpro666 于 2017-8-13 11:13 编辑

呃...看到大家都在电赛,自己暑假无聊玩一下,如果大家觉得有用就拿去吧。用的是启光电子的STM32,OV7670,ILI9325,理论上核心板通过跳线都可以实现。

QQ图片20170812145131.jpg QQ图片20170812145125.jpg QQ图片20170812145041.jpg FL7VQ_V1U8GB06)A[OSI`~V.png

原理图:
0.png

stm32主程序:
  1. /*
  2. OV7670模块 STM32函数

  3. 选用STM32F103RBT6芯片   
  4. TFT驱动IC为  R61505V

  5.   程序中 用到的TFT 为 我们店的 2.4寸TFT  驱动芯片为 R61505V        
  6.   TFT用到的GPIO  PB为数据口        PC.6  CS   PC.7  RS          PC.8  WR   PC.9  RD        PC.10 RES
  7.   此函数 用寄存器方式配置GPIO  在TFT的数据 命令脚配置上用的是寄存器  这样能大幅度提高输出速度

  8. 摄像头方面
  9. 摄像头 数据口 为PA口  AO-A7
  10. SCL    PC11
  11. SDA    PC12
  12. OE           PC3                 //片选信号(OE)
  13. WRST   PC4                 //写指针复位
  14. RRST   PC2                 //读指针复位
  15. RCLK   PC1                 //读数据时钟
  16. WEN           PC5                 //写入FIFO使能
  17. VSY           PD2           //同步信号检测IO

  18. 整个程序流程介绍

  19. 初始化TFT   
  20. 初始化ov7670  ov7670是通过SCCB 初始化的  即 SCL 和SDA就可以初始化OV7670  SDA SCL类似于I2C效果
  21.               所以在调模块前 一定要确定OV7670是否已经初始化
  22. 开启中断 判断帧数据是否接收
  23. 死循环 读取并显示 摄像数据

  24.            
  25. */

  26. #include  "stm32f10x.h"
  27. #include  "delay.h"
  28. #include  "led.h"
  29. #include  "tft.h"
  30. #include  "led.h"
  31. #include "ov7670.h"
  32. #include  "gui.h"
  33. #include "usart.h"
  34. #include  "PID.h"

  35. #define KEY0  GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_9)        
  36. #define KEY1  GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_10)

  37. void coordinate_output(void);


  38. extern u8 ov_sta;        //帧次数 置位标志位
  39. struct{
  40. u8 photo;
  41. u8 pub_buf[14401];//120*120
  42. char buf[200];
  43. }CAM;



  44. //更新LCD显示
  45. void camera_refresh(void)
  46. {
  47.         u32 j;
  48.          u16 color,n565Color;
  49.         u8 cRed=0,cBlue=0,cGreen=0;         
  50.         
  51.         TFT_SCAN(5);   //设置TFT扫描方向 为从下到上 从右到左
  52.         Address_set(60,0,179,119,0,0);                   //设置显示范围        
  53.         if(ov_sta==2)                                //判断缓存区是否存好摄像数据
  54.         {
  55.                   
  56.                 OV7670_RRST_0;                                //开始复位读指针
  57.                 OV7670_RCK_0;
  58.                 OV7670_RCK_1;
  59.                 OV7670_RCK_0;               
  60.                 OV7670_RRST_1;                                //复位读指针结束
  61.                 OV7670_RCK_1;  
  62.                
  63.                 TFT_RS_1;                 //写数据线拉高                为提高写入速度

  64.                 //此用法 是将 下面写数据的步骤完全拆开  主要是为了提高显示速度  (下面注释部分为参考部分)
  65.                      
  66.                 for(j=0;j<14400;j++)                           //分辨率为240x320   每个颜色点要两个字节 所以 240x320x2=76800  ?
  67.                 {                        
  68.                         OV7670_RCK_0;                                 //每一次时钟跳变 读一次数据
  69.                         color=GPIOC->IDR&0XFF;        //读数据   读取颜色高字节数据
  70.                         OV7670_RCK_1;
  71.                         color<<=8;                                   //左移8位  将高字节移到高8位  为接收低8位字节做准备
  72.                         OV7670_RCK_0;
  73.                         color|=GPIOC->IDR&0XFF;        //读数据   读取颜色低字节数据 并转化为16位 565 颜色数据
  74.                         OV7670_RCK_1;
  75.                   cRed = ((color & 0xf800)>>11);
  76.                         CAM.pub_buf[j]=cRed;
  77.                   n565Color = (cRed << 11) + (cGreen << 5) + (cBlue << 0);                 
  78.             GPIOB->ODR=n565Color;                           //对TFT并口快速写数据
  79.       TFT_WR_0;                                                                      //开始写入
  80.       TFT_WR_1;
  81.                 }  
  82.                  coordinate_output();
  83.                 EXTI_ClearITPendingBit(EXTI_Line2);  //清除LINE8上的中断标志位
  84.                 ov_sta=0;                                        //开始下一次采集
  85.         }
  86. }
  87. u16 blight=15;//亮度判断更改这里
  88. void coordinate_output(void)
  89. {
  90.         u16 i,j,k;
  91.         xmax=ymax=0;
  92.         xmin=ymin=119;
  93.         for(i=0;i<120;i++)
  94.         {
  95.                   for(j=0;j<120;j++)
  96.                         {
  97.                                             if(CAM.pub_buf[(i*120)+j]>=blight)
  98.                   {
  99.                                                                                  if(i>1&&i<119&&j<119&&j>1)//如果i值大于8且小于136(限制小球在8~136行),同理,限制小球在16~160列
  100.                                                                                  {
  101.                                                                                             if(j>xmax) {xmax=j;} //不断更新边缘,直到找到小球的直径
  102.                                                                                              if(j<xmin) {xmin=j;}//不断更新边缘,直到找到小球的直径
  103.                                                 
  104.                                                                                             if(i>ymax) {ymax=i;}//不断更新边缘,直到找到小球的直径
  105.                                                                                              if(i<ymin) {ymin=i;}//不断更新边缘,直到找到小球的直径                                                                                         
  106.                                                                                  }
  107.                   }
  108.                         }
  109.         }
  110.         for(k=0;k<14400;k++)
  111.         {
  112.                 CAM.pub_buf[k]=0x00;
  113.         }
  114.         x=(xmax+xmin)/2;
  115.         y=(ymax+ymin)/2;
  116.         if(x<1){x=1;}
  117.         if(x>119){x=119;}
  118.         if(y<1){y=1;}
  119.         if(y>119){y=119;}
  120.         if(x!=0&&y!=0)
  121.         {        
  122. //        GUI_sprintf_hzstr16x(20,200,"                                        ",Red,White);
  123.                 sprintf(CAM.buf,"X:%d   ",x);
  124.                 GUI_sprintf_hzstr16x(20,200,CAM.buf,Red,White);
  125.                 sprintf(CAM.buf,"Y:%d   ",y);
  126.                 GUI_sprintf_hzstr16x(80,200,CAM.buf,Red,White);
  127.                 printf("X:%d",x);
  128.                 printf("Y:%d",y);
  129.                 printf("\r\n");
  130.         }
  131. }

  132. void disp_picture(void)
  133. {
  134.         u16 i,j=0;
  135.         u16 COLOR;
  136.         u8 aRed=0,aBlue=0,aGreen=0;
  137.         coordinate_output();
  138.         
  139.         TFT_SCAN(5);
  140.         Address_set(60,0,179,119,0,0);        
  141.         TFT_RS_1;               

  142.         for(i=0;i<14400;i++)                           
  143.                 {                        
  144.                         aRed=CAM.pub_buf[i];
  145.                   printf("%d        ",aRed);
  146.                         j++;
  147.                         if(j>=120){printf("\r\n");j=0;}               
  148.                   COLOR = (aRed << 11) + (aGreen << 5) + (aBlue << 0);
  149.             GPIOB->ODR=COLOR;                           //对TFT并口快速写数据
  150.       TFT_WR_0;                                                                      //开始写入
  151.       TFT_WR_1;               
  152.                 }               
  153.          
  154.                 EXTI_ClearITPendingBit(EXTI_Line2);
  155.                
  156. }

  157. int main(void)
  158. {
  159. u8 lightmode=2,saturation=3,brightness=2,contrast=2;
  160. u8 effect=0,flag=0;
  161.           delay_init();                     //延时函数初始化
  162.           NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);                        
  163.           Lcd_Init();                         //LCD  初始化
  164.          LED_Init();                  
  165.           uart_init(115200);
  166.          TIM2_Int_Init(1,1799);
  167.           TFT_CS(0);                         //打开LCD片选使能
  168.       GUI_Clear(White);                //清屏

  169.                   while(OV7670_Init())//初始化OV7670
  170.         {
  171.             GUI_sprintf_hzstr16x(20,150,"OV7670 Error!!            ",Blue,White);
  172.                 delay_ms(200);
  173.                 printf("OV7670 ERROR!!!!!\r\n");
  174.         }

  175.           GUI_Clear(White);                //清屏                初始化成功后 红色清屏

  176.                 CAM.photo=0;
  177. //         delay_ms(1500);
  178.          //摄像头 参数设置  屏蔽后为默认效果                    
  179.         OV7670_Light_Mode(lightmode);
  180.         OV7670_Color_Saturation(saturation);
  181.         OV7670_Brightness(brightness);
  182.         OV7670_Contrast(contrast);
  183.          OV7670_Special_Effects(effect);         

  184.         EXTI2_Init();                                           //初始化中断线
  185.         OV7670_Window_Set(10,174,120,120);
  186.         OV7670_CS=0;               
  187.           while(1)                                                           //死循环显示 摄像数据
  188.           {
  189.                 camera_refresh();
  190. //                 test_motor();
  191.           }
  192. }

复制代码

全部资料下载地址:
STM32找球.rar (429.95 KB, 下载次数: 60)

评分

参与人数 1黑币 +100 收起 理由
admin + 100 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:225718 发表于 2017-8-12 14:59 | 显示全部楼层
支持坐标1~119
回复

使用道具 举报

ID:225718 发表于 2017-8-13 11:12 | 显示全部楼层

  1. <p>#include "pid.h"
  2. #include "usart.h"
  3. #include "led.h"</p><p>int x,y,xmax,ymax,xmin,ymin;</p><p>
  4. struct{</p><p>int setspeed;
  5. int actspeed;
  6. int err;  
  7. int err_last;   
  8. int err_next;      
  9. float Kp,Ki,Kd;
  10. float integral;
  11. int incrementSpeed;</p><p>}pid;</p><p>void PID_init(void)
  12. {
  13. pid.setspeed=0;
  14. pid.actspeed=60;
  15. pid.err=0;
  16. pid.Kp=0.25;  //这里调节P参数
  17. pid.Ki=0.25;  //这里调节I参数
  18. pid.Kd=0.21;  //这里调节D参数
  19. }
  20. int PID_adjustment(int speed,int act) //PID处理函数,输入X、Y到目标的距离  输出应该移动的距离
  21. {
  22. pid.setspeed=speed;
  23. pid.actspeed=act;
  24. pid.err=pid.setspeed-pid.actspeed;
  25. pid.incrementSpeed=pid.Kp*(pid.err-pid.err_next)+pid.Ki*pid.err+pid.Kd*(pid.err-2*pid.err_next+pid.err_last);
  26. // pid.actspeed+=pid.incrementSpeed;
  27. pid.err_last=pid.err_next;
  28.     pid.err_next=pid.err;

  29. return pid.incrementSpeed;
  30. }</p><p>void DIST_TO_PWM(int dist,u8 zhou) //距离转换 输入 距离 方向 轴 直接调整
  31. {
  32.    if(zhou==0)//X轴
  33.    {
  34.     timer1=30+dist*0.2;  //这里调整X轴,注意!如果X轴反了,把+改成-
  35. //    printf("PIDPWMX:%d",timer1);
  36. //    printf("\r\n");
  37.    }
  38.    if(zhou==1)//Y轴
  39.    {
  40.     timer2=30+dist*0.2;  //这里调节Y轴,注意!如果Y轴反了,把+改成-
  41. //    printf("PIDPWMY:%d",timer2);
  42. //    printf("\r\n");
  43.    }
  44. }
  45. void zuobiao_process(u16 ux,u16 uy)//输入目标的坐标
  46. {
  47. int xdist,ydist,dist;//x、y离坐标的距离
  48. xdist=ux-x;
  49. ydist=uy-y;
  50. dist=PID_adjustment(0,xdist);
  51. if(dist>50){dist=50;} //限制距离调节在50
  52. // printf("PIDx:%d",dist);
  53. if(dist<0){DIST_TO_PWM(dist,0);}//处理X轴
  54. if(dist>0){DIST_TO_PWM(dist,0);}
  55. if(dist==0){timer1=30;}
  56. dist=PID_adjustment(0,ydist);
  57. if(dist>50){dist=50;}  //限制距离调节在50
  58. // printf("PIDy:%d",dist);
  59. // printf("\r\n");
  60. if(dist!=0){DIST_TO_PWM(dist,1);}//处理Y轴
  61. if(dist==0){timer2=30;}
  62. }

  63. </p>
复制代码

程序更新!谁说STM32F103慢?黑底找球+两路PWM舵机输出(带PID),可以在程序里面调节PID大小,实测输出FPS:20左右!


截屏_20170813_105441.jpg

PID V1.00.rar

4.99 MB, 下载次数: 25, 下载积分: 黑币 -5

回复

使用道具 举报

ID:138247 发表于 2019-4-4 16:19 | 显示全部楼层

谢谢分享。。。。。。。。。
回复

使用道具 举报

ID:138247 发表于 2019-4-4 16:23 | 显示全部楼层

谢谢分享。。。。
回复

使用道具 举报

ID:110257 发表于 2019-4-5 06:51 | 显示全部楼层
谢谢分享!!!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表