【stm32】对射式红外传感器计次以及旋转编码器计次

对射式红外传感器计次

1. 将传感器的功能分装在一个模块里CountsSenser

2.配置外部中断

1.配置RCC,将涉及的外设的时钟都打开

2.配置GPIO,选择端口为输入模式

3.配置AFIO,选择前面使用的一路GPIO,连接到后面的EXTI

4.配置EXTI,选择边沿触发方式

5.配置NVIC,给中断选择一个合适的优先级

c 复制代码
void GPIO_AFIODeInit(void);

复位AFIO

c 复制代码
void GPIO_PinLockConfig(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);

锁定GPIO引脚配置,防止意外更改

c 复制代码
void GPIO_EventOutputConfig(uint8_t GPIO_PortSource, uint8_t GPIO_PinSource);
void GPIO_EventOutputCmd(FunctionalState NewState);

配置AFIO的事件输出功能

c 复制代码
void GPIO_PinRemapConfig(uint32_t GPIO_Remap, FunctionalState NewState);

用来进行引脚重映射,第一个参数可以选择重映射的方式,第二个参数是新的状态

c 复制代码
void GPIO_EXTILineConfig(uint8_t GPIO_PortSource, uint8_t GPIO_PinSource);

配置AFIO的数据选择器,选择想要的中断引脚

EXTI库函数

c 复制代码
void EXTI_DeInit(void);

把Exti的配置都清除,恢复成上电默认的状态

c 复制代码
void EXTI_Init(EXTI_InitTypeDef* EXTI_InitStruct);

可以根据这个结构体里的参数配置EXTI外设,初始化EXTI

c 复制代码
void EXTI_StructInit(EXTI_InitTypeDef* EXTI_InitStruct);

可以把参数传递的结构体变量赋一个默认值

c 复制代码
void EXTI_GenerateSWInterrupt(uint32_t EXTI_Line);

软件触发外部中断

在外设运行的过程中,会产生一些状态标志位,放在状态寄存器中,当程序想要看到这些标志位时,就可以用下面四个函数:

在主程序里:

c 复制代码
FlagStatus EXTI_GetFlagStatus(uint32_t EXTI_Line);

可以获取指定的标志位是否被置1

c 复制代码
void EXTI_ClearFlag(uint32_t EXTI_Line);

可以对置1的标志位进行清除

在中断函数里:

c 复制代码
ITStatus EXTI_GetITStatus(uint32_t EXTI_Line);

获取中断标志位是否被置1

返回值为SET/RESET

c 复制代码
void EXTI_ClearITPendingBit(uint32_t EXTI_Line);

清除中断挂起标志位

NVIC库函数

c 复制代码
void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup);

用来中断分组,参数是中断分组的方式

分组方式整个芯片只能用一种,按理说分组代码整个工程只需要执行一次就行;如果将它放在模块里面进行分组,要确保每个模块分组都选的是同一个;也可以将它放在主函数的最开始,这样模块里就不用再分组了

c 复制代码
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);

根据结构体里面指定的参数初始化NVIC

c 复制代码
void NVIC_SetVectorTable(uint32_t NVIC_VectTab, uint32_t Offset);

设置中断向量表

c 复制代码
void NVIC_SystemLPConfig(uint8_t LowPowerMode, FunctionalState NewState);

系统低功耗配置

代码:

c 复制代码
void CountSenser_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB , &GPIO_InitStructure);
	
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOB , GPIO_PinSource14);
	
	EXTI_InitTypeDef EXTI_InitStruct;
	EXTI_InitStruct.EXTI_Line = EXTI_Line14 ;
	EXTI_InitStruct.EXTI_LineCmd = ENABLE;
	EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;
	EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Falling;
	EXTI_Init(&EXTI_InitStruct);
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
	NVIC_Init(&NVIC_InitStructure);
}

中断函数

中断函数的名字是固定的,每个中断通道都对应一个中断函数,中断函数的名字参考启动函数,其中以

IRQHandler结尾的字符串就是中断函数的名字

c 复制代码
void EXTI15_10_IRQHandler(void)
{
	if(EXTI_GetITStatus(EXTI_Line14) == SET)
	{
		CountSensor_Count ++;
		EXTI_ClearITPendingBit(EXTI_Line14);
	}
}

旋转编码器计次

新建Encoder.c和Encoder.h

例:旋转编码器接PB0和PB1

Encoder.c

c 复制代码
int16_t Encoder_Count;

void Encoder_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB , &GPIO_InitStructure);
	
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOB , GPIO_PinSource0);
	GPIO_EXTILineConfig(GPIO_PortSourceGPIOB , GPIO_PinSource1);
	
	
	EXTI_InitTypeDef EXTI_InitStruct;
	EXTI_InitStruct.EXTI_Line = EXTI_Line0 | EXTI_Line1;
	EXTI_InitStruct.EXTI_LineCmd = ENABLE;
	EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;
	EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Falling;
	EXTI_Init(&EXTI_InitStruct);
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
	NVIC_Init(&NVIC_InitStructure);
	
	NVIC_InitStructure.NVIC_IRQChannel = EXTI1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
	NVIC_Init(&NVIC_InitStructure);
}

int16_t Encoder_Get(void)
{
	int16_t Temp;
	Temp = Encoder_Count;
	Encoder_Count = 0;
	return Temp;
}
void EXTI0_IRQHandler(void)
{
	if(EXTI_GetITStatus(EXTI_Line0) == SET)
	{
		if(GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0)
		{
			Encoder_Count --;
		}
		EXTI_ClearITPendingBit(EXTI_Line0);
	}
}

void EXTI1_IRQHandler(void)
{
	if(GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_0) == 0)
		{
			Encoder_Count ++;
		}
	if(EXTI_GetITStatus(EXTI_Line1) == SET)
	{
		EXTI_ClearITPendingBit(EXTI_Line1);
	}
}

main.c

c 复制代码
int16_t Num;

int main(void)
{
	OLED_Init();
	Encoder_Init();
	
	while(1)
	{
		Num += Encoder_Get();
		OLED_ShowSignedNum(1, 5, Num, 5);
	}
}

注:
1.中断函数里,最好不要执行耗时过长的代码,中断函数要简短快速,不要刚进中断就执行Delay,因为中断是处理突发的事情,如果为了一个突发的事情待在中断里不出来了,主程序就会收到严重阻塞

2.最好不要在中断函数和主函数调用相同的函数或者操作同一个硬件,尤其是硬件相关的函数,在实现功能的时候,可以在中断里操作变量或者标志位,当中断返回时,再对这个变量进行显示和操作,这样既能保证中断函数的简短快速,又能保证不产生冲突的硬件操作

注:

该文章为b站江协课程,为笔记

相关推荐
来自晴朗的明天11 分钟前
13、NMOS 电源防反接电路
单片机·嵌入式硬件·硬件工程
17(无规则自律)1 小时前
深入浅出 Linux 内核模块,写一个内核版的 Hello World
linux·arm开发·嵌入式硬件
芯岭技术1 小时前
PY32MD310单片机:高性能、低功耗的32位电机控制微控制器
单片机·嵌入式硬件
wotaifuzao2 小时前
STM32 + FreeRTOS 的订阅通知组件架构
stm32·嵌入式硬件·架构·freertos·事件驱动·嵌入式架构
小龙报2 小时前
【51单片机】深度解析 51 串口 UART:原理、配置、收发实现与工程化应用全总结
c语言·开发语言·c++·stm32·单片机·嵌入式硬件·51单片机
Lester_11019 小时前
STM32 高级定时器PWM互补输出模式--如果没有死区,突然关闭PWM有产生瞬间导通的可能吗
stm32·单片机·嵌入式硬件·嵌入式软件
小李独爱秋11 小时前
“bootmgr is compressed”错误:根源、笔记本与台式机差异化解决方案深度指南
运维·stm32·单片机·嵌入式硬件·文件系统·电脑故障
梁洪飞13 小时前
内核的schedule和SMP多核处理器启动协议
linux·arm开发·嵌入式硬件·arm
进击的小头13 小时前
实战案例:51单片机低功耗场景下的简易滤波实现
c语言·单片机·算法·51单片机
宵时待雨17 小时前
STM32笔记归纳8:时钟
笔记·stm32·单片机·嵌入式硬件