stm32尽管所有的gpio都可以设置为外部中断的功能,但是不能把所有的gpio同时设置为外部中断。例如不能把PA0和PB0同时设置为外部中断,因为PA0和PB0共用一个中断线,MCU只把最后完成初始化的管脚设置为外部中断。
如果代码编写者明确知道PA0和PB0不会同时触发,并且触发有相互依赖关系,可以通过分时设置PA0和PB0的外部中断功能。但是在大多数情况下,外部中断的触发都是随机的,那么在设计原理图的时候就要考虑到这种情况,把用到的外部中断管脚设置到后缀不同的管脚上,如PA0,PA1,PB3,PC8,....PD12,PF16,在一个工程中最多能使用16个外部中断,并且每个管脚的后缀不同。
如果设计电路板没有考虑到这种情况,则需要根据实际情况把触发频繁的设置为外部中断,不频繁的通过检测管脚电平变化来判断是否有触发。
检测管脚电平变化的一种方法:
设计一个周期1ms的定时器,在定时器中断服务函数中通过判断管脚电平,来捕获上升沿和下降沿。
//捕获下降沿
void timer_isr(void)//定时器1ms的中断服务子函数
{
if(READ_PIN)
{
state = waitting_falling_edge;
}
else
{
if(state == waitting_falling_edge)
{
falling_trigger = 1;//捕获下降沿
state = falling_edge_detected;
}
}
}
//捕获上升沿
void timer_isr(void)//定时器1ms的中断服务子函数
{
if(!READ_PIN)
{
state = waitting_rising_edge;
}
else
{
if(state == waitting_rising_edge)
{
rising_trigger = 1;//捕获上升沿
state = rising_edge_detected;
}
}
}
类比FPGA的捕获上升沿、下降沿代码(verilog语言)
//捕捉上升沿
module capture_rising(iclock, ireset, isignal, orising);
input iclock;
input ireset;
input isignal;
output orising;
reg isignal_temp0;
reg isignal_temp1;
always @(posedge iclock or negedge ireset)
begin
if(!ireset)
begin
isignal_temp0 <= 1'b0;
isignal_temp1 <= 1'b0;
end
else
begin
isignal_temp0 <= isignal;
isignal_temp1 <= isignal_temp0;
end
end
assign orising = ~isignal_temp1 & isignal_temp0;
endmodule
//捕捉下降沿
module capture_falling(iclock, ireset, isignal, ofalling);
input iclock;
input ireset;
input isignal;
output ofalling;
reg isignal_temp0;
reg isignal_temp1;
always @(posedge iclock or negedge ireset)
begin
if(!ireset)
begin
isignal_temp0 <= 1'b0;
isignal_temp1 <= 1'b0;
end
else
begin
isignal_temp0 <= isignal;
isignal_temp1 <= isignal_temp0;
end
end
assign ofalling = isignal_temp1 & ~isignal_temp0;
endmodule
虽然代码不同,但是思路是一样的:寄存上一周期的状态,并与当前状态相结合进行判断。
stm32的1ms定时器(即每1ms执行一次定时器中断服务子函数)相当于fpga的1个时钟100ns(iclock-10mhz),检测速度相差1万倍。
上一篇:STM32-IIC 配置解说
下一篇:STM32在线调试正常,上电运行不正常
推荐阅读最新更新时间:2024-03-16 15:41