1)实验平台:正点原子stm32f103战舰开发板V4
2)平台购买地址:https://detail.tmall.com/item.htm?id=609294757420
3)全套实验源码+手册+视频下载地址: http://www.openedv.com/thread-340252-1-1.html##
本章将介绍使用APM32F407驱动OV2640摄像头,从而获取摄像头输出的图像数据,并将其显示在LCD上,也可通过串口发送至PC上位机软件。通过本章的学习,读者将学习到数字摄像头接口(DCI)的使用。
本章分为如下几个小节:
46.1 硬件设计
46.2 程序设计
46.3 下载验证
46.1 硬件设计
46.1.1 例程功能
图46.1.3.1 摄像头模块与MCU的连接原理图
46.2 程序设计
46.2.1 Geehy标准库的DCI驱动
本章使用通过DCI驱动摄像头模块,并获取摄像头模块返回的图像数据,分为RGB565模式和JPEG模式,其中RGB565模式直接将摄像头返回的数据流通过DMA传输至LCD,JPEG模式下将整帧的JPEG图像数据通过USART2传输至PC上位机软件,因为需要获取JPEG整帧的数据,因此JPEG模式下需要使用到DCI的捕获完成中断,其具体的步骤如下:
①:配置DCI
②:对于JPEG模式,需要使能DCI的捕获完成中断
③:对于JPEG模式,需要使能DCI中断,并配置其相关的中断优先级
④:使能DCI
⑤:使能DCI捕获
⑥:对于JPEG模式,需要读取DCI的捕获完成中断标志
⑦:对于JPEG模式,需要清除DCI的捕获完成中断标志
在Geehy标准库中对应的驱动函数如下:
①:配置DCI
该函数用于配置DCI的各项参数,其函数原型如下所示:
void DCI_Config(DCI_Config_T* dciConfig);
该函数的形参描述,如下表所示:
形参 描述
dciConfig 指向DCI配置结构体的指针
需自行定义,并根据DCI的配置参数填充结构体中的成员变量
表46.2.1.1 函数DCI_Config()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表46.2.1.2 函数DCI_Config()返回值描述
该函数使用DCI_Config_T类型的结构体变量传入DCI的配置参数,该结构体的定义如下所示:
typedef enum
{
DCI_CAPTURE_MODE_CONTINUOUS, /* 连续采集模式 */
DCI_CAPTURE_MODE_SNAPSHOT /* 快照模式 */
} DCI_CAPTURE_MODE_T;
typedef enum
{
DCI_SYNCHRO_MODE_HARDWARE, /* 硬件同步 */
DCI_SYNCHRO_MODE_EMBEDDED /* 内嵌码同步 */
} DCI_SYNCHRO_MODEVAL_T;
typedef enum
{
DCI_PCK_POL_FALLING, /* 上升沿捕获 */
DCI_PCK_POL_RISING /* 下降沿捕获 */
} DCI_PCK_POL_T;
typedef enum
{
DCI_VSYNC_POL_LOW, /* 低电平有效 */
DCI_VSYNC_POL_HIGH /* 高电平有效 */
} DCI_VSYNC_POL_T;
typedef enum
{
DCI_HSYNC_POL_LOW, /* 低电平有效 */
DCI_HSYNC_POL_HIGH /* 高电平有效 */
} DCI_HSYNC_POL_T;
typedef enum
{
DCI_CAPTURE_RATE_ALL_FRAME, /* 捕获所有帧 */
DCI_CAPTURE_RATE_1OF2_FRAME, /* 隔一个帧捕获一次 */
DCI_CAPTURE_RATE_1OF4_FRAME /* 隔三个帧捕获一次 */
} DCI_CAPTURE_RATE_T;
typedef enum
{
DCI_EXTENDED_DATA_MODE_8B, /* 8位数据宽度 */
DCI_EXTENDED_DATA_MODE_10B, /* 10位数据宽度 */
DCI_EXTENDED_DATA_MODE_12B, /* 12位数据宽度 */
DCI_EXTENDED_DATA_MODE_14B /* 14位数据宽度 */
} DCI_EXTENDED_DATA_MODE_T;
typedef struct
{
DCI_CAPTURE_MODE_T captureMode; /* 捕获模式 */
DCI_SYNCHRO_MODEVAL_T synchroMode; /* 同步模式 */
DCI_PCK_POL_T pckPolarity; /* 像素时钟极性 */
DCI_VSYNC_POL_T vsyncPolarity; /* 垂直同步极性 */
DCI_HSYNC_POL_T hsyncPolarity; /* 水平同步极性 */
DCI_CAPTURE_RATE_T capturerate; /* 帧捕获频率 */
DCI_EXTENDED_DATA_MODE_T extendedDataMode; /* 数据宽度 */
} DCI_Config_T;
该函数的使用示例,如下所示:
#include "apm32f4xx.h"
#include "apm32f4xx_dci.h"
void example_fun(void)
{
DCI_Config_T dci_init_struct;
/* 配置DCI */
dci_init_struct.captureMode = DCI_CAPTURE_MODE_CONTINUOUS;
dci_init_struct.synchroMode = DCI_SYNCHRO_MODE_HARDWARE;
dci_init_struct.pckPolarity = DCI_PCK_POL_RISING;
dci_init_struct.vsyncPolarity = DCI_VSYNC_POL_LOW;
dci_init_struct.hsyncPolarity = DCI_HSYNC_POL_LOW;
dci_init_struct.capturerate = DCI_CAPTURE_RATE_ALL_FRAME;
dci_init_struct.extendedDataMode = DCI_EXTENDED_DATA_MODE_8B;
DCI_Config(&dci_init_struct);
}
②:使能DCI的指定中断
该函数用于使能DCI的指定中断,其函数原型如下所示:
void DCI_EnableInterrupt(uint32_t interrupt);
该函数的形参描述,如下表所示:
形参 描述
interrupt 指定DCI的中断
例如:DCI_INT_CC、DCI_INT_OVR等(在apm32f4xx_dci.h文件中有定义)
表46.2.1.3 函数DCI_EnableInterrupt()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表46.2.1.4 函数DCI_EnableInterrupt()返回值描述
该函数的使用示例,如下所示:
#include "apm32f4xx.h"
#include "apm32f4xx_dci.h"
void example_fun(void)
{
/* 使能DCI捕获完成中断 */
DCI_EnableInterrupt(DCI_INT_CC);
}
③:配置DCI中断
请见第12.2.3小节中配置中断的相关内容。
④:使能DCI
该函数用于使能DCI,其函数原型如下所示:
void DCI_Enable(void);
该函数的形参描述,如下表所示:
形参 描述
无 无
表46.2.1.5 函数DCI_Enable()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表46.2.1.6 函数DCI_Enable()返回值描述
该函数的使用示例,如下所示:
#include "apm32f4xx.h"
#include "apm32f4xx_dci.h"
void example_fun(void)
{
/* 使能DCI */
DCI_Enable();
}
⑤:使能DCI捕获
该函数用于使能DCI捕获,其函数原型如下所示:
void DCI_EnableCapture(void);
该函数的形参描述,如下表所示:
形参 描述
无 无
表46.2.1.7 函数DCI_EnableCapture()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表46.2.1.8 函数DCI_EnableCapture()返回值描述
该函数的使用示例,如下所示:
#include "apm32f4xx.h"
#include "apm32f4xx_dci.h"
void example_fun(void)
{
/* 使能DCI捕获 */
DCI_EnableCapture();
}
⑥:读取DCI指定中断标志
该函数用于读取DCI的指定中断标志,其函数原型如下所示:
uint16_t DCI_ReadIntFlag(DCI_INT_T flag);
该函数的形参描述,如下表所示:
形参 描述
flag 指定DCI的中断标志
例如:DCI_INT_CC、DCI_INT_OVR等(在apm32f4xx_dci.h文件中有定义)
表46.2.1.9 函数DCI_ReadIntFlag()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
SET 中断标志发生
RESET 中断标志未发生
表46.2.1.10 函数DCI_ReadIntFlag()返回值描述
该函数的使用示例,如下所示:
#include "apm32f4xx.h"
#include "apm32f4xx_dci.h"
void example_fun(void)
{
uint8_t flag;
/* 获取DCI的捕获完成中断标志 */
flag = DCI_ReadIntFlag(DCI_INT_CC);
if (flag == SET)
{
/* Do something. */
}
else
{
/* Do something. */
}
}
⑦:清除DCI指定中断标志
该函数用于清除DCI的指定中断标志,其函数原型如下所示:
void DCI_ClearIntFlag(uint16_t flag);
该函数的形参描述,如下表所示:
形参 描述
flag 指定DCI的中断标志
例如:DCI_INT_CC、DCI_INT_OVR等(在apm32f4xx_dci.h文件中有定义)
表46.2.1.11 函数DCI_CleatIntFlag()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表46.2.1.12 函数DCI_ClearIntFlag()返回值描述
该函数的使用示例,如下所示:
#include "apm32f4xx.h"
#include "apm32f4xx_dci.h"
void example_fun(void)
{
/* 清除DCI的捕获完成中断标志 */
DCI_ClearIntFlag(DCI_INT_CC);
}
46.2.2 DCI驱动代码
本章实验的DCI驱动主要负责向应用层提供DCI的初始化和启动DCI传输等函数。本章实验中,DCI的驱动代码包括dci.c和dci.h两个文件。
由于DCI使用了大量的GPIO引脚,因此对于GPIO的相关定义,请读者自行查看dci.c这个文件。
DCI驱动中,DCI的初始化函数,如下所示:
/**
* @brief 初始化DCI
* @note 引脚对应关系如下:
* 摄像头模块 ------------ APM32F407最小系统板
* OV_D0~D7 ------------ PC6/PC7/PC8/PC9/PC11/PB6/PE5/PE6
* OV_SCL ------------ PD6
* OV_SDA ------------ PD7
* OV_VSYNC ------------ PB7
* OV_HREF ------------ PA4
* OV_PCLK ------------ PA6
* OV_PWDN ------------ PG9
* OV_RESET ------------ PG15
* OV_XCLK ------------ PA8
* 本函数仅初始化与DCI接口相连接的11个引脚
* (OV_D0~D7/OV_VSYNC/OV_HREF/OV_PCLK)
* @param 无
* @retval 无
*/
void dci_init(void)
{
GPIO_Config_T gpio_init_struct;
DCI_Config_T dci_init_struct;
/* 使能时钟 */
RCM_EnableAHB2PeriphClock(RCM_AHB2_PERIPH_DCI); /* 使能DCI时钟 */
/* 配置DCI使用到的GPIO引脚,代码省略 */
/* 配置DCI */
DCI_Rest();
dci_init_struct.captureMode = DCI_CAPTURE_MODE_CONTINUOUS;
dci_init_struct.synchroMode = DCI_SYNCHRO_MODE_HARDWARE;
dci_init_struct.pckPolarity = DCI_PCK_POL_RISING;
dci_init_struct.vsyncPolarity = DCI_VSYNC_POL_LOW;
dci_init_struct.hsyncPolarity = DCI_HSYNC_POL_LOW;
dci_init_struct.capturerate = DCI_CAPTURE_RATE_ALL_FRAME;
dci_init_struct.extendedDataMode = DCI_EXTENDED_DATA_MODE_8B;
DCI_Config(&dci_init_struct);
/* 使能DCI及其相关中断 */
DCI_EnableInterrupt(DCI_INT_CC); /* 使能捕获完成中断 */
NVIC_EnableIRQRequest(DCI_IRQn, 2, 0); /* 使能DCI中断 */
DCI_Enable(); /* 使能DCI */
}
在DCI的初始化函数中可以看到,配置并使能了DCI,还为JPEG模式使能了DCI的捕获完成中断。
DCI驱动中,DCI的中断回调函数,如下所示:
/**
* @brief DCI中断服务函数
* @param 无
* @retval 无
*/
void DCI_IRQHandler(void)
{
if (DCI_ReadIntFlag(DCI_INT_CC) == SET) /* 判断捕获完成中断标志 */
{
jpeg_data_process(); /* 处理JPEG数据 */
LED1_TOGGLE(); /* LED1闪烁 */
g_ov_frame++; /* 更新帧数 */
DCI_ClearIntFlag(DCI_INT_CC); /* 清除捕获完成中断标志 */
}
}
可以看到,该函数主要就是调用了JPEG数据的处理函数,以及统计帧率。
摄像头返回的数据是使用DMA进行传输的,DCI驱动中,DMA的初始化函数,如下所示:
/**
* @brief 配置DCI DMA
* @note DCI使用DMA2 数据流1 通道1
* @param mem0addr: 存储器0地址
* @param mem1addr: 存储器1地址,0:不使用双缓冲模式
* @param memsize : 传输的数据项数目,范围:0~65535
* @param memblen : 存储器数据大小,见DMA_MEMORY_DATA_SIZE_T
* @param meminc : 存储器增量模式,见DMA_MEMORY_INC_T
* @retval 无
*/
void dci_dma_init( uint32_t mem0addr,
uint32_t mem1addr,
uint16_t memsize,
uint32_t memblen,
uint32_t meminc)
{
DMA_Config_T dma_init_struct;
/* 使能时钟 */
RCM_EnableAHB1PeriphClock(RCM_AHB1_PERIPH_DMA2); /* 使能DMA2时钟 */
/* 复位DMA */
DMA_DisableInterrupt(DMA2_Stream1, DMA_INT_TCIFLG);
DMA_Reset(DMA2_Stream1); /* 复位DMA */
while (DMA_ReadCmdStatus(DMA2_Stream1) != DISABLE); /* 等待DMA可配置 */
/* 配置DMA */
dma_init_struct.channel = DMA_CHANNEL_1;
dma_init_struct.peripheralBaseAddr = (uint32_t)&(DCI->DATA);
dma_init_struct.memoryBaseAddr = mem0addr;
dma_init_struct.dir = DMA_DIR_PERIPHERALTOMEMORY;
dma_init_struct.bufferSize = (uint32_t)memsize;
dma_init_struct.peripheralInc = DMA_PERIPHERAL_INC_DISABLE;
dma_init_struct.memoryInc = (DMA_MEMORY_INC_T)meminc;
dma_init_struct.peripheralDataSize = DMA_PERIPHERAL_DATA_SIZE_WORD;
dma_init_struct.memoryDataSize = (DMA_MEMORY_DATA_SIZE_T)memblen;
dma_init_struct.loopMode = DMA_MODE_CIRCULAR;
dma_init_struct.priority = DMA_PRIORITY_HIGH;
dma_init_struct.fifoMode = DMA_FIFOMODE_ENABLE;
dma_init_struct.fifoThreshold = DMA_FIFOTHRESHOLD_HALFFULL;
dma_init_struct.memoryBurst = DMA_MEMORYBURST_SINGLE;
dma_init_struct.peripheralBurst = DMA_PERIPHERALBURST_SINGLE;
DMA_Config(DMA2_Stream1, &dma_init_struct);
if (mem1addr == 0) /* 不使用双缓冲区模式 */
{
/* 禁用双缓冲区模式 */
DMA_DisableDoubleBufferMode(DMA2_Stream1);
NVIC_DisableIRQRequest(DMA2_STR1_IRQn);
}
else /* 使用双缓冲区模式 */
{
/* 使能双缓冲区模式 */
DMA_EnableDoubleBufferMode(DMA2_Stream1);
/* 配置存储器1地址 */
DMA_ConfigMemoryTarget(DMA2_Stream1, mem1addr, DMA_MEMORY_1);
/* 使能DMA传输完成中断 */
DMA_EnableInterrupt(DMA2_Stream1, DMA_INT_TCIFLG);
/* 使能DMA中断 */
NVIC_EnableIRQRequest(DMA2_STR1_IRQn, 2, 0);
}
}
可以看到,该函数配置DMA将DCI的数据传输至存储器,且可以配置单缓冲和双缓冲。
DCI驱动中,启动DCI传输的函数,如下所示:
/**
* @brief 启动DCI传输
* @note 无
* @retval 无
*/
void dci_start(void)
{
lcd_set_cursor(0, 0); /* 设置LCD光标到原点 */
lcd_write_ram_prepare(); /* 准备写GRAM */
DMA_Enable(DMA2_Stream1); /* 使能DMA */
DCI_EnableCapture(); /* 使能DCI捕获 */
}
因为在RGB565模式下,DMA将直接将DCI的数据传输至LCD上进行显示,因此在DCI的启动传输函数中对LCD进行了相应的操作,并使能了DMA和DCI。
DCI的驱动就介绍这么多,更多的请读者查看本章实验配套实验例程的源码。
46.2.3 SCCB驱动
本章实验的SCCB驱动主要负责向OV2640驱动提供配置OV2640摄像头的各种函数,SCCB协议与IIC协议十分相似,也可兼容IIC协议,因此请读者结合SCCB和IIC协议的相关文档查看本章实验配套实验源码中SCCB的相关驱动文件。本章实验中,SCCB的驱动代码包括sccb.c和sccb.h两个文件。
46.2.4 OV2640驱动
本章实验的OV2640驱动主要负责向应用层提供OV2640的初始化和各种配置函数,请读者结合正点原子OV2640模块(ATK-MC2640)的用户手册查看本章实验配套例程源码中的OV2640驱动代码。本章实验中,OV2640的驱动代码包括ov2640.c和ov2640.h两个文件。
46.2.5 实验应用代码
本章实验的应用代码,如下所示:
int main(void)
{
uint8_t t = 0;
uint8_t key;
NVIC_ConfigPriorityGroup(NVIC_PRIORITY_GROUP_3); /* 设置中断优先级分组为组3 */
sys_apm32_clock_init(336, 8, 2, 7); /* 配置系统时钟 */
delay_init(168); /* 初始化延时功能 */
usart_init(115200); /* 初始化串口 */
usmart_dev.init(84); /* 初始化USMART */
led_init(); /* 初始化LED */
key_init(); /* 初始化按键 */
lcd_init(); /* 初始化LCD */
usart2_init(921600); /* 初始化USART2 */
btmr_tmrx_int_init(10000 - 1, 8400 - 1);/* 初始化基本定时器中断,中断频率1Hz */
lcd_show_string(30, 50, 200, 16, 16, "APM32", RED);
lcd_show_string(30, 70, 200, 16, 16, "OV2640 TEST", RED);
lcd_show_string(30, 90, 200, 16, 16, "ATOM@ALIENTEK", RED);
while (ov2640_init() != 0) /* 初始化OV2640 */
{
lcd_show_string(30, 130, 240, 16, 16, "OV2640 ERROR", RED);
delay_ms(200);
lcd_fill(30, 130, 239, 170, WHITE);
delay_ms(200);
LED0_TOGGLE();
}
lcd_show_string(30, 130, 200, 16, 16, "OV2640 OK", RED);
ov2640_flash_intctrl(); /* 内部控制闪光灯 */
while (1)
{
t++;
key = key_scan(0);
if (key == KEY0_PRES) /* RGB565模式 */
{
g_ov_mode = 0;
break;
}
else if (key == WKUP_PRES) /* JPEG模式 */
{
g_ov_mode = 1;
break;
}
if (t == 100)
{
lcd_show_string( 30,
150,
230,
16,
16,
"KEY0:RGB565 KEY_UP:JPEG",
RED);
}
else if (t == 200)
{
t = 0;
lcd_fill(30, 150, 230, 150 + 16, WHITE);
LED0_TOGGLE();
}
delay_ms(5);
}
if (g_ov_mode == 0)
{
rgb565_test(); /* RGB565模式测试 */
}
else
{
jpeg_test(); /* JPEG模式测试 */
}
}
从上面的代码中可以看出,在初始化完OV2640摄像头后,并不断地扫描按键,若检测到KEY0按键被按下,则进入RGB565模式的测试,即将DCI获取到的摄像头图像数据,直接在LCD上进行显示,若检测到KEY_UP按键被按下,则进入JPEG模式的测试,即将DCI获取到的摄像头数据,通过USART2传输至PC上位机。
46.3 下载验证
在完成编译和烧录操作后,先将开发板断电,随后将摄像头模块通过摄像头延长线与开发板进行连接,同时连接好LCD模式和USART2与PC的连接,最后再给开发板供电(在插拔开发板上的接插件和模块时,要求必须断电操作,否则容易烧毁硬件)。程序运行后,可以看到LCD上提示按下KEY0按键进入RGB565模式的测试,按下KEY_UP按键进入JPEG模式的测试。
若此时按下KEY0按键,若开发板连接了LCD模块,便能够在LCD显示屏上看到摄像头采集到的实时画面,并且按下KEY0按键可以改变画面的对比度,按下KEY_UP按键可以改变画面的尺寸。
若此时按下KEY_UP按键,若开发板的USART2通过USB转串口模式与PC进行连接,并配置好PC端的上位机(正点原子 视频传输上位机,读者可前往正点原子资料下载中心下载该上位机软件),便可以在上位机软件中看到摄像头采集到的实时画面,并且按下KEY0按键可以改变画面的对比度,按下KEY_UP按键可以改变画面的尺寸。
在进入RGB565模式或JPEG模式测试后,可以通过串口调试助手查看到摄像头的实时帧率。