首页 > 代码库 > C51指针与A51汇编接口之间关系研究
C51指针与A51汇编接口之间关系研究
最近在研究单片机C51对汇编的接口问题。char和int等都比较简单,使用寄存器或固定地地址传值都是可以的,具体可以参考keil的C51 user‘s guide。本篇短文主要重点讨论一下A51下如何遵循C51的接口标准来实现C51的指针。主要原因是,现在用C51的人越来越多,大家都图省事和方便。网上面有关A51的资料少得可怜,知道用汇编来实现代码优化的少之又少。本人是一直坚持用汇编写东西的。在嵌入式领域,很多东西都与硬件有关,多知道点底层东西还是有好处。
使用工具主要为keil,在windows 7环境下。C51测试程序如下:
#include <reg52.h> unsigned pos; bit abc(unsigned *pos){ *pos=100; return 0; } void main(){ while(1){ abc(&pos); }
这是测试用源程序,很明显程序中使用了int型指针 pos。为了能够生成汇编代码,加入编译器控制:
#pragma SRC #pragma SMALL #include <reg52.h> unsigned pos; bit abc(unsigned *pos){ *pos=100; return 0; } void main(){ while(1){ abc(&pos); } }
注意,#pragma必需在文件的开始位置,否则会报错。另外编译器可能会跳“ EXCEPTION 0021H: PATH OR FILE NOT FOUND:main.obj” ,没有关系,我们的目标是生成汇编的src文件。编译后生成src如下:
; main.SRC generated from: main.c ; COMPILER INVOKED BY: ; D:\Program Files\Keil\C51\BIN\C51.EXE main.c BROWSE DEBUG OBJECTEXTEND TABS(2) $NOMOD51 NAME MAIN ;寄存器及内存声明 P0 DATA 080H P1 DATA 090H P2 DATA 0A0H P3 DATA 0B0H T0 BIT 0B0H.4 AC BIT 0D0H.6 T1 BIT 0B0H.5 T2 BIT 090H.0 EA BIT 0A8H.7 IE DATA 0A8H EXF2 BIT 0C8H.6 RD BIT 0B0H.7 ES BIT 0A8H.4 IP DATA 0B8H RI BIT 098H.0 INT0 BIT 0B0H.2 CY BIT 0D0H.7 TI BIT 098H.1 INT1 BIT 0B0H.3 RCAP2H DATA 0CBH PS BIT 0B8H.4 SP DATA 081H T2EX BIT 090H.1 OV BIT 0D0H.2 RCAP2L DATA 0CAH C_T2 BIT 0C8H.1 WR BIT 0B0H.6 RCLK BIT 0C8H.5 TCLK BIT 0C8H.4 SBUF DATA 099H PCON DATA 087H SCON DATA 098H TMOD DATA 089H TCON DATA 088H IE0 BIT 088H.1 IE1 BIT 088H.3 B DATA 0F0H CP_RL2 BIT 0C8H.0 ACC DATA 0E0H ET0 BIT 0A8H.1 ET1 BIT 0A8H.3 TF0 BIT 088H.5 ET2 BIT 0A8H.5 TF1 BIT 088H.7 TF2 BIT 0C8H.7 RB8 BIT 098H.2 TH0 DATA 08CH EX0 BIT 0A8H.0 IT0 BIT 088H.0 TH1 DATA 08DH TB8 BIT 098H.3 EX1 BIT 0A8H.2 IT1 BIT 088H.2 TH2 DATA 0CDH P BIT 0D0H.0 SM0 BIT 098H.7 TL0 DATA 08AH SM1 BIT 098H.6 TL1 DATA 08BH SM2 BIT 098H.5 TL2 DATA 0CCH PT0 BIT 0B8H.1 PT1 BIT 0B8H.3 RS0 BIT 0D0H.3 PT2 BIT 0B8H.5 TR0 BIT 088H.4 RS1 BIT 0D0H.4 TR1 BIT 088H.6 TR2 BIT 0C8H.2 PX0 BIT 0B8H.0 PX1 BIT 0B8H.2 DPH DATA 083H DPL DATA 082H EXEN2 BIT 0C8H.3 REN BIT 098H.4 T2CON DATA 0C8H RXD BIT 0B0H.0 TXD BIT 0B0H.1 F0 BIT 0D0H.5 PSW DATA 0D0H
?PR?_abc?MAIN SEGMENT CODE ?PR?main?MAIN SEGMENT CODE ?DT?MAIN SEGMENT DATA EXTRN CODE (?C?ISTPTR) EXTRN CODE (?C_STARTUP) PUBLIC pos PUBLIC main PUBLIC _abc RSEG ?DT?MAIN pos: DS 2 ; #pragma SRC ; #pragma SMALL ; #include <reg52.h> ; ; unsigned pos; ; ; bit abc(unsigned *pos){ RSEG ?PR?_abc?MAIN _abc: ; SOURCE LINE # 7 ;---- Variable 'pos?040' assigned to Register 'R1/R2/R3' ---- ; *pos=100; ; SOURCE LINE # 8 ;---------关键代码------------------- CLR A MOV B,#064H LCALL ?C?ISTPTR ;return 0; ; SOURCE LINE # 9 CLR C; }
; SOURCE LINE # 10 ?C0001:RET ; END OF _abc; ; void main(){ RSEG ?PR?main?MAIN main: USING 0; SOURCE LINE # 12 ?C0002:; while(1){ ; SOURCE LINE # 13 ; abc(&pos); ; SOURCE LINE # 14 ;---------关键代码------------------- MOV R3,#00H MOV R2,#HIGH (pos) MOV R1,#LOW (pos) LCALL _abc; } ; SOURCE LINE # 15 SJMP ?C0002 ; END OF mainEND
对于A51的程序格式,这里不多解释,有兴趣可以自己去Keil官网学习。这里主要说一下指针的C51汇编接口。代码中内存和寄存器声明,这个不重要,可以跳过。主要看标记为关键代码的部分。我们可以看到,在main中,在调用abc函数前,主程序初始了三个寄存器,分别是R1\R2\R3,代码如下:
MOV R3,#00H MOV R2,#HIGH (pos) MOV R1,#LOW (pos)
这里得着重讲一下。很明显,这是寄存器传值。依据C51规定,指针传递使用R1\R2\R3寄存器。其中,Mem type in R3, MSB in R2, LSB in R1。也就是说,R3表指针类型,R1为指针内容低位,R2为内容高位。从上面三个语句可以看出,程序将类型设为0,将pos地址高位传给R2,低位给R1。接下来再看abc函数中的操作
CLR A MOV B,#064H LCALL ?C?ISTPTR
我们可以看到,在清零累加器后,将B赋为100,之后调用了?C?ISTPTR。这个函数是C51S.LIB中的系统函数。依据官方介绍这是一个Small model library without floating-point arithmetic。由于是lib文件,具体代码无从得知。将pos的类型改成unsigned char后,得到如下代码:
MOV A,#01H LCALL ?C?CSTPTR
其它部分都一致。由此比较后,我们可以推测:在给系统函数传参时,使用了A\B寄存,当只有一字节时,直接用A,两字节时用A当高位,B为低位。在修改pos大小为1000及改为int后印证了我的想法:
*pos=1000; ; SOURCE LINE # 8 MOV A,#03H MOV B,#0E8H LCALL ?C?ISTPTR
3e8h刚好是1000。具体?C?ISTPTR和?C?CSTPTR是什么呢?从官方手册可知,这些被称为C51: ?C? LOAD AND STORE LIBRARY ROUTINES,也即加载和存储过程。它们通过函数调用方式实现,而非内嵌代码。命名方式形如:?C?tffmmm。其中,t表类型,ff为函数域,mmm为内存类型域。能知道的信息就这么多,无法得到源代码。
另外,需要特别注意的是,C51对R3并没有完全的规定,不管数据类型是什么,R3一直传的是0,并且针对不同的数据类型,编译器会做相应优化,这个可以在将pos改为Long后看出来:
?PR?_abc?MAIN SEGMENT CODE ?PR?main?MAIN SEGMENT CODE ?DT?MAIN SEGMENT DATA EXTRN CODE (?C?LSTKPTR) EXTRN CODE (?C_STARTUP) PUBLIC pos PUBLIC main PUBLIC _abc RSEG ?DT?MAIN pos: DS 4 ; #pragma SRC ; #pragma SMALL ; #include <reg52.h> ; unsigned long pos; ; ; bit abc(unsigned long *pos){ RSEG ?PR?_abc?MAIN _abc: USING 0 ; SOURCE LINE # 6 ;---- Variable 'pos?040' assigned to Register 'R1/R2/R3' ---- ; *pos=100; ; SOURCE LINE # 7 LCALL ?C?LSTKPTR DB 00H DB 00H DB 00H DB 064H ; return 0; ; SOURCE LINE # 8 CLR C ; } ; SOURCE LINE # 9 ?C0001: RET ; END OF _abc ; ; void main(){ RSEG ?PR?main?MAIN main: USING 0 ; SOURCE LINE # 11 ; abc(&pos); ; SOURCE LINE # 12 MOV R3,#00H MOV R2,#HIGH (pos) MOV R1,#LOW (pos) LJMP _abc ; END OF main