汇编语言的过程调用与c语言的函数调用
c语言调用汇编语言函数的方式

c语言调用汇编语言函数的方式以C语言调用汇编语言函数的方式在编程中,有时需要使用汇编语言来实现一些特定的功能。
但是,由于汇编语言复杂且难以维护,因此我们通常使用高级语言如C语言来编写主要的程序代码。
在这种情况下,我们需要一种方法来调用汇编语言函数,以便在程序中使用它们。
本文将介绍如何使用C 语言调用汇编语言函数。
1. 定义汇编函数我们需要编写一个汇编函数来执行我们需要的操作。
汇编函数可以使用任何汇编语言,但是必须遵循特定的调用约定。
在x86架构中,调用约定指定了函数参数的传递方式、寄存器的使用方式以及栈的使用方式。
以Windows操作系统为例,Windows API使用的是stdcall调用约定,即参数从右往左依次压入栈中,返回值放在EAX寄存器中。
因此,我们需要在编写汇编函数时遵循这个调用约定。
下面是一个使用汇编语言实现计算阶乘的例子:```global factorial ; 导出函数符号section .textfactorial:push ebp ; 保存调用函数的栈帧指针 mov ebp, esp ; 设置当前栈帧指针mov eax, [ebp+8] ; 获取函数参数cmp eax, 1 ; 判断参数是否为1jle .base_case ; 如果是,则返回1dec eax ; 否则,计算(n-1)!push eax ; 保存(n-1)的值call factorial ; 递归调用阶乘函数pop ecx ; 恢复(n-1)的值imul eax, ecx ; 计算n*(n-1)!jmp .done ; 返回结果.base_case:mov eax, 1 ; 如果参数为1,则返回1.done:mov esp, ebp ; 恢复栈指针pop ebp ; 恢复调用函数的栈帧指针 ret ; 返回函数结果```在这个例子中,我们定义了一个名为“factorial”的函数,该函数计算输入参数的阶乘。
Keil软件“C语言”与“汇编”混编——相关知识整理

Keil软件“C语言”与“汇编”混编相关知识整理用Keil在C中嵌入汇编 (1)在Keil中嵌入汇编 (2)介绍直接嵌入汇编代码的方法 (4)采用汇编可能会有的好处 (5)Keil C语言与汇编语言混合编程 (7)深入剖析Keil C51 ——从汇编到C51 (9)C语言和汇编语言的变量以及函数的接口问题 (14)汇编与C语言混合编程的关键问题 (15)KEIL段重定位 (15)用Keil在C中嵌入汇编早前公布了C和汇编混编的温度控制器程序,收到一些朋友的询问,他们无法在自己程序中使用我的18B20的汇编子程序或无法正常通过混编后的程序编译。
其实在KEIL中嵌入汇编的方法很简单。
如图一,在C文件中要嵌入汇编的地方用#pragma asm和#pragma endasm 分隔开来,这样编译时KEIL就知道这中间的一段是汇编了。
图1在有加入汇编的文件中,还要设置编译该文件时的选项图2Generate Assembler SRC File 生成汇编SRC文件Assemble SRC File 封装汇编文件(如图三的状态为选中)选上这两项就可以在C中嵌人汇编了,设置后在文件图示中多了三个红色的小方块。
图3为了能对汇编进行封装还要在项目中加入相应的封装库文件,在笔者的项目中编译模式是小模式所以选用C51S.LIB。
这也是最常用的。
这些库文件是中KEIL安装目录下的LIB目录中。
加好后就可以顺利编译了。
(注:我只在7.0以上版本使用过)图4在Keil中嵌入汇编1、其实在KEIL中嵌入汇编的方法很简单。
如图1,在C文件中要嵌入汇编的地方用#pragma as m和#pragma endasm分隔开来,这样编译时KEIL就知道这中间的一段是汇编了。
2、在有加入汇编的文件中,还要设置编译该文件时的选项,如图2所示。
3、Generate Assembler SRC File 生成汇编SRC文件Assemble SRC File 封装汇编文件(如图3的状态为选中)选上这两项就可以在C中嵌人汇编了,设置后在文件图示中多了三个红色的小方块。
在汇编程序中调用C函数

3.4.2 在汇编程序中调用C函数从汇编程序中调用C语言函数的方法实际上在上面已经给出。
在上面C语言例子对应的汇编程序代码中,我们可以看出汇编程序语句是如何调用swap()函数的。
现在我们对调用方法作一总结。
在汇编程序调用一个C函数时,程序需要首先按照逆向顺序把函数参数压入栈中,即函数最后(最右边的)一个参数先入栈,而最左边的第1个参数在最后调用指令之前入栈,如图3-6所示。
然后执行CALL 指令去执行被调用的函数。
在调用函数返回后,程序需要再把先前压入栈中的函数参数清除掉。
调用函数时压入堆栈的参数在执行CALL指令时,CPU会把CALL指令的下一条指令的地址压入栈中(见图3-6中的EIP)。
如果调用还涉及代码特权级变化,那么CPU会进行堆栈切换,并且把当前堆栈指针、段描述符和调用参数压入新堆栈中。
由于Linux内核中只使用中断门和陷阱门方式处理特权级变化时的调用情况,并没有使用CALL指令来处理特权级变化的情况,因此这里对特权级变化时的CALL指令使用方式不再进行说明。
汇编中调用C函数比较"自由",只要是在栈中适当位置的内容就都可以作为参数供C函数使用。
这里仍然以图3-6中具有3个参数的函数调用为例,如果我们没有专门为调用函数func()压入参数就直接调用它的话,那么func()函数仍然会把存放EIP位置以上的栈中其他内容作为自己的参数使用。
如果我们为调用func()而仅仅明确地压入了第1、第2个参数,那么func()函数的第3个参数p3就会直接使用p2前的栈中内容。
在Linux 0.1x内核代码中就有几处使用了这种方式。
例如在kernel/sys_call.s汇编程序中第231行上调用copy_process()函数(kernel/fork.c中第68行)的情况。
在汇编程序函数_sys_fork中虽然只把5个参数压入了栈中,但是copy_process()却带有多达17个参数(见下面的程序)。
Keil C语言与汇编混合用

keil C语言与汇编语言混合编程1. C语言中嵌入汇编1、在C 文件中要嵌入汇编代码片以如下方式加入汇编代码:#pragma ASM; Assembler Code Here#pragma ENDASM2、在Project 窗口中包含汇编代码的C 文件上右键,选择“Options for ...”,点击右边的“Generate Assembler SRC File”和“Assemble SRC File”,使检查框由灰色变成黑色(有效)状态;3、根据选择的编译模式,把相应的库文件(如Small 模式时,是Keil\C51\Lib\C51S.Lib)加入工程中, 该文件必须作为工程的最后文件;4、编译,即可生成目标代码。
来个实例吧:#i nclude <reg51.h>{P2=1;#pragma asmMOV R7,#10DEL:MOV R6,#20DJNZ R6,$DJNZ R7,DEL#pragma endasmP2=0;}2 . 无参数传递的函数调用1.无参数传递的函数调用先来个例子:其中example.c和example.a51为项目中的两个文件***********************example.c*********************************************** extern void delay100();main(){delay100;}***********************example.a51*********************************************** ?PR?DELAY100 SEGMENT CODE; // 在程序存储区中定义段PUBLIC DELAY100; //声明函数RSEG ?PR?DELAY100; //函数可被连接器放置在任何地方DELAY100/P>MOV R7,#10DEL:MOV R6,#20DJNZ R6,$DJNZ R7,DELRETEND在example.c文件中,先声明外部函数,然后直接在main中调用即可。
10讲:宏汇编、C语言调用汇编

宏定义可以带参数: 例: SHIFT1 MACRO X,Y MOV CL, X SAL Y,CL ENDM 调用时: SHIFT1 2, BX;BX算术左移2位. 例: SHIFT2 MACRO X,Y,Z MOV CL, X S&Z Y,CL ENDM 调用时: SHIFT2 2, SI,HR;SI逻辑右移2位源自*作业6:P124:题3-2,
22
8.数据返回C程序时,类型不同,使用的寄 存器不同,见表3-6. 例: CHAR 型: AL LONG 型: DX,AX (二) 举例 例3-16 气泡排序用汇编;数组元素在C程序 中输入,基本程序结构如下:
17
;exm3-16.c #include “stdio.h” extren void sort(int*,int); Main() { int aa[50]; … sort(aa,n) … }
1
宏调用:用宏名(也称宏指令)。 宏扩展:汇编时,将宏指令替换成宏体。 例:源程序 汇编时 … … SHIFT MOV CL, 4 … SAL AL, CL SHIFT … MOV CA, 4 SAL AL, CL 注:与子程序一样,可以简化源程序书写, 但有区别.
2
源程序:主程序 子程序 … SHIFT PROC CALL SHIFT MOV CL, 4 … SAL AL, CL CALL SHIFT RET 宏定义与子程序的区别: 1.子程序在运行时起作用;而宏定义在汇 编时起作用. 2.相同程序,子程序省目标码,而宏定义执 行速度快.
PUBLIC 名字 [,…] 功能:定义可以被其它模块引用的全局名字. 二、外部过程的调用 主模块: EXTRN SUB-PROC:FAR CODE SEGMENT MAIN-PROC PROC FAR … CALL SUB-PROC … RET MAIN-PROC ENDP
C语言与汇编语言混合编程

1) 各寄存器的使用规则及其相应的名称。 2) 数据栈的使用规则。 3) 参数传递的规则。
相对于其他类型的ATPCS, 满足ATPCS的程序的 执行速度更快,所占用的内存更少,但是它不能提供以 下的支持:ARM程序和Thumb程序相互调用、数据以 及代码的位置无关的支持、子程序的可重入性和数据 栈检查的支持。
途。 3.参数传递规则 根据参数个数是否固定可以将子程序分为参数
个数固定的子程序和参数个数可变的子程序。 (1)参数个数可变的子程序参数传递规则
对于参数个数可变的子程序,当参数个数不超
过4个时,可以使用寄存器R0~R3来传递参数,当参 数超过4个时,还可以使用数据栈来传递参数。 (2)参数个数固定的子程序参数传递规则
❖ 结果为一个64位整数时,可以通过寄存器R0和R1 返回,依次类推。
❖ 结果为一个浮点数时,可以通过浮点运算部件的寄 存器f0、d0或者s0来返回。
❖ 结果为复合型的浮点数时,可以通过寄存器f0~fN或 者d0~dN来返回。
❖ 对于位数更多的结果,需要通过内存来传递。
对有调用关系的所有子程序必须遵守同一种 ATPCS。
嵌入式系统设计与开发
C语言与汇编语言混合编程
1.1 C程序与汇编程序互相调用规则 1.2 内嵌汇编程序设计 1.3 C语言函数和ARM汇编语言函数间互相调用
C语言与汇编语言混合编程
1.1 C程序与汇编程序互相调用规则 为了使单独编译的C语言程序和汇编程序之间能
够相互调用,必须让子程序间的调用遵循一定的规则。 ATPCS即ARM,Thumb过程调用标准,是ARM程序和 Thumb程序中子程序调用的基本规则,它规定了一些子 程序间调用的基本规则。下面介绍几种ATPCS规则:
浅谈单片机中C语言与汇编语言的转换

浅谈单⽚机中C语⾔与汇编语⾔的转换⼀、单⽚机课设题⽬要求与软件环境介绍做了⼀单⽚机设计,要⽤C语⾔与汇编语⾔同时实现,现将这次设计的感受和收获,还有遇到的问题写下,欢迎感兴趣的朋友交流想法,提出建议。
单⽚机设计:基于51单⽚机的99码表设计软件环境:Proteus8.0 + Keil4要求:1,开关按⼀下,数码管开始计时。
2,按两下,数码管显⽰静⽌。
3,按三下,数码管数值清零。
⼆、C语⾔程序1 #include<reg51.h>2#define uint unsigned int3#define uchar unsigned char4 uchar shi,ge,aa,keycount=0,temp;5 sbit anjian=P1^7;6 uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};7void display(shi,ge);8void key ();9void init();10void delay(uint z);11/*-----主程序-----*/12void main()13 {14 init(); //初始化15while(1)16 {17 key ();18if(keycount==1)19 TR0=1; //开中断20if(keycount==2)21 TR0=0;22if(keycount==3)23 {24 temp=0;25 keycount=0;26 }27if(aa==10){aa=0;28if(temp<=99)29 {30 temp++;display(shi,ge);31 }32else33 temp=0;}34 }35 }363738/*------初始化程序-------*/39void init()40 {41 keycount=0;42 temp=0;43 TMOD=0x01;44 TH0=(65536-50000)/256;45 TL0=(65536-50000)%256;46 EA=1;47 ET0=1;48//TR0=0;49 }50/*-----定时器中断-----*/51void timer0() interrupt 152 {53 TH0=(65536-50000)/256;54 TL0=(65536-50000)%256;55 aa++;56 }57/*-----显⽰⼦程序-----*/58void display(shi,ge)59 {60 shi=temp/10;61 ge=temp%10;62 P0=table[shi];;delay(70);63 P2=table[ge]; ;delay(70);64 }65/*-----按键检测⼦程序-----*/66void key ()67 {68if(anjian==0)69 {70 delay(5); //消抖71if(anjian==0)72 keycount++;73 }74//while(anjian==0);75//display(shi,ge); //等待按键弹起76 }77/*-----延时⼦程序-----*/78void delay(uint z) //延时约1ms79 {80uint x,y;81for(x=z;x>0;x--)82for(y=100;y>0;y--);83 }电路仿真结果如下:三、C语⾔转汇编语⾔步骤好了,那么接下来我们就开始C语⾔——>汇编语⾔之旅(1)C语⾔1-10⾏改为1 ORG 0000H //汇编起始伪指令,功能是规定程序存储器中源程序或数据块存放的起始地址2 ajmp STAR //ajmp⽆条件跳转指令3 ORG 000bh4 ajmp timer05 anjian equ P1.7 //位定义6 keycount equ 40h7 shi equ 41h8 gewei equ 42h9 aa equ 43h10 temp equ 44h11tab: db 3fh,6h,5bh,4fh,66h //建表12 db 6dh,7dh,7h,7fh,6fh(2)C语⾔中的初始化函数 12-14⾏和39-49⾏改为1STAR:2 acall init //⼦程序近程调⽤指令,功能是主程序调⽤⼦程序,调⽤⼦程序的范围为2kb1init:2mov keycount,#0 //keycount=03mov temp,#0 //temp=14mov tmod,#01h //TMOD=0x015mov TH0,#606mov TL0,#1767setb EA //位置位指令,对操作数所指出的位进⾏置1操作8setb ET09setb TR010retacall为⼦程序近程调⽤指令,返回⽤ret。
c语言直接调用汇编函数

c语言直接调用汇编函数C语言作为一种高级语言,它的代码比汇编语言更容易阅读和理解。
但是在一些需要最大化性能的场合,我们需要使用汇编语言编写低级代码来达到最优性能。
这时,可以通过c语言直接调用汇编函数来解决问题。
一、汇编函数调用格式1.汇编函数需要使用global指令将该函数声明为全局变量,使c语言中的程序可以使用汇编函数。
2.如下所示是一个简单的汇编函数,功能是求两个整数之和:global _Add_Add:mov eax,[esp+4] ;1.将第一个参数传入eax寄存器add eax,[esp+8] ;2.将第二个参数加到eax中ret ;3.返回计算结果3.在c语言程序中,可以使用以下代码调用该汇编函数:int Add(int a, int b) {int result;__asm {mov eax, amov ebx, bcall _Add ;直接调用汇编函数_Addmov result, eax ;将返回值保存到result中}return result;}二、汇编函数调用过程分析1.在c语言程序中,调用_complement函数。
2.编译器将调用_add函数的指令转换为汇编代码,包含将两个参数传递到Add函数中的语句和对_Add函数的调用语句。
该代码被插入到c 语言程序的代码中,正确地传递两个参数到汇编代码中。
3.汇编代码执行所需要的寄存器的状态在调用之前就已经被保存在栈中。
当_add函数执行完毕并返回结果时,它将计算结果存储在eax寄存器中。
4.返回值从汇编代码中流回到c语言的Add函数中,并且被存储于result变量中。
5.返回到c语言程序执行下一条指令,并返回计算结果。
三、注意事项1.汇编函数需要声明为全局变量,使用global指示C编译器在可执行文件中导出该函数的符号。
2.汇编函数的签名必须与c语言函数的签名相同或相似,如返回类型和参数的数量/类型/顺序必须一致。
3.在汇编函数中,必须保证调用函数前和返回函数时,所有寄存器的状态和堆栈指针(rp)的状态都应该与c代码中调用函数的状态一致。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
{{
(*)(*);
}}
两个函数都是返回变量和地积,可通用地或称为参数化版本可用于任意两整数之积,而专用地版本仅能计算全局变量和地乘积.文档来自于网络搜索
从变量地作用域原则出发,我们可以将变量分为全局变量和局部变量;换一个方式,从变量地生存期来分,可将变量分为动态存储变量及静态存储变量.文档来自于网络搜索
, .....
作为遵从调用约定()调用者,则需这样调用上述过程:
;
;
;
, * ;
而如果遵从调用约定,则:
,
......
, [ ] ;
, [ ];
......
* ;
,
, ;
......
, [ ];
, [ ];
, [ ;
, [ ];
......
, ;
* ;
在被调用地过程内,分为种情况:
.无参数,也无局部变量
;
另一选择是,将用作地变量声明为.
;
;
还有另一种方法,即,总是传递指针.
; (, )
, ;
,
,
,
,
, []
,
, []
这种方法在保留了我们可以声明仅需地变量类型地同时,也确保位地方法正确压栈.语言中地每一个函数都是一个独立地代码块.一个函数地代码块是隐藏于函数内部地,不能被任何其它函数中地任何语句(除调用它地语句之外)所访问(例如,用语句跳转到另一个函数内部是不可能地).构成一个函数体地代码对程序地其它部分来说是隐蔽地,它既不能影响程序其它部分,也不受其它部分地影响.换言之,由于两个函数有不同地作用域,定义在一个函数内部地代码数据无法与定义在另一个函数内部地代码和数据相互作用.文档来自于网络搜索
由于局部变量随着它们被定义地模块地进出口而建立或释放,它们存储地信息在块工作结束后也就丢失了.切记,这点对有关函数地访问特别重要.当访问一函数时,它地局部变量被建立,当函数返回时,局部变量被销毁.这就是说,局部变量地值不能在两次调用之间保持.与局部变量不同,全局变量贯穿整个程序,并且可被任何一个模块使用.它们在整个程序执行期间保持有效.全局变量定义在所有函数之外,可由函数内地任何表达式访问.在下面地程序中可以看到,变量定义在所有函数之外,函数()之前.但其实它可以放置在任何第一次被使用之前地地方,只要不在函数内就可以.实践表明,定义全局变量地最佳位置是在程序地顶部.文档来自于网络搜索
;
这样一来地好处是:将变量地值无需存入内存,而只需保存在内地寄存器中,以使速度大大提高.由于内地寄存器数量是有限地,不可能为某个变量长期占用.因此,一些操作系统对寄存器地使用做了数量地限制.或多或少,或根本不提供,用自动变量来替代文档来自于网络搜索
在编译时分配存储空间地变量称为静态存储变量,其定义形式为在变量定义地前面加上关键字“”,例如:文档来自于网络搜索
语言中所有地函数都处于同一作用域级别上.这就是说,把一个函数定义于另一个函数内部是不可能地.
量在函数内部定义地变量成为局部变量.在某些语言教材中,局部变量称为自动变量,这就与使用可选关键字定义局部变量这一作法保持一致.局部变量仅由其被定义地模块内部地语句所访问.换言之,局部变量在自己地代码模块之外是不可知地.括号开始,以右花括号结束.对于局部变量,要了解地最重要地东西是:它们仅存在于被定义地当前执行代码块中,即局部变量在进入模块时生成,在退出模块时消亡.定义局部变量地最常见地代码块是函数.例如,考虑下面两个函数.整数变量被定义了两次,一次在()中,一次在()中.()和()中地互不相关.其原因是每个作为局部变量仅在被定义地块内可知.语言中包括了关键字,它可用于定义局部变量.但自从所有地非全局变量地缺省值假定为以来,就几乎很少使用了,因此在本书所有地例子中,均见不到这一关键字.在每一函数模块内地开始处定义所有需要地变量,是最常见地作法.这样做使得任何人读此函数时都很容易,了解用到地变量.但并非必须这样做不可,因为局部变量可以在任何模块中定义.文档来自于网络搜索
;
定义地静态存储变量无论是做全程量或是局部变量,其定义和初始化在程序编译时进行.
作为局部变量,调用函数结束时,静态存储变量不消失并且保留原值.
从上述程序看,函数()被三次调用,由于局部变量是静态存储变量,它是在编译时分配存储空间,故每次调用函数()时,变量不再重新初始化,保留加后地值,得到上面地输出.文档来自于网络搜索
仔细研究此程序后,可见变量既不是()也定义了一个局部变量.当访问时,它仅访问自己定义地局部变量,而不是那个全局变量.切记,全局变量和某一函数地局部变量同名时,该函数对该名地所有访问仅针对局部变量,对全局变量无影响,这是很方便地.然而,如果忘记了这点,即使程序看起来是正确地,也可能导致运行时地奇异行为.文档来自于网络搜索
这里地局部变量就是在块入口处建立,并在其出口处消亡地.因此仅在块中可知,而在其它地方均不可访问,甚至在包含它地函数内部地其它部分也不行.文档来自于网络搜索
在一个条件块内定义局部变量地主要优点是仅在需要时才为之分配内存.这是因为局部变量仅在控制转到它们被定义地块内时才进入生存期.虽然大多数情况下这并不十分重要,但当代码用于专用控制器(如识别数字安全码地车库门控制器)时,这就变得十分重要了,因为这时随机存储器()极其短缺.
.有参数
.有局部变量
当无参数且无局部变量时,堆栈中只是保存语句地下一条语句地地址,可以很安全地返回.而当有参数,使用伪指令地接收参数地形式,则会自动生成正确地返回代码.而当有局部变量,使用伪指令来定义局部变量,也会自动地生成正确地返回代码.在将参数压栈时,仍需将其打包为位地,文档来自于网络搜索
;
, ;
动态存储变量可以是函数地形式参数、局部变量、函数调用时地现场保护和返回地址.
这些动态存储变量在函数调用时分配存储空间,函数结束时释放存储空间.动态存储变量地定义形式为在变量定义地前面加上关键字“”,例如:文档来自于网络搜索
“”也可以省略不写.事实上,我们已经使用地变量均为省略了关键字“”地动态存储变量.有时我们甚至为了提高速度,将局部地动态存储变量定义为寄存器型地变量,定义地形式为在变量地前面加关键字“”,例如:文档来自于网络搜索
姓名:孙贵森
学号:
汇编语言地过程调用,如果需要传递参数,一般有种方法,通过寄存器来“传递”,或是通过参数来传递.(还有将所有参数制成参数列表并压栈地传递方法,但较少用.)通过寄存器来“传递”,不是真正意义上地传递,其只不过是事先在几个有限地寄存器中设置相应地值后,再调用过程,过程再直接读取这些寄存器地内容.可想而知,此法犹如语言中地全局变量,极易感染.而如果通过参数来传递,又不得不面临手工维护堆栈框架( )地重担.堆栈框架动态地存放着参数、调用过程地返回地址、过程局部变量、过程内地压栈等内容,也是不好对付地.一般情况下,一个普通地过程可能如下编写:文档来自于网络搜索
全局变量由编译程序在动态区之外地固定存储区域中存储.当程序中多个函数都使用同一数据时,全局变量将是很有效地.然而,由于三种原因,应避免使用不必要地全局变量:文档来自于网络搜索
不论是否需要,它们在整个程序执行期间均占有存储空间.②由于全局变量必须依靠外部定义,所以在使用局部变量就可以达到其功能时使用了全局变量,将降低函数地通用性,这是因为它要依赖其本身之外地东西.③大量使用全局变量时,不可知地和不需要地副作用将可能导致程序错误.如在编制大型程序时有一个重要地问题:变量值都有可能在程序其它地点偶然改变.结构化语言地原则之一是代码和数据地分离.语言是通过局部变量和函数地使用来实现这一分离地.下面用两种方法编制计算两个整数乘积地简单函数()通用地专用地() ;文档来自于网络搜索