全國電子設計競賽,STM32F4+ov76攝像頭 PID控制 供參考使用
制作出來的實物圖如下:
IMG_20190428_225915.jpg (6.89 MB, 下載次數: 43)
下載附件
2020-3-25 09:46 上傳
單片機源程序如下:
- #include "sys.h"
- #include "delay.h"
- #include "usart.h"
- #include "led.h"
- #include "key.h"
- #include "lcd.h"
- #include "usmart.h"
- #include "usart2.h"
- #include "timer.h"
- #include "ov2640.h"
- #include "dcmi.h"
- #include "string.h"
- #include "pwm.h"
- //JPEG尺寸支持列表
- const u16 jpeg_img_size_tbl[][2]=
- {
- 176,144, //QCIF
- 160,120, //QQVGA
- 352,288, //CIF
- 320,240, //QVGA
- 640,480, //VGA
- 800,600, //SVGA
- 1024,768, //XGA
- 1280,1024, //SXGA
- 1600,1200, //UXGA
- };
- void TIM3_Int_Init(u16 arr,u16 psc);
- //RGB565測試
- //RGB數據直接顯示在LCD上面
- void rgb565_test(void)
- {
- u8 key;
- LCD_Clear(WHITE);
- POINT_COLOR=RED;
-
- OV2640_ImageWin_Set((800-480)/2,(600-600)/2,480,600);
- OV2640_RGB565_Mode(); //RGB565模式
- My_DCMI_Init(); //DCMI配置
- DCMI_DMA_Init((u32)&LCD->LCD_RAM,1,DMA_MemoryDataSize_HalfWord,DMA_MemoryInc_Disable);//DCMI DMA配置
- OV2640_OutSize_Set(lcddev.width,600);
- DCMI_Start(); //啟動傳輸
- while(1)
- {
- key=KEY_Scan(0);
- if(key)
- {
- DCMI_Stop(); //停止顯示
- switch(key)
- {
- case KEY0_PRES: //對比度設置
-
- break;
- case KEY1_PRES: //飽和度Saturation
-
- break;
- case KEY2_PRES: //特效設置
-
- break;
- case WKUP_PRES:
-
- break;
- }
-
- DCMI_Start();//重新開始傳輸
- }
- delay_ms(10);
- }
- }
- u16 rgb_buf[144][176];
- u16 gray;
- extern u8 flag;
- u16 hang=0;
- u8 X_MAX,Y_MAX=0; //小球的坐標信息
- u8 X_MAX_LSAT, X_MIN_LSAT, Y_MAX_LSAT, Y_MIN_LSAT=0; //上一次小球坐標位置信息
- u8 X,Y=0; //小球的質心信息
- u8 X_MIN,Y_MIN=180;
- int PWM_X,PWM_Y=0; //pid參數初始化
- float Err_X,Err_Y=0;
- float Err_X_LAST,Err_Y_LAST=0;
- float Aim_X,Aim_Y=0;
- float Kp,Ki,Kd=0;
- int main(void)
- {
- u16 i,j;
- NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//設置系統中斷優先級分組2
- delay_init(168); //初始化延時函數
- uart_init(115200); //初始化串口波特率為115200
- usart2_init(42,115200); //初始化串口2波特率為115200
- LED_Init(); //初始化LED
- LCD_Init(); //LCD初始化
- KEY_Init(); //按鍵初始化
- TIM3_Int_Init(200-1,8400-1);//10Khz計數,1秒鐘中斷一次
-
- TIM14_PWM_Init(10000-1,168-1); //重裝載值10000,所以PWM頻率為 150hz. //舵機預留
- TIM11_PWM_Init(10000-1,336-1); //重裝載值10000,所以PWM頻率為 150hz.
-
- usmart_dev.init(84); //初始化USMART
- POINT_COLOR=RED;//設置字體為紅色
- LCD_ShowString(30,50,200,16,16,"Explorer STM32F4");
- LCD_ShowString(30,70,200,16,16,"OV2640 TEST");
- LCD_ShowString(30,90,200,16,16,"ATOM@ALIENTEK");
- LCD_ShowString(30,110,200,16,16,"2014/5/14");
- while(OV2640_Init())//初始化OV2640
- {
- LCD_ShowString(30,130,240,16,16,"OV2640 ERR");
- delay_ms(200);
- LCD_Fill(30,130,239,170,WHITE);
- delay_ms(200);
- }
- LCD_ShowString(30,130,200,16,16,"OV2640 OK");
-
- OV2640_OutSize_Set(176,144); //注意此處圖像選取QCIF格式
- OV2640_RGB565_Mode(); //RGB565模式
- My_DCMI_Init(); //DCMI配置
- DCMI_DMA_Init((u32)rgb_buf,sizeof(rgb_buf)/4,DMA_MemoryDataSize_HalfWord,DMA_MemoryInc_Enable);//DCMI DMA配置
- DCMI_Start(); //啟動傳輸
-
-
- //=============================================二值化和尋點部分========================================================
- while(1)
- {
- hang=0;
- LCD_SetCursor(0,0);
- LCD_WriteRAM_Prepare(); //開始寫入GRAM
- for(i=0;i<144;i++)
- {
- for(j=0;j<176;j++)
- {
- if(j==175)
- {
- hang++;
- LCD_SetCursor(0,i+1);
- LCD_WriteRAM_Prepare(); //開始寫入GRAM
- }
- gray=((rgb_buf[i][j]>>11)*19595+((rgb_buf[i][j]>>5)&0x3f)*38469 +(rgb_buf[i][j]&0x1f)*7472)>>16; //灰度計算。公式請百度
- if(gray>=23) //固定閾值二值化
- {
-
- if(i>8&&i<136&&j<160&&j>16) //此處遍歷圖像尋找小球最上最下 最左 最右四個點坐標
- {
- if(i>X_MAX) X_MAX=i;
- if(i<X_MIN) X_MIN=i;
-
-
-
- if(j>Y_MAX) Y_MAX=j;
- if(j<Y_MIN) Y_MIN=j;
-
- }
- LCD->LCD_RAM=WHITE;
- }
- else
- {
-
- LCD->LCD_RAM=BLACK;
- }
- }
- }
-
- X_MAX_LSAT = X_MAX; //更新pid的real坐標信息
- X_MIN_LSAT = X_MIN;
- Y_MAX_LSAT = Y_MAX;
- Y_MIN_LSAT = Y_MIN;
-
- X_MAX=0;
- X_MIN=180;
- Y_MAX=0;
- Y_MIN=180; //清除掉本次坐標用于再次遍歷最大值 最小值
-
- X=(X_MAX_LSAT+X_MIN_LSAT)/2;
- Y=(Y_MAX_LSAT+Y_MIN_LSAT)/2; //通過四個點坐標計算小球質心
-
- // TIM_SetCompare1(TIM14,9340); //修改比較值,修改占空比 調試舵機使用
-
- // TIM_SetCompare1(TIM11,9300); //修改比較值,修改占空比 調試舵機使用
-
- }
- }
- //=============================================pid計算部分(中斷?=============================================
- void TIM3_IRQHandler(void) //pid控制中斷函數 沒記錯的話是50Hz
- {
- if(TIM_GetITStatus(TIM3,TIM_IT_Update)==SET) //溢出中斷
- {
- Kp=2.4; //pd參數
- Kd=90.9;
- Aim_X=72; //目標點
- Aim_Y=88;
-
- Err_X=X-Aim_X;
- Err_Y=Y-Aim_Y;
-
-
- PWM_X=9360+(Err_X*Kp+(Err_X-Err_X_LAST)*Kd); //增量式pd計算
- PWM_Y=9300+(Err_Y*Kp+(Err_Y-Err_Y_LAST)*Kd);
-
- if(PWM_Y>9370)PWM_Y=9370; //pid輸出限幅度 防止抽風
- if(PWM_Y<9230)PWM_Y=9230;
-
- if(PWM_X>9410)PWM_X=9410;
- if(PWM_X<9270)PWM_X=9270;
-
- Err_X_LAST=Err_X;
- Err_Y_LAST=Err_Y;
-
- TIM_SetCompare1(TIM14,PWM_X); //修改比較值,修改占空比 輸出pid計算值
- TIM_SetCompare1(TIM11,PWM_Y); //修改比較值,修改占空比
- }
- TIM_ClearITPendingBit(TIM3,TIM_IT_Update); //清除中斷標志位
- }
復制代碼
所有資料51hei提供下載:
板球系統單點定位代碼.7z
(371.15 KB, 下載次數: 13)
2020-3-25 16:34 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
|