某客戶使用STM32F4系列芯片做產品開發,用到USART外設,將其配置在智能卡模式。USART配置為智能卡模式后,并開啟了奇偶校驗。
當MCU通過UART從卡端讀取數據時,如果讀到的數據發生奇偶校驗錯誤,根據相關通信協議,USART硬件會自動在剛收到數據的結尾處強行拉低數據線一個時間段告知智能卡控制器,表示USART接收到的數據奇偶校驗有錯,期待數據重發。
1.jpg (26.7 KB, 下載次數: 99)
下載附件
2016-11-3 11:58 上傳
他發現,當UART從智能卡接收數據遇到奇偶校驗出錯時,的確可以從硬件線路上觀測到數據的重發。可他的軟件接收代碼里卻沒法分辨數據正誤,也就是說不管是否發生校驗錯誤,一律當作正確的數據接收了進來。他覺得甚為奇怪。
重點懷疑代碼問題,查看其相關代碼,他的數據接收流程大致是這樣的:
先檢測到USART_SR寄存器中的RXNE為1;然后從USAR_DR寄存器讀取數據;再接著檢測USART_SR寄存器中的校驗出錯位PE位是否為1,如果是1則丟棄剛才收到的數據。咋看上去,貌似沒啥問題。
如果查看STM32相關芯片的參考手冊就會發現,通過對USART_SR的讀和接著對USART_DR的讀操作序列會導致對RXNE和PE位的清零。
2.png (31.02 KB, 下載次數: 77)
下載附件
2016-11-3 11:58 上傳
既然這樣,按照該客戶的做法,先讀SR,然后讀DR。這個連續操作之后已經就將RXNE和PE清零了,若再來讀PE,永遠讀不到它為1的時候,即發現不了校驗出錯的情況,自然導致數據全部被當做正確的收納了。
所以,他的接收代碼需要稍微調整下,先檢查到RXNE為1后,接著檢查PE是否為1,根據PE是否為1 來決定從DR中讀得數據的取舍并完成對PE和RXNE的清零。
STM32的寄存器中,尤其是那些狀態寄存器的部分狀態標志的置位和清零并不一定是簡單地、對應地直接置1寫0。比方有些寄存器位的清零是對相關寄存器位寫1;有些寄存器位的清零則個軟件操作序列。比方STM32F4/STM32F1系列中USART的PE位、ORE位就是通過軟件操作序列實現清零。STM32產品線眾多,即使相同外設的寄存器不同系列間的操作可能略有差異。比如這里談到的USART的PE位、ORE位,在STM32F0系列里是可以通過軟件對相關寄存器位寫1達到對其清零的目的,此時無需軟件操作序列。
當然,如果你使用ST官方的參考庫函數的話,有些細節可能感受不到。開發過程中在具體使用到某些并不熟悉的寄存器位時,適當地核對下手冊往往是個不錯的舉動。有時一個無意的想當然的舉動可能會浪費很多時間和精力。
簡單問題,分享出來,互為提醒。其實,開發過程中很多折騰人的地方往往就是些小細節。
|