找回密码
 立即注册

QQ登录

只需一步,快速开始

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

stm32F4+OV2640图像rgb转yuv转二值化DMA传输源码

[复制链接]
跳转到指定楼层
楼主
ID:291662 发表于 2018-3-14 10:26 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
MCU:STM32F407摄像头:OV2640
处理方法:rgb转yuv转二值化图
数据传输:DMA

单片机源程序如下:
  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 "usmart.h"  
  8. #include "usart2.h"  
  9. #include "timer.h"
  10. #include "ov2640.h"
  11. #include "dcmi.h"
  12. #define jpeg_buf_size 31*1024                          //定义JPEG数据缓存jpeg_buf的大小(*4字节)
  13. u32 black=0,white=0;
  14. float  precent;
  15. u16  ov2640_mode=0,i=0,cnt,threshold=60,line;                                                //工作模式:0,RGB565模式;1,JPEG模式
  16. __align(4) u32 jpeg_buf[jpeg_buf_size];        //JPEG数据缓存buf
  17. volatile u32 jpeg_data_len=0;                         //buf中的JPEG有效数据长度
  18. volatile u8 jpeg_data_ok=0;                                //JPEG数据采集完成标志
  19. u16 fps[320];
  20. u16 yuv[320];

  21. u16 yuv422_y_to_bitmap(u8 threshold,u16 yuv422)
  22. {
  23.         u8 temp;        //用于储存yuv422格式数据中的亮度值Y量
  24.         temp = (u8)(yuv422>>8);//把yuv422格式数据中的亮度值Y量提取出来赋值给temp变量(Y值在高字节,根据OV2640寄存器设置决定)
  25.         if(temp >= threshold)//如果,Y值 大于等于阀值
  26.         {
  27.         //白
  28.                 return 1;
  29.         }
  30.         else
  31.         {        //黑
  32.                 return 0;
  33.         }
  34. }

  35. //中断处理函数
  36. void DCMI_IRQHandler(void)
  37. {
  38.         if(DCMI_GetITStatus(DCMI_IT_FRAME)==SET)//捕获到一帧图像
  39.         {
  40.                 DCMI_ClearITPendingBit(DCMI_IT_FRAME);//清除帧中断
  41.                 LED1=!LED1;
  42.                 precent=((black/1.0)/((black/1.0+white/1.0)))*100;
  43.                 USART_SendData(USART1,precent);
  44.                 black=0;white=0;       
  45.           DCMI_Start();                 //启动传输
  46.                 DCMI_CaptureCmd(ENABLE);
  47.         }
  48.         if (DCMI_GetITStatus(DCMI_IT_LINE) != RESET)//DCMI_ITConfig(DCMI_IT_LINE, ENABLE);
  49.   {
  50.     DCMI_ClearITPendingBit(DCMI_IT_LINE);
  51.                 for(i=0;i<320;i++)
  52.                 {
  53.                         cnt=yuv422_y_to_bitmap(threshold,fps[i]);
  54.                         if(cnt==1)
  55.                         {
  56.                                 white++;
  57.                                 LCD->LCD_RAM=0xffff;
  58.                         }
  59.                         else
  60.                         {
  61.                                 LCD->LCD_RAM=0x0000;
  62.                           black++;
  63.                         }
  64.                 }
  65.   }
  66. }
  67. //RGB565测试
  68. //RGB数据直接显示在LCD上面
  69. void yuv_test(void)
  70. {
  71.   u8 key;       
  72.         LCD_Clear(WHITE);
  73.   POINT_COLOR=RED;
  74.         OV2640_YUV422_Mode();        //RGB565模式
  75.         My_DCMI_Init();                        //DCMI配置
  76.         DCMI_DMA_Init((u32)fps,320,DMA_MemoryDataSize_HalfWord,DMA_MemoryInc_Enable);//DCMI DMA配置  
  77.         OV2640_OutSize_Set(lcddev.width,lcddev.height);
  78.         DCMI_Start();                 //启动传输
  79.         while(1)
  80.         {
  81.                         key=KEY_Scan(1); //按键扫描函数
  82.                         if(key)
  83.                         {
  84.                                 switch(key)
  85.                                 {                                    
  86.                                         case WKUP_PRES:        //阀值+1
  87.                                                 threshold++;break;
  88.                                         case KEY1_PRES:        //阀值-1
  89.                                                 threshold--;break;
  90.                                 }
  91.                         }       
  92.         }   
  93. }
  94. int main(void)
  95. {
  96.         NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
  97.         delay_init(168);  //初始化延时函数
  98.         uart_init(115200);                //初始化串口波特率为115200
  99.         usart2_init(42,115200);                //初始化串口2波特率为115200
  100.         LED_Init();                                        //初始化LED
  101.         LCD_Init();                                        //LCD初始化  
  102.         KEY_Init();                                        //按键初始化
  103.         TIM3_Int_Init(10000-1,8400-1);//10Khz计数,1秒钟中断一次
  104.         usmart_dev.init(84);                //初始化USMART
  105.         POINT_COLOR=RED;//设置字体为红色  
  106.         while(OV2640_Init())//初始化OV2640
  107.         {
  108.                 LCD_ShowString(30,130,240,16,16,"OV2640 ERR");
  109.                 delay_ms(200);
  110.           LCD_Fill(30,130,239,170,WHITE);
  111.                 delay_ms(200);
  112.         }          
  113.         while(1)
  114.         {       
  115.     yuv_test();
  116.         }               
  117. }
  118.        
复制代码

所有资料51hei提供下载:
摄像头实验.rar (590.69 KB, 下载次数: 133)


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

使用道具 举报

沙发
ID:273692 发表于 2018-5-17 15:41 | 只看该作者
程序出现了3分屏啊,且屏幕利用率低啊
回复

使用道具 举报

板凳
ID:510730 发表于 2019-5-8 17:04 | 只看该作者
这个程序是怎么用一个循环扫完所有点的,我想得到所有黑点的坐标,可是我只找到一个循环,找不到横纵坐标
回复

使用道具 举报

地板
ID:370706 发表于 2019-7-21 16:23 | 只看该作者
学习下
回复

使用道具 举报

5#
ID:590012 发表于 2019-8-5 16:03 | 只看该作者
aaaa1254 发表于 2018-5-17 15:41
程序出现了3分屏啊,且屏幕利用率低啊

你解决这个问题了  可以教教我不
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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