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