白10:可以通过读取请求挂起寄存器;来判断哪个通断出发了中断。
中断挂起置为1就会继续向左运行。
白110:中断屏蔽给的就是1,看请求挂起给的是几。这个与门就相当于开关
#include "stm32f10x.h" // Device header
uint16_t CountSensor_Count;
void CountSensor_Init(void)
{
//配置RCC
//配置GPIO
//配置AFIO
//配置EXIT 触发方式,不需要开启时钟
//配置NVIC 不需要开启时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
/*1. GPIO口产生电平变化时
2. 需要AFIO(AFIO主要完成两个任务:复用功能引脚重映射、中断引脚选择(相同的Pin不能同时触发中断,所以只能选择多个GPIO(A-G)中的一个))
3. 接下来EXTI将立即向NVIC发出中断申请
4. 经过NVIC裁决后即可中断CPU主程序
*/
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉模式
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;
GPIO_Init(GPIOB, &GPIO_InitStructure);
//配置AFIO的数据选择器
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_Rising; //支持的触发方式:上升沿/下降沿/双边沿触发
//上升沿:拿出来的时候也计数加1
//下降沿:遮挡的时候+1
EXTI_Init(&EXTI_InitStruct); //双边沿触发:遮挡的时候+1,拿出来的时候也加1
//响应中断,
//用来中断分组,参数是中断分组的方式
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
NVIC_InitTypeDef NVIC_InitStruct; //EXTI15_10_IRQn = 40, /*!< External Line[15:10] Interrupts */
NVIC_InitStruct.NVIC_IRQChannel = EXTI15_10_IRQn; //EXIT_LINE14 所以STM32F10X_MD_VL型号下的IRQC选EXTI15_10_IRQn,
NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 1; //抢占优先 ,因为中断分组选的是NVIC_PriorityGroup_2 所以 在0-3之间选一个
NVIC_InitStruct.NVIC_IRQChannelSubPriority = 1; //响应优先
NVIC_Init(&NVIC_InitStruct);
}
//获取计数
uint16_t CountSensor_Get(void)
{
return CountSensor_Count;
}
void EXTI15_10_IRQHandler()
{
//获取中断标志位 //返回值是set 和 reset,表示中断标志为已被设置
//中断标志位是否为1
if(EXTI_GetITStatus(EXTI_Line14)==SET)
{ //如果是 ,执行中断程序
//如果已经设置就加1
CountSensor_Count++; //如果是set,说明获取中断标志位
EXTI_ClearITPendingBit(EXTI_Line14); //每次中断程序结束后,都要清除挂起的中断标志位
}
}
最好不要在中断函数和主函数调用相同的函数或者操作同一个硬件,尤其是硬件相关的函数,比如,OLED显示函数,OLED就会显示错误,啪,进中断了,结果中断里还是OLED显示函数,需要继续原来的显示, 就出问题了!!!!!!!!!!!
为什么呢?因为在主程序中,OLED刚显示一半,程序中断结束后,所以再回来的时候,继续显示的内容跟着跑到其他地方去了,这就出现问题了。
虽然在进入中断和退出的时候会现场保护和现场恢复,但只能保证CPU程序正常返回时不出问题。对于外部硬件,并没有在进入中断时,进行现场保护。因此最好不要在主程序和中断程序里,操作可能冲突的硬件。
在实现的功能的时候,在中断函数里操作变量或者标志位,在其他地方,大家也可以多用变量和标志位来减少代码的耦合性,让部分代码相互独立,仅使用变量,标志位,或者函数作为接口。