找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 7851|回复: 0
打印 上一主题 下一主题
收起左侧

ATK-OV2640摄像头模块说明+源码+封装库等开发资料

[复制链接]
跳转到指定楼层
楼主


该文件夹下包含8个源码:
(寄存器版本,适合探索者STM32F4开发板)实验42 照相机实验.rar
(库函数版本,适合探索者STM32F4开发板)实验42 照相机实验.rar

(寄存器版本,适合战舰V3和精英STM32开发板)扩展实验6 ATK-OV2640 摄像头模块测试实验.rar
(库函数版本,适合战舰V3和精英STM32开发板)扩展实验6 ATK-OV2640 摄像头模块测试实验.rar

(寄存器版本,适合阿波罗STM32F429开发板)扩展实验10 ATK-OV2640 摄像头模块测试实验.rar
(库函数版本,适合阿波罗STM32F429开发板)扩展实验10 ATK-OV2640 摄像头模块测试实验.rar

(寄存器版本,适合阿波罗STM32F767开发板)扩展实验10 ATK-OV2640 摄像头模块测试实验.rar
(库函数版本,适合阿波罗STM32F767开发板)扩展实验10 ATK-OV2640 摄像头模块测试实验.rar

包含多款开发板的例程,每个例程又分为寄存器和库函数两个版本,请大家务必根据自己的开发板,下载正确的例程!不要下载错了。

例程使用说明请看源码里面的readme.txt或者相关AN文档。


单片机源程序如下:
  1. #include "sys.h"
  2. #include "delay.h"
  3. #include "usart.h"
  4. #include "led.h"
  5. #include "key.h"
  6. #include "lcd.h"
  7. #include "ltdc.h"
  8. #include "string.h"
  9. #include "sdram.h"
  10. #include "malloc.h"
  11. #include "w25qxx.h"
  12. #include "ov2640.h"
  13. #include "dcmi.h"  
  14. #include "pcf8574.h"
  15. #include "ff.h"
  16. #include "exfuns.h"
  17. #include "string.h"
  18. #include "usmart.h"
  19. #include "sdio_sdcard.h"
  20. #include "fontupd.h"
  21. #include "text.h"
  22. #include "piclib.h"       
  23. #include "string.h"               
  24. #include "math.h"

  25. vu8 bmp_request=0;                                                //bmp拍照请求:0,无bmp拍照请求;1,有bmp拍照请求,需要在帧中断里面,关闭DCMI接口.
  26. u8 ovx_mode=0;                                                        //bit0:0,RGB565模式;1,JPEG模式
  27. u16 curline=0;                                                        //摄像头输出数据,当前行编号
  28. u16 yoffset=0;                                                        //y方向的偏移量

  29. #define jpeg_buf_size   4*1024*1024                //定义JPEG数据缓存jpeg_buf的大小(4M字节)
  30. #define jpeg_line_size        2*1024                        //定义DMA接收数据时,一行数据的最大值

  31. u32 *dcmi_line_buf[2];                                        //RGB屏时,摄像头采用一行一行读取,定义行缓存  
  32. u32 *jpeg_data_buf;                                                //JPEG数据缓存buf

  33. volatile u32 jpeg_data_len=0;                         //buf中的JPEG有效数据长度
  34. volatile u8 jpeg_data_ok=0;                                //JPEG数据采集完成标志
  35.                                                                                 //0,数据没有采集完;
  36.                                                                                 //1,数据采集完了,但是还没处理;
  37.                                                                                 //2,数据已经处理完成了,可以开始下一帧接收
  38. //处理JPEG数据
  39. //当采集完一帧JPEG数据后,调用此函数,切换JPEG BUF.开始下一帧采集.
  40. void jpeg_data_process(void)
  41. {
  42.         u16 i;
  43.         u16 rlen;                        //剩余数据长度
  44.         u32 *pbuf;
  45.         curline=yoffset;        //行数复位
  46.         if(ovx_mode&0X01)        //只有在JPEG格式下,才需要做处理.
  47.         {
  48.                 if(jpeg_data_ok==0)        //jpeg数据还未采集完?
  49.                 {
  50.                         __HAL_DMA_DISABLE(&DMADMCI_Handler);//关闭DMA
  51.                         while(DMA2_Stream1->CR&0X01);        //等待DMA2_Stream1可配置
  52.                         rlen=jpeg_line_size-__HAL_DMA_GET_COUNTER(&DMADMCI_Handler);//得到剩余数据长度       
  53.                         pbuf=jpeg_data_buf+jpeg_data_len;//偏移到有效数据末尾,继续添加
  54.                         if(DMADMCI_Handler.Instance->CR&(1<<19))for(i=0;i<rlen;i++)pbuf[i]=dcmi_line_buf[1][i];//读取buf1里面的剩余数据
  55.                         else for(i=0;i<rlen;i++)pbuf[i]=dcmi_line_buf[0][i];//读取buf0里面的剩余数据
  56.                         jpeg_data_len+=rlen;                        //加上剩余长度
  57.                         jpeg_data_ok=1;                                 //标记JPEG数据采集完按成,等待其他函数处理
  58.                 }
  59.                 if(jpeg_data_ok==2)        //上一次的jpeg数据已经被处理了
  60.                 {
  61.             __HAL_DMA_SET_COUNTER(&DMADMCI_Handler,jpeg_line_size);        //传输长度为jpeg_buf_size*4字节
  62.                         __HAL_DMA_ENABLE(&DMADMCI_Handler); //打开DMA
  63.                         jpeg_data_ok=0;                                        //标记数据未采集
  64.                         jpeg_data_len=0;                                //数据重新开始
  65.                 }
  66.         }else
  67.         {  
  68.                 if(bmp_request==1)        //有bmp拍照请求,关闭DCMI
  69.                 {
  70.                         DCMI_Stop();        //停止DCMI
  71.                         bmp_request=0;        //标记请求处理完成.
  72.                 }
  73.                 LCD_SetCursor(0,0);  
  74.                 LCD_WriteRAM_Prepare();                                //开始写入GRAM  
  75.         }  
  76. }

  77. //jpeg数据接收回调函数
  78. void jpeg_dcmi_rx_callback(void)
  79. {  
  80.         u16 i;
  81.         u32 *pbuf;
  82.         pbuf=jpeg_data_buf+jpeg_data_len;//偏移到有效数据末尾
  83.         if(DMADMCI_Handler.Instance->CR&(1<<19))//buf0已满,正常处理buf1
  84.         {
  85.                 for(i=0;i<jpeg_line_size;i++)pbuf[i]=dcmi_line_buf[0][i];//读取buf0里面的数据
  86.                 jpeg_data_len+=jpeg_line_size;//偏移
  87.         }else //buf1已满,正常处理buf0
  88.         {
  89.                 for(i=0;i<jpeg_line_size;i++)pbuf[i]=dcmi_line_buf[1][i];//读取buf1里面的数据
  90.                 jpeg_data_len+=jpeg_line_size;//偏移
  91.         }
  92. }

  93. //RGB屏数据接收回调函数
  94. void rgblcd_dcmi_rx_callback(void)
  95. {  
  96.         u16 *pbuf;
  97.         if(DMADMCI_Handler.Instance->CR&(1<<19))//DMA使用buf1,读取buf0
  98.         {
  99.                 pbuf=(u16*)dcmi_line_buf[0];
  100.         }else                                                 //DMA使用buf0,读取buf1
  101.         {
  102.                 pbuf=(u16*)dcmi_line_buf[1];
  103.         }        
  104.         LTDC_Color_Fill(0,curline,lcddev.width-1,curline,pbuf);//DM2D填充
  105.         if(curline<lcddev.height)curline++;
  106.         if(bmp_request==1&&curline==(lcddev.height-1))//有bmp拍照请求,关闭DCMI
  107.         {
  108.                 DCMI_Stop();        //停止DCMI
  109.                 bmp_request=0;        //标记请求处理完成.
  110.         }
  111. }

  112. //切换为OV2640模式
  113. void sw_ov2640_mode(void)
  114. {
  115.     GPIO_InitTypeDef GPIO_Initure;
  116.    
  117.     OV2640_PWDN_Set(0); //OV2640 Power Up
  118.         //GPIOC8/9/11切换为 DCMI接口
  119.     GPIO_Initure.Pin=GPIO_PIN_8|GPIO_PIN_9|GPIO_PIN_11;  
  120.     GPIO_Initure.Mode=GPIO_MODE_AF_PP;          //推挽复用
  121.     GPIO_Initure.Pull=GPIO_PULLUP;              //上拉
  122.     GPIO_Initure.Speed=GPIO_SPEED_HIGH;         //高速
  123.     GPIO_Initure.Alternate=GPIO_AF13_DCMI;      //复用为DCMI   
  124.     HAL_GPIO_Init(GPIOC,&GPIO_Initure);         //初始化
  125. }
  126. //切换为SD卡模式
  127. void sw_sdcard_mode(void)
  128. {
  129.     GPIO_InitTypeDef GPIO_Initure;
  130.       
  131.     OV2640_PWDN_Set(1); //OV2640 Power Down  
  132.         //GPIOC8/9/11切换为 SDIO接口
  133.     GPIO_Initure.Pin=GPIO_PIN_8|GPIO_PIN_9|GPIO_PIN_11;  
  134.     GPIO_Initure.Mode=GPIO_MODE_AF_PP;          //推挽复用
  135.     GPIO_Initure.Pull=GPIO_PULLUP;              //上拉
  136.     GPIO_Initure.Speed=GPIO_SPEED_HIGH;         //高速
  137.     GPIO_Initure.Alternate=GPIO_AF12_SDMMC1;    //复用为SDIO  
  138.     HAL_GPIO_Init(GPIOC,&GPIO_Initure);
  139. }

  140. //文件名自增(避免覆盖)
  141. //mode:0,创建.bmp文件;1,创建.jpg文件.
  142. //bmp组合成:形如"0:PHOTO/PIC13141.bmp"的文件名
  143. //jpg组合成:形如"0:PHOTO/PIC13141.jpg"的文件名
  144. void camera_new_pathname(u8 *pname,u8 mode)
  145. {         
  146.         u8 res;                                         
  147.         u16 index=0;
  148.         while(index<0XFFFF)
  149.         {
  150.                 if(mode==0)sprintf((char*)pname,"0:PHOTO/PIC%05d.bmp",index);
  151.                 else sprintf((char*)pname,"0:PHOTO/PIC%05d.jpg",index);
  152.                 res=f_open(ftemp,(const TCHAR*)pname,FA_READ);//尝试打开这个文件
  153.                 if(res==FR_NO_FILE)break;                //该文件名不存在=正是我们需要的.
  154.                 index++;
  155.         }
  156. }  

  157. //OV2640拍照jpg图片
  158. //返回值:0,成功
  159. //    其他,错误代码
  160. u8 ov2640_jpg_photo(u8 *pname)
  161. {
  162.         FIL* f_jpg;
  163.         u8 res=0,headok=0;
  164.         u32 bwr;
  165.         u32 i,jpgstart,jpglen;
  166.         u8* pbuf;
  167.         f_jpg=(FIL *)mymalloc(SRAMIN,sizeof(FIL));        //开辟FIL字节的内存区域
  168.         if(f_jpg==NULL)return 0XFF;                                //内存申请失败.
  169.         ovx_mode=1;
  170.         jpeg_data_ok=0;
  171.         sw_ov2640_mode();                                                //切换为OV2640模式
  172.         OV2640_JPEG_Mode();                                                //JPEG模式  
  173.         OV2640_ImageWin_Set(0,0,1600,1200);                         
  174.         OV2640_OutSize_Set(1600,1200);          //拍照尺寸为1600*120
  175.         dcmi_rx_callback=jpeg_dcmi_rx_callback;        //JPEG接收数据回调函数
  176.         DCMI_DMA_Init((u32)dcmi_line_buf[0],(u32)dcmi_line_buf[1],jpeg_line_size,DMA_MDATAALIGN_WORD,DMA_MINC_ENABLE);//DCMI DMA配置        
  177.         DCMI_Start();                         //启动传输
  178.         while(jpeg_data_ok!=1);        //等待第一帧图片采集完
  179.         jpeg_data_ok=2;                        //忽略本帧图片,启动下一帧采集
  180.         while(jpeg_data_ok!=1);        //等待第二帧图片采集完,第二帧,才保存到SD卡去.
  181.         DCMI_Stop();                         //停止DMA搬运
  182.         ovx_mode=0;
  183.         sw_sdcard_mode();                //切换为SD卡模式
  184.         res=f_open(f_jpg,(const TCHAR*)pname,FA_WRITE|FA_CREATE_NEW);//模式0,或者尝试打开失败,则创建新文件         
  185.         if(res==0)
  186.         {
  187.                 printf("jpeg data size:%d\r\n",jpeg_data_len*4);//串口打印JPEG文件大小
  188.                 pbuf=(u8*)jpeg_data_buf;
  189.                 jpglen=0;        //设置jpg文件大小为0
  190.                 headok=0;        //清除jpg头标记
  191.                 for(i=0;i<jpeg_data_len*4;i++)//查找0XFF,0XD8和0XFF,0XD9,获取jpg文件大小
  192.                 {
  193.                         if((pbuf[i]==0XFF)&&(pbuf[i+1]==0XD8))//找到FF D8
  194.                         {
  195.                                 jpgstart=i;
  196.                                 headok=1;        //标记找到jpg头(FF D8)
  197.                         }
  198.                         if((pbuf[i]==0XFF)&&(pbuf[i+1]==0XD9)&&headok)//找到头以后,再找FF D9
  199.                         {
  200.                                 jpglen=i-jpgstart+2;
  201.                                 break;
  202.                         }
  203.                 }
  204.                 if(jpglen)                        //正常的jpeg数据
  205.                 {
  206.                         pbuf+=jpgstart;        //偏移到0XFF,0XD8处
  207.                         res=f_write(f_jpg,pbuf,jpglen,&bwr);
  208.                         if(bwr!=jpglen)res=0XFE;
  209.                        
  210.                 }else res=0XFD;
  211.         }
  212.         jpeg_data_len=0;
  213.         f_close(f_jpg);
  214.         sw_ov2640_mode();                //切换为OV2640模式
  215.         OV2640_RGB565_Mode();        //RGB565模式  
  216.         if(lcdltdc.pwidth!=0)        //RGB屏
  217.         {
  218.                 dcmi_rx_callback=rgblcd_dcmi_rx_callback;//RGB屏接收数据回调函数
  219.                 DCMI_DMA_Init((u32)dcmi_line_buf[0],(u32)dcmi_line_buf[1],lcddev.width/2,DMA_MDATAALIGN_HALFWORD,DMA_MINC_ENABLE);//DCMI DMA配置
  220.         }else                                        //MCU 屏
  221.         {
  222.                 DCMI_DMA_Init((u32)&LCD->LCD_RAM,0,1,DMA_MDATAALIGN_HALFWORD,DMA_MINC_DISABLE);                        //DCMI DMA配置,MCU屏,竖屏
  223.         }
  224.         myfree(SRAMIN,f_jpg);
  225.         return res;
  226. }  

  227. int main(void)
  228. {       
  229.         u8 res;                                                         
  230.         u8 *pname;                                        //带路径的文件名
  231.         u8 key;                                                //键值                  
  232.         u8 i;                                                 
  233.         u8 sd_ok=1;                                        //0,sd卡不正常;1,SD卡正常.
  234.         u8 scale=1;                                        //默认是全尺寸缩放
  235.         u8 msgbuf[15];                                //消息缓存区
  236.     u16 outputheight=0;
  237.    
  238.     Stm32_Clock_Init(360,25,2,8);   //设置时钟,180Mhz   
  239.     HAL_Init();                     //初始化HAL库
  240.     delay_init(180);                //初始化延时函数
  241.     uart_init(115200);              //初始化USART
  242.     usmart_dev.init(90);            //初始化USMART
  243.     LED_Init();                     //初始化LED
  244.     KEY_Init();                     //初始化按键
  245.     SDRAM_Init();                   //SDRAM初始化
  246.     LCD_Init();                     //LCD初始化
  247.     OV2640_Init();                                    //初始化OV2640
  248.         sw_sdcard_mode();                            //首先切换为SD卡模式
  249.     PCF8574_Init();                 //初始化PCF8574
  250.     W25QXX_Init();                  //初始化W25Q256
  251.     my_mem_init(SRAMIN);            //初始化内部内存池
  252.     my_mem_init(SRAMEX);            //初始化外部SDRAM内存池
  253.     my_mem_init(SRAMCCM);           //初始化内部CCM内存池
  254.     POINT_COLOR=RED;  
  255.     exfuns_init();                            //为fatfs相关变量申请内存  
  256.     f_mount(fs[0],"0:",1);          //挂载SD卡
  257.         f_mount(fs[1],"1:",1);          //挂载SPI FLASH.
  258.         while(font_init())                         //检查字库
  259.         {            
  260.                 LCD_ShowString(30,50,200,16,16,"Font Error!");
  261.                 delay_ms(200);                                  
  262.                 LCD_Fill(30,50,240,66,WHITE);//清除显示             
  263.                 delay_ms(200);                                  
  264.         }           
  265.         Show_Str(30,50,200,16,"阿波罗STM32F4/F7开发板",16,0);                                              
  266.         Show_Str(30,70,200,16,"OV2640照相机实验",16,0);                                             
  267.         Show_Str(30,90,200,16,"KEY0:拍照(bmp格式)",16,0);                                     
  268.         Show_Str(30,110,200,16,"KEY1:拍照(jpg格式)",16,0);                                     
  269.         Show_Str(30,130,200,16,"KEY2:自动对焦",16,0);                                                     
  270.         Show_Str(30,150,200,16,"WK_UP:FullSize/Scale",16,0);                                             
  271.         Show_Str(30,170,200,16,"2016年1月15日",16,0);
  272.     res=f_mkdir("0:/PHOTO");                //创建PHOTO文件夹
  273.         if(res!=FR_EXIST&&res!=FR_OK)         //发生了错误
  274.         {               
  275.                 res=f_mkdir("0:/PHOTO");                //创建PHOTO文件夹               
  276.                 Show_Str(30,190,240,16,"SD卡错误!",16,0);
  277.                 delay_ms(200);                                  
  278.                 Show_Str(30,190,240,16,"拍照功能将不可用!",16,0);
  279.                 delay_ms(200);                                  
  280.                 sd_ok=0;         
  281.         }
  282.     dcmi_line_buf[0]=mymalloc(SRAMIN,jpeg_line_size*4);        //为jpeg dma接收申请内存       
  283.         dcmi_line_buf[1]=mymalloc(SRAMIN,jpeg_line_size*4);        //为jpeg dma接收申请内存       
  284.         jpeg_data_buf=mymalloc(SRAMEX,jpeg_buf_size);                //为jpeg文件申请内存(最大4MB)
  285.         pname=mymalloc(SRAMIN,30);//为带路径的文件名分配30个字节的内存         
  286.         while(pname==NULL||!dcmi_line_buf[0]||!dcmi_line_buf[1]||!jpeg_data_buf)        //内存分配出错
  287.         {            
  288.                 Show_Str(30,190,240,16,"内存分配失败!",16,0);
  289.                 delay_ms(200);                                  
  290.                 LCD_Fill(30,190,240,146,WHITE);//清除显示             
  291.                 delay_ms(200);                                  
  292.         }   
  293.         while(OV2640_Init())//初始化OV2640
  294.         {
  295.                 Show_Str(30,190,240,16,"OV2640 错误!",16,0);
  296.                 delay_ms(200);
  297.             LCD_Fill(30,190,239,206,WHITE);
  298.                 delay_ms(200);
  299.         }
  300.     Show_Str(30,210,230,16,"OV2640 正常",16,0);
  301.         //自动对焦初始化
  302.         OV2640_RGB565_Mode();        //RGB565模式
  303.         OV2640_Light_Mode(0);        //自动模式
  304.         OV2640_Color_Saturation(3);//色彩饱和度0
  305.         OV2640_Brightness(4);        //亮度0
  306.         OV2640_Contrast(3);                //对比度0
  307.         DCMI_Init();                        //DCMI配置
  308.         if(lcdltdc.pwidth!=0)        //RGB屏
  309.         {
  310.                 dcmi_rx_callback=rgblcd_dcmi_rx_callback;//RGB屏接收数据回调函数
  311.                 DCMI_DMA_Init((u32)dcmi_line_buf[0],(u32)dcmi_line_buf[1],lcddev.width/2,DMA_MDATAALIGN_HALFWORD,DMA_MINC_ENABLE);//DCMI DMA配置  
  312.         }else                                        //MCU 屏
  313.         {
  314.                 DCMI_DMA_Init((u32)&LCD->LCD_RAM,0,1,DMA_MDATAALIGN_HALFWORD,DMA_MINC_DISABLE);                        //DCMI DMA配置,MCU屏,竖屏
  315.         }
  316.     if(lcddev.height>800)
  317.         {
  318.                 yoffset=(lcddev.height-800)/2;
  319.                 outputheight=800;
  320.                 SCCB_WR_Reg(0xff,0x01);
  321.                 SCCB_WR_Reg(0x11,0x01);        //需要降低帧率
  322.         }else
  323.         {
  324.                 yoffset=0;
  325.                 outputheight=lcddev.height;
  326.         }
  327.         curline=yoffset;                //行数复位
  328.         OV2640_OutSize_Set(lcddev.width,outputheight);        //满屏缩放显示
  329.         LCD_Clear(BLACK);
  330.         DCMI_Start();                         //启动传输
  331.     while(1)
  332.         {       
  333.                 key=KEY_Scan(0);//不支持连按
  334.                 if((key)&&(key!=KEY2_PRES))
  335.                 {
  336.             if(key==KEY0_PRES)//如果是BMP拍照,则等待1秒钟,去抖动,以获得稳定的bmp照片       
  337.             {
  338.                 delay_ms(300);
  339.                 bmp_request=1;                //请求关闭DCMI
  340.                 while(bmp_request);        //等待请求处理完成
  341.             }else DCMI_Stop();
  342.                         if(key==WKUP_PRES)                //缩放处理
  343.                         {
  344.                                 scale=!scale;  
  345.                                 if(scale==0)
  346.                                 {
  347.                                         OV2640_ImageWin_Set((1600-lcddev.width)/2,(1200-outputheight)/2,lcddev.width,outputheight);//1:1真实尺寸
  348.                                         sprintf((char*)msgbuf,"Full Size 1:1");
  349.                                 }else
  350.                                 {
  351.                     OV2640_ImageWin_Set(0,0,1600,1200);                                //全尺寸缩放
  352.                                         sprintf((char*)msgbuf,"Scale");
  353.                                 }
  354.                                 OV2640_OutSize_Set(lcddev.width,outputheight);
  355.                                 delay_ms(800);        
  356.                         }else if(sd_ok)//SD卡正常才可以拍照
  357.                         {   
  358.                                 sw_sdcard_mode();        //切换为SD卡模式
  359.                                 if(key==KEY0_PRES)        //BMP拍照
  360.                                 {
  361.                                         camera_new_pathname(pname,0);        //得到文件名       
  362.                                         res=bmp_encode(pname,0,yoffset,lcddev.width,outputheight,0);
  363.                                         sw_ov2640_mode();                                //切换为OV2640模式
  364. ……………………

  365. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码



所有资料51hei提供下载:
ATK-OV2640摄像1头模块.rar (19.37 MB, 下载次数: 116)



评分

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

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏4 分享淘帖 顶1 踩
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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