• (40)STM32——OV2640摄像头实验


    目录

    学习目标

    运行结果

    内容

    OV2640

    特点

    时序

    帧输出时序

    配置

    DCMI

    特点

    信号

    DMA

    寄存器

    配置

    硬件连接

    代码

    总结 


    学习目标

            今天我们要学习的是OV2640摄像头实验,采用的是DCMI接口,进行传输。个人觉得难度较大,加上没有相应的串口线,导致部分实验无法进行,所以就先讲解理论知识,等串口线到了再把电脑端的实验补上。

    运行结果

    摄像头实验

    内容

    OV2640

            OV2640 是 OV(OmniVision)公司生产的一颗 1/4 寸的 CMOS UXGA(1632*1232)图像传感器。该传感器体积小、工作电压低,提供单片 UXGA 摄像头和影像处理器的所有功能。通过 SCCB 总线控制,可以输出整帧、子采样、缩放和取窗口等方式的各种分辨率 8/10 位影像数据。该产品 UXGA 图像最高达到 15 帧/秒(SVGA 可达 30 帧,CIF 可达 60 帧)。用户可以完全控制图像质量、数据格式和传输方式。所有图像处理功能过程包括伽玛曲线、白平衡、对比度、色度等都可以通过 SCCB 接口编程。

    特点

    • 高灵敏度、低电压适合嵌入式应用
    • 标准的 SCCB 接口,兼容 IIC 接口
    • 支持 RawRGB、RGB(RGB565/RGB555)、GRB422、YUV(422/420)和 YCbCr(422) 输出格式
    • 支持 UXGA、SXGA、SVGA 以及按比例缩小到从 SXGA 到 40*30 的任何尺寸
    • 支持自动曝光控制、自动增益控制、自动白平衡、自动消除灯光条纹、自动黑电平校 准等自动控制功能。同时支持色饱和度、色相、伽马、锐度等设置。
    • 支持闪光灯
    • 支持图像缩放、平移和窗口设置
    • 支持图像压缩,即可输出 JPEG 图像数据
    • 自带嵌入式微处理器
    • UXGA , 即 分 辨 率 位 1600*1200 的输出格式,类似的还有: SXGA(1280*1024) 、 WXGA+(1440*900)、XVGA(1280*960)、WXGA(1280*800)、XGA(1024*768)、SVGA(800*600)、 VGA(640*480)、CIF(352*288)、WQVGA(400*240)、QCIF(176*144)和 QQVGA(160*120)等。
    • PCLK,即像素时钟,一个 PCLK 时钟,输出一个像素(或半个像素)。
    • VSYNC,即帧同步信号。
    • HREF /HSYNC,即行同步信号。

            我们的数据输出(通过 Y[9:0]),就是通过PCLK,VSYNC和HREF /HSYNC来实现的,接下来我们就来讲解一下OV2640的时序部分。

    时序

            图像数据在 HREF 为高的时候输出,当 HREF 变高后,每一个 PCLK 时钟,输出一个 8 位/10 位数据。我们采用 8 位接口,所以每个 PCLK 输出 1 个字节,且在 RGB/YUV 输出格式下,每个 tp=2 个 Tpclk,如果是 Raw 格式,则一个 tp=1 个 Tpclk。比如我们采用 UXGA 时序,RGB565 格式输出,每 2 个字节组成一个像素的颜色(高低字节顺序可通过 0XDA 寄存器设置),这样每行输出总共有 1600*2 个 PCLK 周期,输出 1600*2 个字节。

    帧输出时序

            我们以UXGA举例, 首先VSYNC发送一个高电平数据表示起始信号,然后在HREF为高电平期间采集有效数据,一共1600条,然后重复1200次,就达到了输出的效果。

    窗口设置

            传感器窗口设置,该功能允许用户设置整个传感器区域(1632*1220)的感兴趣部分,也就是在传感器里面开窗,开窗范围从 2*2~1632*1220 都可以设置,不过要求这个窗口必须大于等于随后设置的图像尺寸。传感器窗口设置,通过:0X03/0X19/0X1A/0X07/0X17/0X18 等寄存器设置,使用的函数是OV2640_Window_Set。

            图像尺寸设置,也就是 DSP 输出(最终输出到 LCD 的)图像的最大尺寸,该尺寸要小于等于前面我们传感器窗口设置所设定的窗口尺寸。图像尺寸通过:0XC0/0XC1/0X8C 等寄存器设置。使用的函数是OV2640_ImageSize_Set。

            图像窗口设置,这里起始和前面的传感器窗口设置类似,只是这个窗口是在我们前面设置的图像尺寸里面,再一次设置窗口大小,该窗口必须小于等于前面设置的图像尺寸。该窗口设置后的图像范围,将用于输出到外部。图像窗口设置通过:0X51/0X52/0X53/0X54/0X55/0X57等寄存器设置。使用的函数是OV2640_ImageWin_Set。

            图像输出大小设置,这是最终输出到外部的图像尺寸。该设置将图像窗口设置所决定的窗口大小,通过内部 DSP 处理,缩放成我们输出到外部的图像大小。该设置将会对图像进行缩放处理,如果设置的图像输出大小不等于图像窗口设置图像大小,那么图像就会被缩放处理,只有这两者设置一样大的时候,输出比例才是 1:1 的。使用的函数是OV2640_OutSize_Set。

    整体效果如下所示

    配置

    初始化

            这是初始化的流程,首先初始化好IO口,然后上电复位,读取ID,执行初始化序列,这个是OV2640厂家做好的,我们就不需要重新写过了。 

     读取图像数据

    DCMI

            STM32F4 自带了一个数字摄像头(DCMI)接口,该接口是一个同步并行接口,能够接收外部 8 位、10 位、12 位或 14 位 CMOS 摄像头模块发出的高速数据流。可支持不同的数据格式:YCbCr4:2:2/RGB565 逐行视频和压缩数据 (JPEG)。

            DCMI 接口是一个同步并行接口,可接收高速(可达 54 MB/s)数据流。该接口包含多达14 条数据线(D13-D0)和一条像素时钟线(PIXCLK)。像素时钟的极性可以编程,因此可以在像素时钟的上升沿或下降沿捕获数据。

    特点

    • 8 位、10 位、12 位或 14 位并行接口
    • 内嵌码/外部行同步和帧同步
    • 连续模式或快照模式
    • 裁剪功能
    • 支持以下数据格式:
      • 8/10/12/14 位逐行视频:单色或原始拜尔(Bayer)格式
      • YCbCr 4:2:2 逐行视频
      • RGB 565 逐行视频
      • 压缩数据:JPEG

    信号

    1. 数据输入(D[0:13]),用于接摄像头的数据输出,接 OV2640 我们只用了 8 位数据。
    2. 水平同步(行同步)输入(HSYNC),用于接摄像头的 HSYNC/HREF 信号。
    3. 垂直同步(场同步)输入(VSYNC),用于接摄像头的 VSYNC 信号。
    4. 像素时钟输入(PIXCLK),用于接摄像头的 PCLK 信号。

            DCMI 接口的数据与 PIXCLK(即 PCLK)保持同步,并根据像素时钟的极性在像素时钟上升沿/下降沿发生变化。HSYNC(HREF)信号指示行的开始/结束,VSYNC 信号指示帧的开始/结束。DCMI 信号波形如图所示:

            上图中,对应设置为:DCMI_PIXCLK 的捕获沿为下降沿,DCMI_HSYNC 和 DCMI_VSYNC 的有效状态为 1,注意,这里的有效状态实际上对应的是指示数据在并行接口上无效时, HSYNC/VSYNC 引脚上面的引脚电平。

            我们用到 DCMI 的 8 位数据宽度,通过设置 DCMI_CR 中的 EDM[1:0]=00 设置。此时 DCMI_D0~D7 有效,DCMI_D8~D13 上的数据则忽略,这个时候,每次需要 4 个像素时钟来捕获一个 32 位数据。捕获的第一个数据存放在 32 位字的 LSB 位置,第四个数据存放在 32 位字的 MSB 位置 ,捕获数据字节在 32 位字中的排布如表所示:

    DMA

            DCMI 接口支持 DMA 传输,当 DCMI_CR 寄存器中的 CAPTURE 位置 1 时,激活 DMA 接口。摄像头接口每次在其寄存器中收到一个完整的 32 位数据块时,都将触发一个 DMA请求。

    寄存器

            寄存器就不详细介绍了。

    配置

    1. 配置 OV2640 控制引脚,并配置 OV2640 工作模式。
    2. 配置相关引脚的模式和复用功能(AF13),使能时钟。
    3. 配置 DCMI 相关设置。
    4. 配置 DMA。
    5. 设置 OV2640 的图像输出大小,使能 DCMI 捕获。

    硬件连接

    代码

    1. //dcmi.c
    2. #include "sys.h"
    3. #include "dcmi.h"
    4. #include "led.h"
    5. #include "ov2640.h"
    6. u8 ov_frame=0; //帧率
    7. extern void jpeg_data_process(void); //JPEG数据处理函数
    8. DCMI_InitTypeDef DCMI_InitStructure;
    9. //DCMI中断服务函数
    10. void DCMI_IRQHandler(void)
    11. {
    12. if(DCMI_GetITStatus(DCMI_IT_FRAME)==SET)//捕获到一帧图像
    13. {
    14. jpeg_data_process(); //jpeg数据处理
    15. DCMI_ClearITPendingBit(DCMI_IT_FRAME);//清除帧中断
    16. LED1=!LED1;
    17. ov_frame++;
    18. }
    19. }
    20. //DCMI DMA配置
    21. //DMA_Memory0BaseAddr:存储器地址 将要存储摄像头数据的内存地址(也可以是外设地址)
    22. //DMA_BufferSize:存储器长度 0~65535
    23. //DMA_MemoryDataSize:存储器位宽
    24. //DMA_MemoryDataSize:存储器位宽 @defgroup DMA_memory_data_size :DMA_MemoryDataSize_Byte/DMA_MemoryDataSize_HalfWord/DMA_MemoryDataSize_Word
    25. //DMA_MemoryInc:存储器增长方式 @defgroup DMA_memory_incremented_mode /** @defgroup DMA_memory_incremented_mode : DMA_MemoryInc_Enable/DMA_MemoryInc_Disable
    26. void DCMI_DMA_Init(u32 DMA_Memory0BaseAddr,u16 DMA_BufferSize,u32 DMA_MemoryDataSize,u32 DMA_MemoryInc)
    27. {
    28. DMA_InitTypeDef DMA_InitStructure;
    29. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2,ENABLE);//DMA2时钟使能
    30. DMA_DeInit(DMA2_Stream1);
    31. while (DMA_GetCmdStatus(DMA2_Stream1) != DISABLE){}//等待DMA2_Stream1可配置
    32. /* 配置 DMA Stream */
    33. DMA_InitStructure.DMA_Channel = DMA_Channel_1; //通道1 DCMI通道
    34. DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&DCMI->DR;//外设地址为:DCMI->DR
    35. DMA_InitStructure.DMA_Memory0BaseAddr = DMA_Memory0BaseAddr;//DMA 存储器0地址
    36. DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;//外设到存储器模式
    37. DMA_InitStructure.DMA_BufferSize = DMA_BufferSize;//数据传输量
    38. DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设非增量模式
    39. DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc;//存储器增量模式
    40. DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;//外设数据长度:32位
    41. DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize;//存储器数据长度
    42. DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;// 使用循环模式
    43. DMA_InitStructure.DMA_Priority = DMA_Priority_High;//高优先级
    44. DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Enable; //FIFO模式
    45. DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;//使用全FIFO
    46. DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;//外设突发单次传输
    47. DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;//存储器突发单次传输
    48. DMA_Init(DMA2_Stream1, &DMA_InitStructure);//初始化DMA Stream
    49. }
    50. //DCMI初始化
    51. void My_DCMI_Init(void)
    52. {
    53. GPIO_InitTypeDef GPIO_InitStructure;
    54. NVIC_InitTypeDef NVIC_InitStructure;
    55. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA|RCC_AHB1Periph_GPIOB|RCC_AHB1Periph_GPIOC|RCC_AHB1Periph_GPIOE, ENABLE);//使能GPIOA B C E 时钟
    56. RCC_AHB2PeriphClockCmd(RCC_AHB2Periph_DCMI,ENABLE);//使能DCMI时钟
    57. //PA4/6初始化设置
    58. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_6;//PA4/6 复用功能输出
    59. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; //复用功能输出
    60. GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
    61. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHz
    62. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
    63. GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化
    64. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7|GPIO_Pin_6;// PB6/7 复用功能输出
    65. GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化
    66. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7|GPIO_Pin_8|GPIO_Pin_9|GPIO_Pin_11;//PC6/7/8/9/11 复用功能输出
    67. GPIO_Init(GPIOC, &GPIO_InitStructure);//初始化
    68. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5|GPIO_Pin_6;//PE5/6 复用功能输出
    69. GPIO_Init(GPIOE, &GPIO_InitStructure);//初始化
    70. GPIO_PinAFConfig(GPIOA,GPIO_PinSource4,GPIO_AF_DCMI); //PA4,AF13 DCMI_HSYNC
    71. GPIO_PinAFConfig(GPIOA,GPIO_PinSource6,GPIO_AF_DCMI); //PA6,AF13 DCMI_PCLK
    72. GPIO_PinAFConfig(GPIOB,GPIO_PinSource7,GPIO_AF_DCMI); //PB7,AF13 DCMI_VSYNC
    73. GPIO_PinAFConfig(GPIOC,GPIO_PinSource6,GPIO_AF_DCMI); //PC6,AF13 DCMI_D0
    74. GPIO_PinAFConfig(GPIOC,GPIO_PinSource7,GPIO_AF_DCMI); //PC7,AF13 DCMI_D1
    75. GPIO_PinAFConfig(GPIOC,GPIO_PinSource8,GPIO_AF_DCMI); //PC8,AF13 DCMI_D2
    76. GPIO_PinAFConfig(GPIOC,GPIO_PinSource9,GPIO_AF_DCMI); //PC9,AF13 DCMI_D3
    77. GPIO_PinAFConfig(GPIOC,GPIO_PinSource11,GPIO_AF_DCMI);//PC11,AF13 DCMI_D4
    78. GPIO_PinAFConfig(GPIOB,GPIO_PinSource6,GPIO_AF_DCMI); //PB6,AF13 DCMI_D5
    79. GPIO_PinAFConfig(GPIOE,GPIO_PinSource5,GPIO_AF_DCMI); //PE5,AF13 DCMI_D6
    80. GPIO_PinAFConfig(GPIOE,GPIO_PinSource6,GPIO_AF_DCMI); //PE6,AF13 DCMI_D7
    81. DCMI_DeInit();//清除原来的设置
    82. DCMI_InitStructure.DCMI_CaptureMode=DCMI_CaptureMode_Continuous;//连续模式
    83. DCMI_InitStructure.DCMI_CaptureRate=DCMI_CaptureRate_All_Frame;//全帧捕获
    84. DCMI_InitStructure.DCMI_ExtendedDataMode= DCMI_ExtendedDataMode_8b;//8位数据格式
    85. DCMI_InitStructure.DCMI_HSPolarity= DCMI_HSPolarity_Low;//HSYNC 低电平有效
    86. DCMI_InitStructure.DCMI_PCKPolarity= DCMI_PCKPolarity_Rising;//PCLK 上升沿有效
    87. DCMI_InitStructure.DCMI_SynchroMode= DCMI_SynchroMode_Hardware;//硬件同步HSYNC,VSYNC
    88. DCMI_InitStructure.DCMI_VSPolarity=DCMI_VSPolarity_Low;//VSYNC 低电平有效
    89. DCMI_Init(&DCMI_InitStructure);
    90. DCMI_ITConfig(DCMI_IT_FRAME,ENABLE);//开启帧中断
    91. DCMI_Cmd(ENABLE); //DCMI使能
    92. NVIC_InitStructure.NVIC_IRQChannel = DCMI_IRQn;
    93. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;//抢占优先级1
    94. NVIC_InitStructure.NVIC_IRQChannelSubPriority =0; //子优先级3
    95. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
    96. NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器、
    97. }
    98. //DCMI,启动传输
    99. void DCMI_Start(void)
    100. {
    101. DMA_Cmd(DMA2_Stream1, ENABLE);//开启DMA2,Stream1
    102. DCMI_CaptureCmd(ENABLE);//DCMI捕获使能
    103. }
    104. //DCMI,关闭传输
    105. void DCMI_Stop(void)
    106. {
    107. DCMI_CaptureCmd(DISABLE);//DCMI捕获使关闭
    108. while(DCMI->CR&0X01); //等待传输结束
    109. DMA_Cmd(DMA2_Stream1,DISABLE);//关闭DMA2,Stream1
    110. }
    111. //以下两个函数,供usmart调用,用于调试代码
    112. //DCMI设置显示窗口
    113. //sx,sy;LCD的起始坐标
    114. //width,height:LCD显示范围.
    115. void DCMI_Set_Window(u16 sx,u16 sy,u16 width,u16 height)
    116. {
    117. DCMI_Stop();
    118. // LCD_Clear(WHITE);
    119. // LCD_Set_Window(sx,sy,width,height);
    120. OV2640_OutSize_Set(width,height);
    121. // LCD_SetCursor(0,0);
    122. // LCD_WriteRAM_Prepare(); //开始写入GRAM
    123. DMA_Cmd(DMA2_Stream1,ENABLE); //开启DMA2,Stream1
    124. DCMI_CaptureCmd(ENABLE);//DCMI捕获使能
    125. }
    126. //通过usmart调试,辅助测试用.
    127. //pclk/hsync/vsync:三个信号的有限电平设置
    128. void DCMI_CR_Set(u8 pclk,u8 hsync,u8 vsync)
    129. {
    130. DCMI_DeInit();//清除原来的设置
    131. DCMI_InitStructure.DCMI_CaptureMode=DCMI_CaptureMode_Continuous;//连续模式
    132. DCMI_InitStructure.DCMI_CaptureRate=DCMI_CaptureRate_All_Frame;//全帧捕获
    133. DCMI_InitStructure.DCMI_ExtendedDataMode= DCMI_ExtendedDataMode_8b;//8位数据格式
    134. DCMI_InitStructure.DCMI_HSPolarity= hsync<<6;//HSYNC 低电平有效
    135. DCMI_InitStructure.DCMI_PCKPolarity= pclk<<5;//PCLK 上升沿有效
    136. DCMI_InitStructure.DCMI_SynchroMode= DCMI_SynchroMode_Hardware;//硬件同步HSYNC,VSYNC
    137. DCMI_InitStructure.DCMI_VSPolarity=vsync<<7;//VSYNC 低电平有效
    138. DCMI_Init(&DCMI_InitStructure);
    139. DCMI_CaptureCmd(ENABLE);//DCMI捕获使能
    140. DCMI_Cmd(ENABLE); //DCMI使能
    141. }
    1. // sccb.c
    2. #include "sys.h"
    3. #include "sccb.h"
    4. #include "delay.h"
    5. //初始化SCCB接口
    6. void SCCB_Init(void)
    7. {
    8. GPIO_InitTypeDef GPIO_InitStructure;
    9. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);//使能GPIOD时钟
    10. //GPIOF9,F10初始化设置
    11. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6|GPIO_Pin_7;//PD6,7 推挽输出
    12. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; //PD6,7 推挽输出
    13. GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
    14. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//100MHz
    15. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
    16. GPIO_Init(GPIOD, &GPIO_InitStructure);//初始化
    17. GPIO_SetBits(GPIOD,GPIO_Pin_6|GPIO_Pin_7);
    18. SCCB_SDA_OUT();
    19. }
    20. //SCCB起始信号
    21. //当时钟为高的时候,数据线的高到低,为SCCB起始信号
    22. //在激活状态下,SDA和SCL均为低电平
    23. void SCCB_Start(void)
    24. {
    25. SCCB_SDA=1; //数据线高电平
    26. SCCB_SCL=1; //在时钟线高的时候数据线由高至低
    27. delay_us(50);
    28. SCCB_SDA=0;
    29. delay_us(50);
    30. SCCB_SCL=0; //数据线恢复低电平,单操作函数必要
    31. }
    32. //SCCB停止信号
    33. //当时钟为高的时候,数据线的低到高,为SCCB停止信号
    34. //空闲状况下,SDA,SCL均为高电平
    35. void SCCB_Stop(void)
    36. {
    37. SCCB_SDA=0;
    38. delay_us(50);
    39. SCCB_SCL=1;
    40. delay_us(50);
    41. SCCB_SDA=1;
    42. delay_us(50);
    43. }
    44. //产生NA信号
    45. void SCCB_No_Ack(void)
    46. {
    47. delay_us(50);
    48. SCCB_SDA=1;
    49. SCCB_SCL=1;
    50. delay_us(50);
    51. SCCB_SCL=0;
    52. delay_us(50);
    53. SCCB_SDA=0;
    54. delay_us(50);
    55. }
    56. //SCCB,写入一个字节
    57. //返回值:0,成功;1,失败.
    58. u8 SCCB_WR_Byte(u8 dat)
    59. {
    60. u8 j,res;
    61. for(j=0;j<8;j++) //循环8次发送数据
    62. {
    63. if(dat&0x80)SCCB_SDA=1;
    64. else SCCB_SDA=0;
    65. dat<<=1;
    66. delay_us(50);
    67. SCCB_SCL=1;
    68. delay_us(50);
    69. SCCB_SCL=0;
    70. }
    71. SCCB_SDA_IN(); //设置SDA为输入
    72. delay_us(50);
    73. SCCB_SCL=1; //接收第九位,以判断是否发送成功
    74. delay_us(50);
    75. if(SCCB_READ_SDA)res=1; //SDA=1发送失败,返回1
    76. else res=0; //SDA=0发送成功,返回0
    77. SCCB_SCL=0;
    78. SCCB_SDA_OUT(); //设置SDA为输出
    79. return res;
    80. }
    81. //SCCB 读取一个字节
    82. //在SCL的上升沿,数据锁存
    83. //返回值:读到的数据
    84. u8 SCCB_RD_Byte(void)
    85. {
    86. u8 temp=0,j;
    87. SCCB_SDA_IN(); //设置SDA为输入
    88. for(j=8;j>0;j--) //循环8次接收数据
    89. {
    90. delay_us(50);
    91. SCCB_SCL=1;
    92. temp=temp<<1;
    93. if(SCCB_READ_SDA)temp++;
    94. delay_us(50);
    95. SCCB_SCL=0;
    96. }
    97. SCCB_SDA_OUT(); //设置SDA为输出
    98. return temp;
    99. }
    100. //写寄存器
    101. //返回值:0,成功;1,失败.
    102. u8 SCCB_WR_Reg(u8 reg,u8 data)
    103. {
    104. u8 res=0;
    105. SCCB_Start(); //启动SCCB传输
    106. if(SCCB_WR_Byte(SCCB_ID))res=1; //写器件ID
    107. delay_us(100);
    108. if(SCCB_WR_Byte(reg))res=1; //写寄存器地址
    109. delay_us(100);
    110. if(SCCB_WR_Byte(data))res=1; //写数据
    111. SCCB_Stop();
    112. return res;
    113. }
    114. //读寄存器
    115. //返回值:读到的寄存器值
    116. u8 SCCB_RD_Reg(u8 reg)
    117. {
    118. u8 val=0;
    119. SCCB_Start(); //启动SCCB传输
    120. SCCB_WR_Byte(SCCB_ID); //写器件ID
    121. delay_us(100);
    122. SCCB_WR_Byte(reg); //写寄存器地址
    123. delay_us(100);
    124. SCCB_Stop();
    125. delay_us(100);
    126. //设置寄存器地址后,才是读
    127. SCCB_Start();
    128. SCCB_WR_Byte(SCCB_ID|0X01); //发送读命令
    129. delay_us(100);
    130. val=SCCB_RD_Byte(); //读取数据
    131. SCCB_No_Ack();
    132. SCCB_Stop();
    133. return val;
    134. }
    1. //ov2640.c
    2. #include "sys.h"
    3. #include "ov2640.h"
    4. #include "ov2640cfg.h"
    5. #include "timer.h"
    6. #include "delay.h"
    7. #include "usart.h"
    8. #include "sccb.h"
    9. //初始化OV2640
    10. //配置完以后,默认输出是1600*1200尺寸的图片!!
    11. //返回值:0,成功
    12. // 其他,错误代码
    13. u8 OV2640_Init(void)
    14. {
    15. u16 i=0;
    16. u16 reg;
    17. //设置IO
    18. GPIO_InitTypeDef GPIO_InitStructure;
    19. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOG, ENABLE);
    20. //GPIOG9,15初始化设置
    21. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9|GPIO_Pin_15;//PG9,15推挽输出
    22. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; //推挽输出
    23. GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
    24. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//100MHz
    25. GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
    26. GPIO_Init(GPIOG, &GPIO_InitStructure);//初始化
    27. OV2640_PWDN=0; //POWER ON
    28. delay_ms(10);
    29. OV2640_RST=0; //复位OV2640
    30. delay_ms(10);
    31. OV2640_RST=1; //结束复位
    32. SCCB_Init(); //初始化SCCB 的IO口
    33. SCCB_WR_Reg(OV2640_DSP_RA_DLMT, 0x01); //操作sensor寄存器
    34. SCCB_WR_Reg(OV2640_SENSOR_COM7, 0x80); //软复位OV2640
    35. delay_ms(50);
    36. reg=SCCB_RD_Reg(OV2640_SENSOR_MIDH); //读取厂家ID 高八位
    37. reg<<=8;
    38. reg|=SCCB_RD_Reg(OV2640_SENSOR_MIDL); //读取厂家ID 低八位
    39. if(reg!=OV2640_MID)
    40. {
    41. printf("MID:%d\r\n",reg);
    42. return 1;
    43. }
    44. reg=SCCB_RD_Reg(OV2640_SENSOR_PIDH); //读取厂家ID 高八位
    45. reg<<=8;
    46. reg|=SCCB_RD_Reg(OV2640_SENSOR_PIDL); //读取厂家ID 低八位
    47. if(reg!=OV2640_PID)
    48. {
    49. printf("HID:%d\r\n",reg);
    50. return 2;
    51. }
    52. //初始化 OV2640,采用UXGA分辨率(1600*1200)
    53. for(i=0;i<sizeof(ov2640_uxga_init_reg_tbl)/2;i++)
    54. {
    55. SCCB_WR_Reg(ov2640_uxga_init_reg_tbl[i][0],ov2640_uxga_init_reg_tbl[i][1]);
    56. }
    57. return 0x00; //ok
    58. }
    59. //OV2640切换为JPEG模式
    60. void OV2640_JPEG_Mode(void)
    61. {
    62. u16 i=0;
    63. //设置:YUV422格式
    64. for(i=0;i<(sizeof(ov2640_yuv422_reg_tbl)/2);i++)
    65. {
    66. SCCB_WR_Reg(ov2640_yuv422_reg_tbl[i][0],ov2640_yuv422_reg_tbl[i][1]);
    67. }
    68. //设置:输出JPEG数据
    69. for(i=0;i<(sizeof(ov2640_jpeg_reg_tbl)/2);i++)
    70. {
    71. SCCB_WR_Reg(ov2640_jpeg_reg_tbl[i][0],ov2640_jpeg_reg_tbl[i][1]);
    72. }
    73. }
    74. //OV2640切换为RGB565模式
    75. void OV2640_RGB565_Mode(void)
    76. {
    77. u16 i=0;
    78. //设置:RGB565输出
    79. for(i=0;i<(sizeof(ov2640_rgb565_reg_tbl)/2);i++)
    80. {
    81. SCCB_WR_Reg(ov2640_rgb565_reg_tbl[i][0],ov2640_rgb565_reg_tbl[i][1]);
    82. }
    83. }
    84. //自动曝光设置参数表,支持5个等级
    85. const static u8 OV2640_AUTOEXPOSURE_LEVEL[5][8]=
    86. {
    87. {
    88. 0xFF,0x01,
    89. 0x24,0x20,
    90. 0x25,0x18,
    91. 0x26,0x60,
    92. },
    93. {
    94. 0xFF,0x01,
    95. 0x24,0x34,
    96. 0x25,0x1c,
    97. 0x26,0x00,
    98. },
    99. {
    100. 0xFF,0x01,
    101. 0x24,0x3e,
    102. 0x25,0x38,
    103. 0x26,0x81,
    104. },
    105. {
    106. 0xFF,0x01,
    107. 0x24,0x48,
    108. 0x25,0x40,
    109. 0x26,0x81,
    110. },
    111. {
    112. 0xFF,0x01,
    113. 0x24,0x58,
    114. 0x25,0x50,
    115. 0x26,0x92,
    116. },
    117. };
    118. //OV2640自动曝光等级设置
    119. //level:0~4
    120. void OV2640_Auto_Exposure(u8 level)
    121. {
    122. u8 i;
    123. u8 *p=(u8*)OV2640_AUTOEXPOSURE_LEVEL[level];
    124. for(i=0;i<4;i++)
    125. {
    126. SCCB_WR_Reg(p[i*2],p[i*2+1]);
    127. }
    128. }
    129. //白平衡设置
    130. //0:自动
    131. //1:太阳sunny
    132. //2,阴天cloudy
    133. //3,办公室office
    134. //4,家里home
    135. void OV2640_Light_Mode(u8 mode)
    136. {
    137. u8 regccval=0X5E;//Sunny
    138. u8 regcdval=0X41;
    139. u8 regceval=0X54;
    140. switch(mode)
    141. {
    142. case 0://auto
    143. SCCB_WR_Reg(0XFF,0X00);
    144. SCCB_WR_Reg(0XC7,0X10);//AWB ON
    145. return;
    146. case 2://cloudy
    147. regccval=0X65;
    148. regcdval=0X41;
    149. regceval=0X4F;
    150. break;
    151. case 3://office
    152. regccval=0X52;
    153. regcdval=0X41;
    154. regceval=0X66;
    155. break;
    156. case 4://home
    157. regccval=0X42;
    158. regcdval=0X3F;
    159. regceval=0X71;
    160. break;
    161. }
    162. SCCB_WR_Reg(0XFF,0X00);
    163. SCCB_WR_Reg(0XC7,0X40); //AWB OFF
    164. SCCB_WR_Reg(0XCC,regccval);
    165. SCCB_WR_Reg(0XCD,regcdval);
    166. SCCB_WR_Reg(0XCE,regceval);
    167. }
    168. //色度设置
    169. //0:-2
    170. //1:-1
    171. //2,0
    172. //3,+1
    173. //4,+2
    174. void OV2640_Color_Saturation(u8 sat)
    175. {
    176. u8 reg7dval=((sat+2)<<4)|0X08;
    177. SCCB_WR_Reg(0XFF,0X00);
    178. SCCB_WR_Reg(0X7C,0X00);
    179. SCCB_WR_Reg(0X7D,0X02);
    180. SCCB_WR_Reg(0X7C,0X03);
    181. SCCB_WR_Reg(0X7D,reg7dval);
    182. SCCB_WR_Reg(0X7D,reg7dval);
    183. }
    184. //亮度设置
    185. //0:(0X00)-2
    186. //1:(0X10)-1
    187. //2,(0X20) 0
    188. //3,(0X30)+1
    189. //4,(0X40)+2
    190. void OV2640_Brightness(u8 bright)
    191. {
    192. SCCB_WR_Reg(0xff, 0x00);
    193. SCCB_WR_Reg(0x7c, 0x00);
    194. SCCB_WR_Reg(0x7d, 0x04);
    195. SCCB_WR_Reg(0x7c, 0x09);
    196. SCCB_WR_Reg(0x7d, bright<<4);
    197. SCCB_WR_Reg(0x7d, 0x00);
    198. }
    199. //对比度设置
    200. //0:-2
    201. //1:-1
    202. //2,0
    203. //3,+1
    204. //4,+2
    205. void OV2640_Contrast(u8 contrast)
    206. {
    207. u8 reg7d0val=0X20;//默认为普通模式
    208. u8 reg7d1val=0X20;
    209. switch(contrast)
    210. {
    211. case 0://-2
    212. reg7d0val=0X18;
    213. reg7d1val=0X34;
    214. break;
    215. case 1://-1
    216. reg7d0val=0X1C;
    217. reg7d1val=0X2A;
    218. break;
    219. case 3://1
    220. reg7d0val=0X24;
    221. reg7d1val=0X16;
    222. break;
    223. case 4://2
    224. reg7d0val=0X28;
    225. reg7d1val=0X0C;
    226. break;
    227. }
    228. SCCB_WR_Reg(0xff,0x00);
    229. SCCB_WR_Reg(0x7c,0x00);
    230. SCCB_WR_Reg(0x7d,0x04);
    231. SCCB_WR_Reg(0x7c,0x07);
    232. SCCB_WR_Reg(0x7d,0x20);
    233. SCCB_WR_Reg(0x7d,reg7d0val);
    234. SCCB_WR_Reg(0x7d,reg7d1val);
    235. SCCB_WR_Reg(0x7d,0x06);
    236. }
    237. //特效设置
    238. //0:普通模式
    239. //1,负片
    240. //2,黑白
    241. //3,偏红色
    242. //4,偏绿色
    243. //5,偏蓝色
    244. //6,复古
    245. void OV2640_Special_Effects(u8 eft)
    246. {
    247. u8 reg7d0val=0X00;//默认为普通模式
    248. u8 reg7d1val=0X80;
    249. u8 reg7d2val=0X80;
    250. switch(eft)
    251. {
    252. case 1://负片
    253. reg7d0val=0X40;
    254. break;
    255. case 2://黑白
    256. reg7d0val=0X18;
    257. break;
    258. case 3://偏红色
    259. reg7d0val=0X18;
    260. reg7d1val=0X40;
    261. reg7d2val=0XC0;
    262. break;
    263. case 4://偏绿色
    264. reg7d0val=0X18;
    265. reg7d1val=0X40;
    266. reg7d2val=0X40;
    267. break;
    268. case 5://偏蓝色
    269. reg7d0val=0X18;
    270. reg7d1val=0XA0;
    271. reg7d2val=0X40;
    272. break;
    273. case 6://复古
    274. reg7d0val=0X18;
    275. reg7d1val=0X40;
    276. reg7d2val=0XA6;
    277. break;
    278. }
    279. SCCB_WR_Reg(0xff,0x00);
    280. SCCB_WR_Reg(0x7c,0x00);
    281. SCCB_WR_Reg(0x7d,reg7d0val);
    282. SCCB_WR_Reg(0x7c,0x05);
    283. SCCB_WR_Reg(0x7d,reg7d1val);
    284. SCCB_WR_Reg(0x7d,reg7d2val);
    285. }
    286. //彩条测试
    287. //sw:0,关闭彩条
    288. // 1,开启彩条(注意OV2640的彩条是叠加在图像上面的)
    289. void OV2640_Color_Bar(u8 sw)
    290. {
    291. u8 reg;
    292. SCCB_WR_Reg(0XFF,0X01);
    293. reg=SCCB_RD_Reg(0X12);
    294. reg&=~(1<<1);
    295. if(sw)reg|=1<<1;
    296. SCCB_WR_Reg(0X12,reg);
    297. }
    298. //设置图像输出窗口
    299. //sx,sy,起始地址
    300. //width,height:宽度(对应:horizontal)和高度(对应:vertical)
    301. void OV2640_Window_Set(u16 sx,u16 sy,u16 width,u16 height)
    302. {
    303. u16 endx;
    304. u16 endy;
    305. u8 temp;
    306. endx=sx+width/2; //V*2
    307. endy=sy+height/2;
    308. SCCB_WR_Reg(0XFF,0X01);
    309. temp=SCCB_RD_Reg(0X03); //读取Vref之前的值
    310. temp&=0XF0;
    311. temp|=((endy&0X03)<<2)|(sy&0X03);
    312. SCCB_WR_Reg(0X03,temp); //设置Vref的start和end的最低2位
    313. SCCB_WR_Reg(0X19,sy>>2); //设置Vref的start高8位
    314. SCCB_WR_Reg(0X1A,endy>>2); //设置Vref的end的高8位
    315. temp=SCCB_RD_Reg(0X32); //读取Href之前的值
    316. temp&=0XC0;
    317. temp|=((endx&0X07)<<3)|(sx&0X07);
    318. SCCB_WR_Reg(0X32,temp); //设置Href的start和end的最低3位
    319. SCCB_WR_Reg(0X17,sx>>3); //设置Href的start高8位
    320. SCCB_WR_Reg(0X18,endx>>3); //设置Href的end的高8位
    321. }
    322. //设置图像输出大小
    323. //OV2640输出图像的大小(分辨率),完全由改函数确定
    324. //width,height:宽度(对应:horizontal)和高度(对应:vertical),width和height必须是4的倍数
    325. //返回值:0,设置成功
    326. // 其他,设置失败
    327. u8 OV2640_OutSize_Set(u16 width,u16 height)
    328. {
    329. u16 outh;
    330. u16 outw;
    331. u8 temp;
    332. if(width%4)return 1;
    333. if(height%4)return 2;
    334. outw=width/4;
    335. outh=height/4;
    336. SCCB_WR_Reg(0XFF,0X00);
    337. SCCB_WR_Reg(0XE0,0X04);
    338. SCCB_WR_Reg(0X5A,outw&0XFF); //设置OUTW的低八位
    339. SCCB_WR_Reg(0X5B,outh&0XFF); //设置OUTH的低八位
    340. temp=(outw>>8)&0X03;
    341. temp|=(outh>>6)&0X04;
    342. SCCB_WR_Reg(0X5C,temp); //设置OUTH/OUTW的高位
    343. SCCB_WR_Reg(0XE0,0X00);
    344. return 0;
    345. }
    346. //设置图像开窗大小
    347. //由:OV2640_ImageSize_Set确定传感器输出分辨率从大小.
    348. //该函数则在这个范围上面进行开窗,用于OV2640_OutSize_Set的输出
    349. //注意:本函数的宽度和高度,必须大于等于OV2640_OutSize_Set函数的宽度和高度
    350. // OV2640_OutSize_Set设置的宽度和高度,根据本函数设置的宽度和高度,由DSP
    351. // 自动计算缩放比例,输出给外部设备.
    352. //width,height:宽度(对应:horizontal)和高度(对应:vertical),width和height必须是4的倍数
    353. //返回值:0,设置成功
    354. // 其他,设置失败
    355. u8 OV2640_ImageWin_Set(u16 offx,u16 offy,u16 width,u16 height)
    356. {
    357. u16 hsize;
    358. u16 vsize;
    359. u8 temp;
    360. if(width%4)return 1;
    361. if(height%4)return 2;
    362. hsize=width/4;
    363. vsize=height/4;
    364. SCCB_WR_Reg(0XFF,0X00);
    365. SCCB_WR_Reg(0XE0,0X04);
    366. SCCB_WR_Reg(0X51,hsize&0XFF); //设置H_SIZE的低八位
    367. SCCB_WR_Reg(0X52,vsize&0XFF); //设置V_SIZE的低八位
    368. SCCB_WR_Reg(0X53,offx&0XFF); //设置offx的低八位
    369. SCCB_WR_Reg(0X54,offy&0XFF); //设置offy的低八位
    370. temp=(vsize>>1)&0X80;
    371. temp|=(offy>>4)&0X70;
    372. temp|=(hsize>>5)&0X08;
    373. temp|=(offx>>8)&0X07;
    374. SCCB_WR_Reg(0X55,temp); //设置H_SIZE/V_SIZE/OFFX,OFFY的高位
    375. SCCB_WR_Reg(0X57,(hsize>>2)&0X80); //设置H_SIZE/V_SIZE/OFFX,OFFY的高位
    376. SCCB_WR_Reg(0XE0,0X00);
    377. return 0;
    378. }
    379. //该函数设置图像尺寸大小,也就是所选格式的输出分辨率
    380. //UXGA:1600*1200,SVGA:800*600,CIF:352*288
    381. //width,height:图像宽度和图像高度
    382. //返回值:0,设置成功
    383. // 其他,设置失败
    384. u8 OV2640_ImageSize_Set(u16 width,u16 height)
    385. {
    386. u8 temp;
    387. SCCB_WR_Reg(0XFF,0X00);
    388. SCCB_WR_Reg(0XE0,0X04);
    389. SCCB_WR_Reg(0XC0,(width)>>3&0XFF); //设置HSIZE的10:3位
    390. SCCB_WR_Reg(0XC1,(height)>>3&0XFF); //设置VSIZE的10:3位
    391. temp=(width&0X07)<<3;
    392. temp|=height&0X07;
    393. temp|=(width>>4)&0X80;
    394. SCCB_WR_Reg(0X8C,temp);
    395. SCCB_WR_Reg(0XE0,0X00);
    396. return 0;
    397. }
    1. // main.c
    2. #include "sys.h"
    3. #include "delay.h"
    4. #include "usart.h"
    5. #include "led.h"
    6. #include "key.h"
    7. #include "usmart.h"
    8. #include "usart2.h"
    9. #include "timer.h"
    10. #include "ov2640.h"
    11. #include "dcmi.h"
    12. u8 ov2640_mode=0; //工作模式:0,RGB565模式;1,JPEG模式
    13. #define jpeg_buf_size 31*1024 //定义JPEG数据缓存jpeg_buf的大小(*4字节)
    14. __align(4) u32 jpeg_buf[jpeg_buf_size]; //JPEG数据缓存buf
    15. volatile u32 jpeg_data_len=0; //buf中的JPEG有效数据长度
    16. volatile u8 jpeg_data_ok=0; //JPEG数据采集完成标志
    17. //0,数据没有采集完;
    18. //1,数据采集完了,但是还没处理;
    19. //2,数据已经处理完成了,可以开始下一帧接收
    20. //JPEG尺寸支持列表
    21. const u16 jpeg_img_size_tbl[][2]=
    22. {
    23. 176,144, //QCIF
    24. 160,120, //QQVGA
    25. 352,288, //CIF
    26. 320,240, //QVGA
    27. 640,480, //VGA
    28. 800,600, //SVGA
    29. 1024,768, //XGA
    30. 1280,1024, //SXGA
    31. 1600,1200, //UXGA
    32. };
    33. const u8*EFFECTS_TBL[7]={"Normal","Negative","B&W","Redish","Greenish","Bluish","Antique"}; //7种特效
    34. const u8*JPEG_SIZE_TBL[9]={"QCIF","QQVGA","CIF","QVGA","VGA","SVGA","XGA","SXGA","UXGA"}; //JPEG图片 9种尺寸
    35. //处理JPEG数据
    36. //当采集完一帧JPEG数据后,调用此函数,切换JPEG BUF.开始下一帧采集.
    37. void jpeg_data_process(void)
    38. {
    39. if(ov2640_mode)//只有在JPEG格式下,才需要做处理.
    40. {
    41. if(jpeg_data_ok==0) //jpeg数据还未采集完?
    42. {
    43. DMA_Cmd(DMA2_Stream1, DISABLE);//停止当前传输
    44. while (DMA_GetCmdStatus(DMA2_Stream1) != DISABLE){}//等待DMA2_Stream1可配置
    45. jpeg_data_len=jpeg_buf_size-DMA_GetCurrDataCounter(DMA2_Stream1);//得到此次数据传输的长度
    46. jpeg_data_ok=1; //标记JPEG数据采集完按成,等待其他函数处理
    47. }
    48. if(jpeg_data_ok==2) //上一次的jpeg数据已经被处理了
    49. {
    50. DMA2_Stream1->NDTR=jpeg_buf_size;
    51. DMA_SetCurrDataCounter(DMA2_Stream1,jpeg_buf_size);//传输长度为jpeg_buf_size*4字节
    52. DMA_Cmd(DMA2_Stream1, ENABLE); //重新传输
    53. jpeg_data_ok=0; //标记数据未采集
    54. }
    55. }
    56. }
    57. //JPEG测试
    58. //JPEG数据,通过串口2发送给电脑.
    59. void jpeg_test(void)
    60. {
    61. u32 i;
    62. u8 *p;
    63. u8 key;
    64. u8 effect=0,saturation=2,contrast=2;
    65. u8 size=8; //默认是UXGA
    66. u8 msgbuf[15]; //消息缓存区
    67. printf ("OV2640 JPEG Mode");
    68. printf ("\r\n\r\n");
    69. printf("KEY0:Contrast"); //对比度
    70. printf ("\r\n\r\n");
    71. printf("KEY1:Saturation"); //色彩饱和度
    72. printf ("\r\n\r\n");
    73. printf("KEY2:Effects"); //特效
    74. printf ("\r\n\r\n");
    75. printf("KEY_UP:Size"); //分辨率设置
    76. printf ("\r\n\r\n");
    77. sprintf((char*)msgbuf,"JPEG Size:%s",JPEG_SIZE_TBL[size]);
    78. printf ("%d",msgbuf); //显示当前JPEG分辨率
    79. printf ("\r\n\r\n");
    80. OV2640_JPEG_Mode(); //JPEG模式
    81. My_DCMI_Init(); //DCMI配置
    82. DCMI_DMA_Init((u32)&jpeg_buf,jpeg_buf_size,DMA_MemoryDataSize_Word,DMA_MemoryInc_Enable);//DCMI DMA配置
    83. OV2640_OutSize_Set(jpeg_img_size_tbl[size][0],jpeg_img_size_tbl[size][1]);//设置输出尺寸
    84. DCMI_Start(); //启动传输
    85. while(1)
    86. {
    87. if(jpeg_data_ok==1) //已经采集完一帧图像了
    88. {
    89. p=(u8*)jpeg_buf;
    90. printf ("Sending JPEG data..."); //提示正在传输数据
    91. printf ("\r\n\r\n");
    92. for(i=0;i4;i++) //dma传输1次等于4字节,所以乘以4.
    93. {
    94. while(USART_GetFlagStatus(USART2,USART_FLAG_TC)==RESET); //循环发送,直到发送完毕
    95. USART_SendData(USART2,p[i]);
    96. key=KEY_Scan(0);
    97. if(key)break;
    98. }
    99. if(key) //有按键按下,需要处理
    100. {
    101. printf ("Quit Sending data ");//提示退出数据传输
    102. switch(key)
    103. {
    104. case KEY0_PRES: //对比度设置
    105. contrast++;
    106. if(contrast>4)contrast=0;
    107. OV2640_Contrast(contrast);
    108. sprintf((char*)msgbuf,"Contrast:%d",(signed char)contrast-2);
    109. printf ("\r\n\r\n");
    110. break;
    111. case KEY1_PRES: //饱和度Saturation
    112. saturation++;
    113. if(saturation>4)saturation=0;
    114. OV2640_Color_Saturation(saturation);
    115. sprintf((char*)msgbuf,"Saturation:%d",(signed char)saturation-2);
    116. printf ("\r\n\r\n");
    117. break;
    118. case KEY2_PRES: //特效设置
    119. effect++;
    120. if(effect>6)effect=0;
    121. OV2640_Special_Effects(effect);//设置特效
    122. sprintf((char*)msgbuf,"%s",EFFECTS_TBL[effect]);
    123. printf ("\r\n\r\n");
    124. break;
    125. case WKUP_PRES: //JPEG输出尺寸设置
    126. size++;
    127. if(size>8)size=0;
    128. OV2640_OutSize_Set(jpeg_img_size_tbl[size][0],jpeg_img_size_tbl[size][1]);//设置输出尺寸
    129. sprintf((char*)msgbuf,"JPEG Size:%s",JPEG_SIZE_TBL[size]);
    130. printf ("\r\n\r\n");
    131. break;
    132. }
    133. printf ("%d",msgbuf);//显示提示内容
    134. printf ("\r\n\r\n");
    135. delay_ms(800);
    136. }else printf ("Send data complete!!");//提示传输结束设置
    137. jpeg_data_ok=2; //标记jpeg数据处理完了,可以让DMA去采集下一帧了.
    138. }
    139. }
    140. }
    141. int main(void)
    142. {
    143. u8 key;
    144. u8 t;
    145. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
    146. delay_init(168); //初始化延时函数
    147. uart_init(115200); //初始化串口波特率为115200
    148. usart2_init(42,115200); //初始化串口2波特率为115200
    149. LED_Init(); //初始化LED
    150. KEY_Init(); //按键初始化
    151. TIM3_Int_Init(10000-1,8400-1);//10Khz计数,1秒钟中断一次
    152. usmart_dev.init(84); //初始化USMART
    153. while(OV2640_Init())//初始化OV2640
    154. {
    155. printf ("OV2640 ERR");
    156. printf ("\r\n\r\n");
    157. delay_ms(200);
    158. }
    159. printf("OV2640 OK");
    160. printf ("\r\n\r\n");
    161. printf ("KEY0:RGB565 KEY1:JPEG");
    162. printf ("\r\n\r\n");
    163. while(1)
    164. {
    165. key=KEY_Scan(0);
    166. if(key==KEY0_PRES) //RGB565模式
    167. {
    168. ov2640_mode=0;
    169. break;
    170. }else if(key==KEY1_PRES) //JPEG模式
    171. {
    172. ov2640_mode=1;
    173. break;
    174. }
    175. t++;
    176. delay_ms(5);
    177. }
    178. if(ov2640_mode)jpeg_test();
    179. // else rgb565_test();
    180. }

    总结 

            OV2640摄像头的知识点比较的多,而且目前只有显示屏,所以不能在电脑端显示图像,后期串口线到了之后再补上。 

  • 相关阅读:
    flutter(学习日记篇-1)
    前端学习笔记 第1集:WebStorm 安装、vue3.0 安装、npm安装
    自然语言处理的多行业应用
    day54 django中orm数据库增删改查
    【SQL】mysql创建定时任务执行存储过程--20230928
    5年经验之谈 —— 性能测试如何定位分析性能瓶颈?
    Linux进程通信——IPC、管道、FIFO的引入
    Docker简易部署RabbitMQ集群、分布式事务解决方案+案例(可靠生产、可靠消费)
    Python与数据分析--每天绘制Matplotlib库实例图片3张-第1天
    小红书如何打造爆款内容?
  • 原文地址:https://blog.csdn.net/weixin_66578482/article/details/126805878