stm32尽管所有的gpio都可以设置为外部中断的功能,但是不能把所有的gpio同时设置为外部中断。例如不能把PA0和PB0同时设置为外部中断,因为PA0和PB0共用一个中断线,MCU只把最后完成初始化的管脚设置为外部中断。
如果代码编写者明确知道PA0和PB0不会同时触发,并且触发有相互依赖关系,可以通过分时设置PA0和PB0的外部中断功能。但是在大多数情况下,外部中断的触发都是随机的,那么在设计原理图的时候就要考虑到这种情况,把用到的外部中断管脚设置到后缀不同的管脚上,如PA0,PA1,PB3,PC8,....PD12,PF16,在一个工程中最多能使用16个外部中断,并且每个管脚的后缀不同。
如果设计电路板没有考虑到这种情况,则需要根据实际情况把触发频繁的设置为外部中断,不频繁的通过检测管脚电平变化来判断是否有触发。
检测管脚电平变化的一种方法:
设计一个周期1ms的定时器,在定时器中断服务函数中通过判断管脚电平,来捕获上升沿和下降沿。
- <span style="background-color: white;">//捕获下降沿
- 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</span>
复制代码
虽然代码不同,但是思路是一样的:寄存上一周期的状态,并与当前状态相结合进行判断。
stm32的1ms定时器(即每1ms执行一次定时器中断服务子函数)相当于fpga的1个时钟100ns(iclock-10mhz),检测速度相差1万倍。
|