8051單片機I/0口的讀寫
1、作為I/O口使用時,輸入和輸出原理
作I/O口使用時,在控制信號的作用下,與門關(guān)閉,V1截止,同時多路開關(guān)打向下邊,與鎖存器的反向輸出端Q^相連。
輸出數(shù)據(jù)時,“寫鎖存器”端發(fā)出脈沖,將“內(nèi)部總線”上的數(shù)據(jù)寫入輸出鎖存器,由Q^端控制V2,從而在引腳上反應(yīng)出相應(yīng)的狀態(tài)。此種應(yīng)用時P0端口各引腳應(yīng)外接上拉電阻。
輸入數(shù)據(jù)時,分為讀引腳和讀端口兩種方式。讀端口的位置就是上圖中的“讀鎖存器”,而讀引腳則是上圖中的“讀引腳”。至于何時讀端口,何時讀引腳則是根據(jù)不同的指令由硬件自動完成的,這不需要我們操心。
附:
讀端口的指令
讀端口的指令為端口內(nèi)容取反這樣的“讀-修改-寫”指令。資料稱:下面這些的指令才有Read-Modify-Write功效
ANL (logical AND, e.G., ANL P1,A)
ORL (logical OR, e.g., ORL P2,A)
XRL (logical EX-OR, e.g., XRL P3,A)
JBC (jump if bit = 1 and clear bit, e.g., JBC P1.1, LABEL)
CPL (complement bit, e.g., CPL P3.0)
INC (increment, e.g., INC P2)
DEC (decrement, e.g., DEC P2)
DJNZ (decrement and jump if not zero, e.g., DJNZ P3, LABEL)
MOV PX.Y,C(move carry bit to bit Y ofPORTX)
CLR PX.Y(clear bit Y ofPORTX)
SETB PX.Y(set bit Y ofPORTX)
讀引腳的指令
如下所示,讀之前應(yīng)先將端口置1,即先輸出1,修改鎖存器的內(nèi)容為1。
MOV A, P1
MOV 20H, P1
MOV R0, P1
MOV @R0, P1
2、為什么讀之前要先寫1?
從上圖可知,如果不對端口置1,端口鎖存器原來的狀態(tài)有可能為0,Q端為0,Q^為1,加到場效應(yīng)管柵極的信號為1,該場效應(yīng)管就導通對地呈現(xiàn)低阻抗,此時即使引腳上輸入的信號為1也會因端口的低阻抗而使信號拉低,使得外加的1信號讀入后不一定是1,若先執(zhí)行置1操作則可以使場效應(yīng)管截止,引腳信號直接加到三態(tài)緩沖器中,實現(xiàn)正確的讀入。
3、實際應(yīng)用
P0、P1、P2、P3作I/0口使用時,如果要從外部讀取數(shù)據(jù),讀取之前應(yīng)先將端口置1。因初始化時,四個I/O均被初始化為0xFF,所以若端口在整個程序過程中無輸出時,即輸出鎖存器的狀態(tài)始終為1,則讀數(shù)據(jù)時可不用手動置1。(附:參考引腳內(nèi)部結(jié)構(gòu)圖可知,縱使鎖存器D口數(shù)據(jù)在變(內(nèi)部數(shù)據(jù)線),只要時鐘觸發(fā)端clk沒有觸發(fā)信號,鎖存器中的內(nèi)容永遠都不會改變,也就是說,不管外部引腳信號如何變,也不管執(zhí)行過多少次讀引腳操作,端口鎖存器中的內(nèi)容是不會發(fā)生變化的。那么,端口內(nèi)容何時變化?答曰:只要執(zhí)行過一次輸出,端口內(nèi)容必為輸出值,因為端口就是輸出的數(shù)據(jù)鎖存器。)
而當端口實際應(yīng)用時要不停地同外部數(shù)據(jù)交換時,即又有輸出又有讀入時,如單片機與存儲器24C16通過IIC總線相連,SCL總是由單片機輸出可不用考慮,而SDA則可通過單片機向外發(fā)數(shù)據(jù),又可以從外部讀數(shù)據(jù),假若之前發(fā)送的最后一位數(shù)據(jù)為0時,此時若再讀數(shù)據(jù),因為鎖存器的狀態(tài)仍為0,則之后立即讀數(shù)據(jù)時就有可能出錯。所以實際操作時,在發(fā)送數(shù)據(jù)代碼之后,總會有一個單片機主動將SDA拉高的指令,之后再讀數(shù)據(jù)時才不會出問題。(本人淺見,不一定正確)
而當P0與P2口作總線使用與外部并口連接時,應(yīng)該就不會有上面的問題,因為從上圖可知,當作總線使用時,在控制信號的作用下,與門導通,同時多路開關(guān)通過反向器與“地址/數(shù)據(jù)總線”連接,此時V1與V2的驅(qū)動電路形成反相,形成推拉式電路。在這種情況下輸出時,可直接驅(qū)動電路,不需要接上拉電阻(因為V1導通)。而輸入時,直接讀引腳即可,不需置1。(因為輸入時,地址/數(shù)據(jù)線上沒有數(shù)據(jù),V2應(yīng)該截止(我認為的,此處存疑)。
P1_0=!P1_0在KEIL中編譯后的反匯編指令為CPL 1.0;從上面所述可知為讀端口指令(讀-修改-寫),所以可以在C中直接使用,如用作看門狗的清除脈沖和LED燈的閃爍等。
讀端口指令似乎是要對內(nèi)部的端口寄存器(P0~P4)進行改寫時才進行的,所以說要先讀端口鎖存器狀態(tài),改寫完后同時有鎖存器輸出。而讀引腳指令沒有改寫寄存器。
評論