STM32输入捕获模式设置并用DMA接收数据

本文详细介绍了STM32微控制器中红外学习模块的实现过程,包括输入捕获模式的设置及使用DMA接收数据的方法,旨在优化CPU资源利用并提高数据处理效率。

STM32输入捕获模式设置并用DMA接收数据


本文博客链接:http://blog.youkuaiyun.com/jdh99,作者:jdh,转载请注明.


环境:

主机:WIN7

开发环境:MDK4.72

MCU:STM32F103


说明:

项目中需要进行红外学习,于是采用输入捕获取得电平变化时间.并将数据放在DMA中.这样可以避免频繁中断消耗CPU资源.

采用的是PB1脚,对应TIM3的通道4.


源代码:

/*********************************************************************
*							接口函数:初始化红外学习模块
**********************************************************************/

void inf_infrared_study_init(void)
{
    //初始化io口
  	inf_init_io();
    //初始化中断
    //inf_init_irq();
    //初始化定时器
    inf_init_timer();
    
    //打开DMA
    inf_infrared_study_open_dma(1);
    //打开定时器
    inf_infrared_study_open_timer(1);
}

/*********************************************************************
*							初始化io口
**********************************************************************/

static void inf_init_io(void)
{
	//定义IO初始化结构体
	GPIO_InitTypeDef GPIO_InitStructure;

	//初始化时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	//管脚初始化  
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
	//设置为输入           
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; 
	//初始化    			
    GPIO_Init(GPIOB, &GPIO_InitStructure);     
}

/*********************************************************************
*							初始化中断
**********************************************************************/

static void inf_init_irq(void)
{
	//定义外部中断结构体
	EXTI_InitTypeDef EXTI_InitStructure;

	//初始化中断脚复用时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
	//配置中断源
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource1);
	// 配置下降沿触发
	EXTI_ClearITPendingBit(EXTI_Line1);
	EXTI_InitStructure.EXTI_Line = EXTI_Line1;
	EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
	EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
	EXTI_InitStructure.EXTI_LineCmd = ENABLE;
	EXTI_Init(&EXTI_InitStructure);
}

/*********************************************************************
*							初始化定时器
**********************************************************************/

static void inf_init_timer(void)
{
	//定义定时器结构体
	TIM_TimeBaseInitTypeDef timInitStruct;
    //输入捕获结构体
    TIM_ICInitTypeDef tim_icinit;
	//定义DMA结构体
	DMA_InitTypeDef DMA_InitStructure;
	 
	//启动DMA时钟
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	//DMA1通道3配置
	DMA_DeInit(DMA1_Channel3);
	//外设地址
	DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&TIM3->CCR4);
	//内存地址
	DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)Rx_Buf_Tim_Dma;
	//dma传输方向单向
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
	//设置DMA在传输时缓冲区的长度
	DMA_InitStructure.DMA_BufferSize = RX_LEN_TIM_DMA;
	//设置DMA的外设递增模式,一个外设
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
	//设置DMA的内存递增模式
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
	//外设数据字长
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
	//内存数据字长
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
	//设置DMA的传输模式
	//DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
    DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
	//设置DMA的优先级别
	DMA_InitStructure.DMA_Priority = DMA_Priority_High;
	//设置DMA的2个memory中的变量互相访问
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
	DMA_Init(DMA1_Channel3,&DMA_InitStructure);              

	//开启时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
	//重新将Timer设置为缺省值
	TIM_DeInit(TIM3);
	//采用内部时钟给TIM3提供时钟源
	TIM_InternalClockConfig(TIM3);
	//预分频
    timInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;    
    //计数频率为500ns跳转1次 			
	timInitStruct.TIM_Prescaler = SystemCoreClock / 2000000 - 1;     
	//向上计数  				
    timInitStruct.TIM_CounterMode = TIM_CounterMode_Up; 	
    timInitStruct.TIM_RepetitionCounter = 0;
	//这个值实际上就是TIMX->ARR,延时开始时重新设定即可	
    timInitStruct.TIM_Period = 0xffff; 								
	//初始化定时器3
	TIM_TimeBaseInit(TIM3, &timInitStruct);
    
    //输入捕获配置
    //选择通道
    tim_icinit.TIM_Channel = TIM_Channel_4;
    //硬件滤波
	tim_icinit.TIM_ICFilter = 0x0;
    //触发捕获的电平
    tim_icinit.TIM_ICPolarity = TIM_ICPolarity_Falling;
    //每次检测到触发电平都捕获
	tim_icinit.TIM_ICPrescaler= TIM_ICPSC_DIV1;
    //通道方向选择
	tim_icinit.TIM_ICSelection = TIM_ICSelection_DirectTI;
    //初始化
    TIM_ICInit(TIM3,&tim_icinit);
    
	//禁止ARR预装载缓冲器  
	TIM_ARRPreloadConfig(TIM3, DISABLE);  

    //输入跳变选择
    TIM_SelectInputTrigger(TIM3, TIM_TS_TI2FP2);
    //从机模式:复位模式
    TIM_SelectSlaveMode(TIM3, TIM_SlaveMode_Reset);
    //主从模式选择
    TIM_SelectMasterSlaveMode(TIM3, TIM_MasterSlaveMode_Enable);
 
    //配置定时器的DMA
    TIM_DMAConfig(TIM3,TIM_DMABase_CCR4,TIM_DMABurstLength_2Bytes);
    //产生DMA请求信号
    TIM_DMACmd(TIM3, TIM_DMA_CC4, ENABLE);
    //打开定时器
    TIM_Cmd(TIM3, ENABLE);
}

/*********************************************************************
*							接口函数:打开定时器
*参数:state:状态:0:关闭,1:打开
**********************************************************************/

void inf_infrared_study_open_timer(uint8_t state)
{
    if (state)
    {
        TIM_Cmd(TIM3, ENABLE);
    }
    else
    {
        TIM_Cmd(TIM3, DISABLE);
    }
}

/*********************************************************************
*							接口函数:打开中断
*参数:state:状态:0:关闭,1:打开
**********************************************************************/

void inf_infrared_study_open_irq(uint8_t state)
{
    //定义中断结构体
	NVIC_InitTypeDef NVIC_InitStructure ;
    
    if (state)
    {
        //打开中断
        NVIC_InitStructure.NVIC_IRQChannel = EXTI1_IRQn;				//通道设置为外部中断线
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; 		//中断抢占先等级
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;           	//中断响应优先级
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;           		//打开中断
        NVIC_Init(&NVIC_InitStructure);                                 //初始化
    }
    else
    {
        //关闭中断
        NVIC_InitStructure.NVIC_IRQChannel = EXTI1_IRQn;				//通道设置为外部中断线
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; 		//中断抢占先等级
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;           	//中断响应优先级
        NVIC_InitStructure.NVIC_IRQChannelCmd = DISABLE;           		//打开中断
        NVIC_Init(&NVIC_InitStructure);                                 //初始化
    }
}

/*********************************************************************
*							接口函数:打开DMA
*参数:state:状态:0:关闭,1:打开
**********************************************************************/

void inf_infrared_study_open_dma(uint8_t state)
{
    if (state)
    {
        //设置传输数据长度
		//DMA_SetCurrDataCounter(DMA1_Channel3,RX_LEN_TIM_DMA);
    	//打开DMA
		DMA_Cmd(DMA1_Channel3,ENABLE);
    }
    else
    {
        DMA_Cmd(DMA1_Channel3,DISABLE);
    }
}

/*********************************************************************
*							接口函数:得到DMA接收帧长
*返回:帧长
**********************************************************************/

uint16_t inf_infrared_study_dma_rx_len(void)
{
    //获得接收帧帧长
	return (RX_LEN_TIM_DMA - DMA_GetCurrDataCounter(DMA1_Channel3));
}

注意:

除TIM6和TIM7之外的定时器都只能采用上升沿或者下降沿捕捉而不能采用双边沿捕捉.

#define  TIM_ICPolarity_Rising             ((uint16_t)0x0000)
#define  TIM_ICPolarity_Falling            ((uint16_t)0x0002)
#define  TIM_ICPolarity_BothEdge           ((uint16_t)0x000A)
#define IS_TIM_IC_POLARITY(POLARITY) (((POLARITY) == TIM_ICPolarity_Rising) || \
                                      ((POLARITY) == TIM_ICPolarity_Falling))
#define IS_TIM_IC_POLARITY_LITE(POLARITY) (((POLARITY) == TIM_ICPolarity_Rising) || \
                                           ((POLARITY) == TIM_ICPolarity_Falling)|| \
                                           ((POLARITY) == TIM_ICPolarity_BothEdge))  




<think>好的,用户的问题是关于STM32进行字符识别的实现方案、库函数和示例代码。首先,我需要回顾之前用户提到的引用内容,看看是否有相关的内容可以借鉴。 用户提供的引用中有提到车牌识别教程,使用STM32连接摄像头,处理图像,并用Python算法识别车牌区域[^2]。这可能涉及到图像采集和处理的步骤。另外,用户之前的问题涉及到智能小车的设计,包括硬件连接和库函数开发[^1][^4],还有串口通信和命令解析的示例代码[^3]。这些可能对字符识别的实现有帮助。 首先,我需要明确STM32进行字符识别的常见方法。通常,STM32由于资源有限,不适合运行复杂的OCR算法,所以可能需要结合外部模块或简化算法。可能的方案包括: 1. **使用专用字符识别模块**:比如OCR模块,通过串口或I2C与STM32通信,STM32只需发送图像数据并接收识别结果。 2. **简化图像处理算法**:在STM32上实现二值化、边缘检测等预处理,然后使用模板匹配或简单的机器学习模型(如TinyML)进行识别。 3. **结合上位机处理**:STM32负责采集图像并传输到PC或手机,由上位机运行复杂算法,再将结果返回。这在引用2中提到过车牌识别部分使用Python处理。 接下来需要考虑硬件部分,用户可能需要的硬件包括摄像头模块(如OV7670)、显示屏(如OLED)、存储模块(如SD卡)以及足够的RAM和Flash。例如,引用4中提到的智能小车使用了OLED和摄像头模块[^4],这可能相关。 关于库函数,STM32的标准外设库或HAL库可能需要用于配置摄像头接口、SPI/I2C通信、DMA传输等。例如,引用3中的串口DMA接收和发送函数[^3],可以用于数据传输。 示例代码方面,可能需要初始化摄像头,采集图像数据,进行预处理,然后调用识别函数。例如,使用DCMI接口和DMA捕获图像,接着进行二值化处理,最后用模板匹配识别字符。这部分可能需要参考现有的图像处理库或自行实现简单算法。 同时,用户可能需要优化资源,因为STM32的内存有限,处理高分辨率图像会有困难。可能需要降低图像分辨率或采用分块处理的方法。引用2中的车牌识别教程可能涉及类似的处理步骤,但具体实现可能需要调整[^2]。 最后,生成相关问题需要考虑用户可能的后续疑问,比如性能优化、算法选择、硬件选型等。例如,如何提高识别速度,如何选择适合的摄像头模块,或者如何集成蓝牙模块进行数据传输等,这些在之前的引用中都有涉及[^4][^3]。</think>### STM32字符识别实现方案 #### 一、实现方案选择 1. **轻量级OCR算法本地运行** - 适用场景:固定字体/简单字符(如数字、字母) - 实现步骤: 1. 图像采集(OV7670摄像头) 2. 预处理(二值化+降噪) 3. 特征提取(投影直方图/轮廓分析) 4. 模板匹配识别 2. **云端协同方案**(推荐资源受限场景) - STM32通过WiFi/蓝牙发送图像数据 - 云端服务器执行OCR处理 - 结果回传至STM32显示[^2][^4] #### 二、硬件配置参考 | 模块类型 | 推荐型号 | 接口方式 | |----------------|-------------------|------------| | 图像采集 | OV7670带FIFO | DCMI | | 显示模块 | 0.96寸OLED | SPI/I2C | | 无线传输 | ESP8266/蓝牙HC-05 | UART | | 存储扩展 | W25Q128 Flash | SPI | #### 三、关键代码实现 ```c // 图像采集示例(使用DCMI+DMA) void Capture_Image(uint8_t *buf) { DCMI_Start(); // 启动摄像头采集 HAL_DCMI_Start_DMA(&hdcmi, DCMI_MODE_CONTINUOUS, (uint32_t)buf, IMAGE_SIZE); while(!capture_done); // 等待DMA传输完成 } // 字符识别核心算法伪代码 char OCR_Process(uint8_t *img) { // 1. 二值化处理 Binarize(img, OTSU_THRESHOLD); // 2. 字符分割 Segment_Characters(img, &regions); // 3. 特征提取 for(int i=0; i<regions.count; i++){ Extract_Feature(regions[i], &features); } // 4. 模板匹配 return Match_Template(features); } // 主处理流程 int main() { uint8_t image_buffer[320*240]; Capture_Image(image_buffer); char result = OCR_Process(image_buffer); OLED_ShowString(0,0,result); } ``` #### 四、优化建议 1. **内存管理**:使用$$压缩算法$$减少图像存储空间 $$ P_{compressed} = \sum_{i=0}^{n} (pixel_i \ll (i\%4)*2) $$ 2. **算法加速**:利用STM32的DSP库进行矩阵运算 3. **字体规范**:限定识别字符集可提升准确率[^2] #### 五、开发资源 1. 必用库函数: - `HAL_DCMI_Init()`:摄像头接口初始化 - `HAL_SPI_Transmit()`:OLED屏幕驱动 - `arm_matrix_mult_f32()`:DSP矩阵运算 2. 参考实现: ```c // 二值化处理示例 void Binarize(uint8_t *img, uint8_t threshold) { for(int i=0; i<IMG_SIZE; i++){ img[i] = (img[i] > threshold) ? 255 : 0; } } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值