导航: 老古网老古论坛XMOS公共讨论区XMOS开源项目区单片机程序设计嵌入式系统广告区域
→哪里有"在c51里调用汇编函数"的详细说明?感觉是个挺麻烦的[guest]

 *第14850篇: 哪里有"在c51里调用汇编函数"的详细说明?感觉是个挺麻烦的工作,是不是?

  
楼 主:guest 2003年7月24日21:19
 哪里有"在c51里调用汇编函数"的详细说明?感觉是个挺麻烦的工作,是不是?
好像有个很长的头?
  
2楼:guest 2003年7月24日23:52
 我为了这个问题以头痛了好几个星期了。原以
我为了这个问题以头痛了好几个星期了。原以为只要加上#pragma  asm和#pragma  endasm便可。谁知不行,如果你有了答案请在网上公开一下。
  
3楼:guest 2003年7月25日08:34
 这个问题捆饶我好久了,那位大侠快帮帮兄弟们
  
4楼:guest 2003年7月25日10:39
 没那么简单!
在C51中调用A51函数,主要是A51的命名规则和参数传递规则必须符合C51的要求;下面我主要针对Franklin C51做一简要说明:
一、函数名转换规则;
C51程序模块编译成目标文件后,其中的函数名依据其定义的性质不同回转换为不同的函数名,其转换规则如下:
   1、无参数传递或参数不通过寄存器传递的函数起函数名不做改变转入目标文件中;例如C51中申明的函数 void FunName(void),编译后名称为 FunName;
   2、如果函数通过寄存器传递参数,则编译后在原函数名前加前缀"_";例如C51中申明的函数为:void FunName(char),编译后名称为:_FunName;
   3、如果函数为包含堆栈内参数传递的可重入函数,则加前缀"_?";例如C51中申明的函数为:void FunName(void)reentrant,编译后名称为:_?FunName;
二、函数相关段命名规则;
     一个C51源程序模块被编译后,其中的每一个函数以" ?PR?函数名?模块名 "为名的命名规则被分配到一个独立的CODE段。例如,如果模块“FUNC51”内包含一个名为“func”的函数,则其CODE段名为:“?PR?FUNC?FUNC51”。如果函数中包含有data和bit对象的局部变量,编译器将按“?函数名?BYTE”和"?函数名?BIT"命名规则建立一个data和bit段,他们代表所要传递参数的起始位置,其偏移值为0。这些段是公开的,因而它们可以被其他模块访问。另外这些段被赋予“OVERLAYABLE”标志,故可被连接/定位器作覆盖分析。A51和C51相互调用时,A51必须服从下列命名规则:
   1、程序代码:段类型为CODE,则段名为:?PR?函数名?模块名(所有存储器模式);
   2、局部变量:段类型为DATA,段名为:?DT?函数名?模块名(SMALL模式);段类型为PDATA,段名为:?PD?函数名?模块名(COMPACT模式);段类型为XDATA,段名为:?XD?函数名?模块名(LARGE模式);
  3、局部bit变量:段类型为BIT,段名为::?BI?函数名?模块名(所有存储器模式);
三、参数传递规则;
  Flanklin C51函数最多可以通过CPU寄存器传递3个参数,如果参数太多以至于寄存器不够用时,部分参数将在固定存储区内传递,这种混合情况非常不利于程序员编程,如果在源程序中选择了编译控制命令“#pragma NOREGPARMS”则所有参数传递都发生在固定存储区,存储区的地址空间依赖于所选择的存储器模式。
   1、寄存器参数传递规则:第一个参数,如果是单字节变量就通过R7,双字节变量通过R6R7,4字节通过R4R5R6R7,一般指针通过R1R2R3;     第二个参数,单字节通过R5,双字节变量通过R4R5,4字节通过R4R5R6R7,一般指针通过R1R2R3;  第三个变量,单字节变量通过3,双字节通过R2R3,一般指针通过R1R2R3,不能传送4字节变量;
   2、函数返回值的传递规则:布尔变量通过进位位C传递,其余类型的传递规则和第一个参数的传递规则相同。
   例如:有一个A51程序为FunA51,要在C51中调用该函数,程序如下:
extern FunA51(unsigned char x, unsigned char y)
void main(void)
{
    unsigned char R;
    R=FunA51(0x12,0x32);
}

   NAME  MyA51  ;定义模块名
?PR?FunA51?MyA51   SEGMENT CODE  ;FunA51代码段声明

PUBLIC  _FunA51  ;公开函数名,以便C模块可以调用它, 因为函数有返回值故加了前缀"_"

  RSEG   ?PR?FunA51?MyA51
_FunA51:
   MOV  A, R7   ;第一个参数传递
   MOV  B, R5   ;第二个参数传递
   DIV   AB
   MOV  R7, A  ;结果经R7返回
   RET

   END


以上是一个简单的说明,仅供参考,不足之处请多多指教,对于Keil C51不知道是不是这样就不得而知了.

  
5楼:孟祥 2003年7月25日17:09
 看看这里,应该是比较详细了!
关于在 KEIL C51 中嵌入汇编以及C51与A51间的相互调用 作者[ycong_kuang]©[Youth]© 
--- 摘自《C51BBS离线版光盘》--- 

    如何在 KEIL C51(v6.21) 中调用汇编函数的一个示例 [ycong_kuang]

有关c51调用汇编的方法已经有很多帖子讲到,但是一般只讲要点,很少有对整个过程作详细描述,对于初学者是不够的,这里笔者
通过一个简单例子对这个过程进行描述,希望能对初学者有所帮助。几年来,在这个论坛里笔者得到很多热心人指导,因此也希望
藉此尽一点绵薄之力。

在这个例子里,阐述了编写c51程序调用汇编函数的一种方法,这个外部函数的入口参数是一个字符型变量和一个位变量,返回值是
一个整型变量。例中,先用c51写出这个函数的主体,然后用SRC控制指令编译产生asm文件,进一步修改这个asm文件就得到我们所
要的汇编函数。该方法让编译器自动完成各种段的安排,提高了汇编程序的编写效率。

step1. 按写普通c51程序方法,建立工程,在里面导入main.c文件和CFUNC.c文件。

相关文件如下:
//main.c文件
#include   < reg51.h >  

#define uchar unsigned char
#define uint unsigned int

extern uint AFUNC(uchar v_achr,bit v_bflag);

void main()
{
    bit BFLAG;
    uchar mav_chr;
    uint    mvintrslt;

    mav_chr=0xd4; BFLAG=1;
    mvintrslt=AFUNC(mav_chr,BFLAG);
}

//CFUNC.c文件

#define uchar unsigned char
#define uint unsigned int

uint AFUNC(uchar v_achr,bit v_bflag)
{
    uchar tmp_vchr;
    uint  tp_vint;

    tmp_vchr=v_achr;
    tp_vint=(uint)v_bflag;
    return tmp_vchr+(tp_vint  <  <8);
}

step2. 在 Project 窗口中包含汇编代码的 C 文件上右键,选择“Options for ...”,点击右边的“Generate Assembler SRC
        File”和“Assemble SRC File”,使检查框由灰色变成黑色(有效)状态;

step3. 根据选择的编译模式,把相应的库文件(如 Small 模式时,是 Keil\C51\Lib\C51S.Lib)加入工程中,该文件必须作为工
       程的最后文件;

step4. build这个工程后将会产生一个CFUNC.SRC的文件,将这个文件改名为CFUNC.A51(也可以通过编译选项直接产生CFUNC.A51文
       件),然后在工程里去掉库文件(如C51S.Lib)和CFUNC.c,而将CFUNC.A51添加到工程里。

//CFUNC.SRC文件如下
.\CFUNC.SRC generated from: CFUNC.c
NAME CFUNC

?PR?_AFUNC?CFUNC     SEGMENT CODE
?BI?_AFUNC?CFUNC     SEGMENT BIT OVERLAYABLE
    PUBLIC    ?_AFUNC?BIT
    PUBLIC    _AFUNC

    RSEG  ?BI?_AFUNC?CFUNC
?_AFUNC?BIT:
    v_bflag?041:   DBIT   1
; #define uchar unsigned char
; #define uint unsigned int
;
; uint AFUNC(uchar v_achr,bit v_bflag)

    RSEG  ?PR?_AFUNC?CFUNC
_AFUNC:
    USING    0
            ; SOURCE LINE # 5
;---- Variable 'v_achr?040' assigned to Register 'R7' ----
; {
            ; SOURCE LINE # 6
;     uchar tmp_vchr;
;     uint    tp_vint;
;
;     tmp_vchr=v_achr;
            ; SOURCE LINE # 10
;---- Variable 'tmp_vchr?042' assigned to Register 'R5' ----
    MOV      R5,AR7
;     tp_vint=(uint)v_bflag;
            ; SOURCE LINE # 11
    MOV      C,v_bflag?041
    CLR      A
    RLC      A
;---- Variable 'tp_vint?043' assigned to Register 'R6/R7' ----
;     return tmp_vchr+(tp_vint  <  <8);
            ; SOURCE LINE # 12
    MOV      R6,A
    MOV      R4,#00H
    CLR      A
    ADD      A,R5
    MOV      R7,A
    MOV      A,R4
    ADDC     A,R6
    MOV      R6,A
; }
            ; SOURCE LINE # 13
?C0001:
    RET
; END OF _AFUNC

    END

step5. 检查main.c的“Generate Assembler SRC File”和“Assemble SRC File”是否有效,若是有效则点击使检查框变成无效状
       态;再次build这个工程,到此你已经得到汇编函数的主体,修改函数里面的汇编代码就得到你所需的汇编函数了。

参考文献:
  1.徐爱钧,彭秀华。单片机高级语言C51windows环境编程与应用,电子工业出版社
  2,  C51编程:关于在 KEIL C51 中直接嵌入汇编。。。帖子编号: 83838 发表用户:Youth
  .................................................................................................................
                                 keil中汇编函数调用c51函数 [ycong_kuang]

在keil的写法可参考89852帖子,具体如下:
与89852帖子相比,第一步在工程里多了一个被汇编调用的c51的函数文件(c51func.c),至于汇编函数还是先用c51编写出主体
(a51func.c),这样汇编程序接口和段都交给编译器处理,你只管在编译成汇编代码后按你的要求改写汇编代码就行了。

例程如下:
//main.c
#include   < reg51.h >  

#define uchar unsigned char
#define uint unsigned int

extern uint AFUNC(uchar v_achr,bit v_bflag);

void main()
{
    bit BFLAG;
    uchar mav_chr;
    uint    mvintrslt;

    mav_chr=0xd4; BFLAG=1;
    mvintrslt=AFUNC(mav_chr,BFLAG);
}

//a51FUNC.c

#define uchar unsigned char
#define uint unsigned int

extern uint CFUNC(uint);

uint AFUNC(uchar v_achr,bit v_bflag)    //c51写的汇编函数,最终要变成汇编代码
{
    uchar tmp_vchr;
    uint  tp_vint;

    tmp_vchr=v_achr;
    tp_vint=(uint)v_bflag;

    return CFUNC(tp_vint);             //这里调用一个c51函数
}

//c51FUNC.c

#define uchar unsigned char
#define uint unsigned int

uint CFUNC(uint v_int)                //被汇编函数调用c51函数
{
    return v_int  <  <2;
}

第二步是按89852帖子的step2,3,4把用c51写的(汇编)函数变成a51文件(今天我试了一下step3可以不要)例程编译结果如
下:
; .\a51func.SRC generated from: a51func.c
NAME    A51FUNC

?PR?_AFUNC?A51FUNC   SEGMENT CODE
?DT?_AFUNC?A51FUNC   SEGMENT DATA OVERLAYABLE
?BI?_AFUNC?A51FUNC   SEGMENT BIT OVERLAYABLE
    EXTRN    CODE (_CFUNC)
    PUBLIC    ?_AFUNC?BIT
    PUBLIC    _AFUNC

    RSEG  ?DT?_AFUNC?A51FUNC
?_AFUNC?BYTE:
   tmp_vchr?042:   DS   1

    RSEG  ?BI?_AFUNC?A51FUNC
?_AFUNC?BIT:
    v_bflag?041:   DBIT   1
; //a51FUNC.c
;
; #define uchar unsigned char
; #define uint unsigned int
;
; extern uint CFUNC(uint);
;
; uint AFUNC(uchar v_achr,bit v_bflag)

    RSEG  ?PR?_AFUNC?A51FUNC
_AFUNC:        ;c51所写的函数产生的汇编代码从这里开始
    USING    0
            ; SOURCE LINE # 8
;---- Variable 'v_achr?040' assigned to Register 'R7' ----
; {
            ; SOURCE LINE # 9
;     uchar tmp_vchr;
;     uint  tp_vint;
;
;     tmp_vchr=v_achr;
            ; SOURCE LINE # 13
    MOV      tmp_vchr?042,R7
;     tp_vint=(uint)v_bflag;
            ; SOURCE LINE # 14
    MOV      C,v_bflag?041
    CLR      A
    MOV      R6,A
    RLC      A
    MOV      R7,A
;---- Variable 'tp_vint?043' assigned to Register 'R6/R7' ----
;      这里说明R6,R7内容就是tp_vint
;     return CFUNC(tp_vint);
            ; SOURCE LINE # 16
    LCALL    _CFUNC    ;这里调用了用c51写的函数
; }
            ; SOURCE LINE # 17
?C0001:
    RET
; END OF _AFUNC

    END

这个文件就是你的汇编函数所在文件,把函数里面的汇编代码修改成你所需的汇编函数就ok了。

建议参考 徐爱钧,彭秀华所写的《单片机高级语言C51windows环境编程与应用》或马忠梅所写的
《单片机的c语言应用程序设计》有关混合语言编程有关章节

  .................................................................................................................
                                关于在 KEIL C51 中直接嵌入汇编。。。 [Youth]
有时在C51程序中需要嵌入一些汇编代码,这时当然可以用通常的作法:
按照 C51 与汇编的接口写一个汇编函数,然后在 C51 程序中调用该函数。(此种方法可在论坛里搜索,以前有很多帖子讲到,不再
重复)

下面介绍直接嵌入汇编代码的方法:

1、在 C 文件中要嵌入汇编代码片以如下方式加入汇编代码:
#pragma ASM
 ; Assembler Code Here
#pragma ENDASM

2、在 Project 窗口中包含汇编代码的 C 文件上右键,选择“Options for ...”,点击右边的“Generate Assembler SRC File”
和“Assemble SRC File”,使检查框由灰色变成黑色(有效)状态;

3、根据选择的编译模式,把相应的库文件(如 Small 模式时,是 Keil\C51\Lib\C51S.Lib)加入工程中, 该文件必须作为工程的最
后文件;

4、编译,即可生成目标代码。
 
 


>>>>>>对该主题发表你的看法

本主题贴数5,分页: [第1页]


[上一篇主题]:如何把单片机的普通I/O口用软件方式扩展为全双工串口

[下一篇主题]:[公告]