有一次我在给内部学员直播的时候,有几个小伙伴问我,为什么要进入临界,临界有什么用?
当时一时半会没想到好的解释,今天举几个例子来解答下。
如果有大佬觉得我说得不对请私聊我纠正,不要偷偷在评论区喷我,我改还不行吗。
进入临界和退出临界,我第一次接触是在ucos系统,当时我也感觉这个专业术语很牛逼很复杂的样子。
等我讲完人话,你会发现其实也挺low。
进入临界就是关闭单片机总中断,退出临界就是恢复单片机中断,记住最好是恢复,不是打开,因为进入临界之前单片机总中断未必是开着的。
那为什么要进入临界?有什么作用?
我们假设一种场景:
我们来看这样一段代码。
假设我们程序执行到第12行,也就是a = 0的位置,*p的值肯定是等于2的对吧?
突然有个定时器中断来了,然后在定时器中断处理函数里面,我们执行了p++。
执行完以后回到主程序继续执行,也就是会去执行13行代码,这个时候*p的值你猜猜是多少?
没错,肯定不是2,而是3。
明明条件判断是2,最后值却变成了3,这种bug估计能调到你哭。
这个时候进入临界就稳得一逼了,可以改成如下代码。
STM32不像51单片机这样直接EA=0就能关闭总中断,而是通过__get_PRIMASK和__set_PRIMASK来做,当然也可以用__disable_irq和__enable_irq。
具体的可以在工程里搜索,看注释。
每个单片机关闭总中断的方式都不一样,所以进入临界和退出临界代码也有所区别。
你只需要记住本质就是关总中断,执行完程序后,恢复总中断就行了。
不用过于担心单片机总中断怎么开关的问题,网上随便都能搜到。
那继续聊下我们上面那个程序,很多人可能会说,我写程序的时候规避这种问题,不在定时器里对p指针进行操作不就行了?
如果你是做一些比较大的项目,几乎不可能避免,比如说我们的队列算法。
我们会在串口中断里接收到数据以后,把数据丢进队列,然后在主函数进行解析。
用了队列以后,数据传输稳定性可以大大提升,哪怕是数据传输很快,数据量很大也不会出现丢包的情况。
我们不妨来看下队列算法的入列函数。
这种,如果不用指针,几乎不可能实现,哪怕实现了灵活性和移植性也不高。
试想一下,如果这种在主函数和中断里都会频繁去操作指针,不做临界的话程序很容易因为指针指向异常导致死机或者数据错乱导致的程序逻辑问题。
当然,进入临界除了保护全局变量、数组、结构体等数据不错乱以外,还能是硬件的数据,比如IO、SCI、SPI、flash。
当然,RTOS的或许还要更复杂些,这里就不一一讲解了,理解本质就行。