关于C与汇编的混合编译
汇编语言与c语言混合编程
汇编语言与c语言混合编程在计算机科学领域,汇编语言与C语言是两种常用的编程语言。
汇编语言是一种低级语言,与特定的硬件架构紧密相关,它使用机器指令来直接操作计算机的内部组件。
而C语言则是一种高级语言,具有跨平台性,更易于编写和维护。
汇编语言的主要特点是直接操作硬件,对于性能要求较高的应用程序,使用汇编语言进行编程可以获得更好的效率。
然而,由于汇编语言的语法复杂且与具体硬件相关,编写汇编语言程序需要更深入地了解硬件结构和指令集。
相对而言,C语言的语法更加简洁易懂,对于一些应用程序而言,使用C语言进行编程更加方便快捷。
在一些特定的场景中,汇编语言与C语言混合编程可以充分发挥各自的优势,提高编程效率。
下面将介绍一些混合编程的常用技巧和应用。
一、函数接口的嵌入汇编代码在C语言程序中,使用汇编语言的部分代码可以提高程序的效率。
对于特定的计算任务,采用汇编语言实现可以避免一些C语言的运行时开销。
为了在C语言程序中嵌入汇编代码,可以使用内嵌汇编的方式。
内嵌汇编语言是一种特殊的语法,它允许将汇编代码直接嵌入到C语言程序中。
通过使用特定的关键字和语法规则,可以在C语言程序中编写汇编指令。
汇编指令的执行结果可以存储在C语言变量中,方便后续的处理和计算。
例如,在一个计算图像像素之和的函数中,可以使用汇编语言进行优化,提高计算的速度。
在C语言中,可以通过内嵌汇编语言嵌入与图像像素累加相关的汇编指令,以提高程序的运行效率。
二、调用汇编语言库函数汇编语言库是一组由汇编语言编写的函数的集合。
这些函数通常涉及到与硬件相关的底层操作,如访问特定设备、处理中断等。
在C 语言程序中,可以通过调用汇编语言库函数来实现与这些底层操作相关的功能。
调用汇编语言库函数可以充分发挥汇编语言的性能优势,同时借助C语言的高级特性,实现便捷的程序开发。
例如,在嵌入式系统开发中,可以使用汇编语言库函数操作硬件寄存器,控制外设的工作。
三、使用C语言接口封装汇编语言模块为了提高程序的可读性和可维护性,有时候需要使用C语言接口来封装汇编语言模块。
五、c语言与汇编语言混合编程
2、 C程序调用汇编程序
第一步:建立汇编源文件Factorial.s
/* Factorial.s */ AREA Fctrl, CODE, READONLY EXPORT Factorial Factorial MOV R8 , R0 MOV R9 , #0 SUB R0,R8,#1 Loop MOV R1 , R9 UMULL R8 , R9 , R0 , R8 MLA R9 , R1 , R0 , R9 SUBS R0 , R0 , #1 BNE Loop LDR R0,=0xFFFFFFF0 STMIA R0,{R8,R9} MOV PC,LR MOV PC,LR ; 声明代码段 Fctrl
在ARM开发工具编译环境下设计程序,用ARM 汇编语言调用C语言实现20!的阶乘操作,并将 结果保存到寄存器R0中。
4ห้องสมุดไป่ตู้
1、汇编程序调用C程序
然后建立C语言源文件factorial.c
/* factorial.c */ long long Factorial(char N) { char i; long long Nx=1; for(i=1;i<=N;i++)Nx=Nx*i; return Nx; }
初始化过程
初始化存储器及 SDRAM 初始化堆栈
设置异常向量 初始化C语言运行环 境:设置参数 改变CPU运行模式为用 户模式(User Mode) 运行C程序
15
//内嵌汇编标识
13
本章提要
一 ARM微处理器体系结构 二 ARM 微处理器指令集 三 C语言与汇编语言的混合编程 四 ARM处理器初始化分析
14
应用示例-系统初始化
复位动作
汇编与C的混合编程
.CODE Move PROC NEAR32 c uses eax ebx, countp:word, Sr:byte , Dst:byte, Spa: byte cmp countp , 1 ; NbrDisks = 1? jne elseMore ; skip if more than 1 mov bx,word ptr Sr ; Source mov source,bl ; copy character to output mov bx,word ptr Dst ; destination mov dest,bl ; copy character to output output message ; print line jmp endIfOne ; return elseMore: dec countp INVOKE Move, countp, Sr, Spa, Dst INVOKE Move, 1, Sr, Dst, Spa INVOKE Move, countp, Spa, Dst, Sr endIfOne: ret ; return Move ENDP
若干注意事项
操作码支持8086/8087指令或若干伪指令:db/dw/dd和extern 操作数是操作码可接受的数据:立即数、寄存器名,还可以是C语言程序中的常量、变量和标号等 内嵌的汇编语句可以用分号“;”结束,也可以用换行符结束 使用C的注释,如 / * … * / 正确运用通用寄存器、标号等
start: output prompt ; ask for number of disks input number,16 ; read ASCII characters atoi number ; convert to integer INVOKE Move, ax, 'A', 'B', 'C' INVOKE ExitProcess, 0 ; exit with return code 0 PUBLIC start ; make entry point public END ; end of source code
C语言与汇编语言混合编程
link /subsystem:windows ms.obj mc.obj //这里使用32位的链接器,要设好lib路径 现在得到ms.obj mc.obj ms.exe 三个文件
3、分析一下源代码,显然程序入口点是_start(在使用/coff参数进行编译时必须有下划线),在汇编中 调用了C中的m函数,这是需要重定位的。在C中m调用了sum函数,这也是要重定位的。
end _start
extern void m(){
int d;
long c;
c=1;
2、分别编译成obj文件
ml /c /coff ms.asm //指定生成coff格式的obj文件
cl /c /Fomc.obj mc2.c //指定生成的obj文件名为mc.obj
4、现在利用VC6自带的dumpbin.exe工具,生成解析文件:
dumpbin /all ms.obj>msobj.txt
dumpbin /all mc.obj>mcobj.txt
dumpbin /all ms.exe&a s;
mov ax,cs }
mov ds,ax
call m int e = 1;
stop: int f = 2;
jmp stop int ar[6000000L];
.model flat int k = i;
extrn c m:near int j = 0;
public _start int s = 0;
.code for(j=1;j<=k;j++) s+=j;
C与汇编混合语言编程
1、先准备两个程序,一个汇编、一个C语言
汇编语言与C语言混合编程技术
汇编语⾔与C语⾔混合编程技术汇编语⾔与C语⾔混合编程技术 ARM体系结构⽀持ARM的汇编语⾔与C与C++的混合编程。
在⼀个完整的程序设计的中,除了初始化部分⽤汇编语⾔完成外,其⼤部分的编程任务⼀般都⽤C或C++完成。
汇编程序中访问C程序变量1 在汇编的源程序中调⽤C语⾔风格的字符串需要使⽤IMPORT伪操作。
IMPORT相当于C语⾔中的extern关键字,告诉编译器引⽤的符号不是在本⽂件中定义的,⽽是在其他的源⽂件中定义的。
伪操作的格式: IMPORT symbol [,WEAK] symbol是声明的符号的名称; [,WEAK]指⽰编译器如果发现symbol在所有的源⽂件中都没有找到,那么它也不会产⽣任何的错误信息。
⽰例见下页。
汇编程序中访问C程序变量2 C语⾔代码⽂件str.c⾥⾯只有⼀个简单的字符串的定义: char *strhello="Hello world!\n\0"; 汇编代码⽂件hello.s 1 AREA ||.text||, CODE, READONLY 2 main PROC 3 STMFD sp!,{lr} 4 LDR r0,=strtemp 5 LDR r0,[r0] 6 BL _printf 7 LDMFD sp!,{pc} 8 strtemp 9 DCD strhello 10 ENDP 11 EXPORT main 12 IMPORT strhello 13 IMPORT _main 14 IMPORT _main 15 IMPORT _printf 16 IMPORT ||Lib$$Request$$armlib||, WEAK 17 END C程序中内嵌汇编指令1 在ARM的C语⾔程序中可以使⽤关键字__asm来加⼊⼀段汇编语⾔的程序。
格式: __asm { instruction/*comment*/ … } C程序中内嵌汇编指令2 在C语⾔中嵌⼊的ARM汇编需要注意⼀些问题:在汇编指令中,可以使⽤表达式,使⽤逗号“,”作为分隔符 ; 如果⼀条指令占⽤了多⾏,那么应该使⽤符号“\”续⾏,如果⼀⾏中有多个汇编指令,那么应该使⽤“;”将多个指令隔开。
汇编语言与C的混合编程
1.3常量
• 在内嵌的汇编指令中,常量前的符号#可以省略。如果在一个表达式 前使用#,则该表达式必须是一个#。 • 1.4指令展开 • 内嵌的汇编指令中如果包含常量操作数,则该指令可能会被汇编器展 开成几条指令。例如指令: • ADD R0,R0,#1023 • 可能会被展开成下面的指令序列 • ADD R0,R0,#1024 • SUB R0,R0,#01 • 乘法指令MUL可能会被展开成一系列的加法操作和移位操作。 • 事实上,除了与协处理器相关的指令外,大部分的ARM指令和 Thumb指令中包含常量操作数都可能被展开成多条指令。
1.2物理寄存器
• 在内嵌的汇编指令中使用物理寄存器有一下限制: • 不能直接向PC寄存器中赋值,程序的跳转只能通过B指令和BL指令实 现。 • 在使用物理寄存器的内嵌汇编指令中,不要使用过于复杂的C表达式, 因为当表达式过于复杂时,将会需要较多的物理寄存器,这些寄存器 可能与指令中的物理寄存器的使用冲突。 • 编译器可能会使用R1寄存器或者R13寄存器存放编译的中间结果,在 计算表达式值时可能会将寄存器R0~R3、R2以及R14用于子程序的调 用。因此在内嵌的汇编指令中,不要将这些寄存器同时指定为指令中 的物理寄存器。 • 在内嵌的汇编指令中使用物理寄存器时,如果有C变量使用了该物理 寄存器,编译器将在合适的时候保存并恢复该变量的值。需要注意的 是,当寄存器sp、sl、fp以及sb用做特定的用途时,编译器不能恢复 这些寄存器的值。 • 通常推荐在内嵌的汇编指令中不要指定物理寄存器,因为这可能会影 响编译器分配寄存器,进而可能影响代码的效率。
• 下面是一个在汇编程序中访问C程序全局变量的例子。程序中变量 globv1是在C程序中声明的全局变量。在汇编程序中首先使用 IMPORT伪操作声明该变量;R0中;修改后再将寄存器R0的值赋予变量 globv1.本例中程序如下: • AREA globals,CODE,READONLY • EXPORT asmsub • IMPORT global ;用IMPORT伪指令声明该变量 • asmsum • LDR R1,# globv1 ;将其内存地址读入到寄存器R1中 • LDR R0,[R1] ;再将其值读入到寄存器R0中 • ADD R0,R0,#2 • STR R0,[R1] ;修改后再将寄存器R0的值赋予变量globv1 • MOV PC,LR • END
汇编语言与c语言混合编程
汇编语言与c语言混合编程汇编语言与C语言混合编程是一种程序设计方法,它将汇编语言和C 语言结合起来使用,以实现更高效、更灵活的程序设计。
在这种编程方法中,汇编语言主要用于实现底层操作,而C语言则用于实现高层逻辑。
下面将详细介绍汇编语言与C语言混合编程的基本原理、优点及注意事项。
一、基本原理在汇编语言与C语言混合编程中,通常采用的方法是通过汇编代码调用C函数或者将C函数嵌入到汇编代码中。
具体来说,可以使用以下两种方式:1. 汇编代码调用C函数在这种方式下,首先需要定义一个extern类型的函数声明,并在需要调用该函数的地方使用call指令进行调用。
例如:extern int add(int a, int b);...mov eax, 1mov ebx, 2call add...其中add是一个C函数,在汇编代码中通过extern关键字进行声明,并使用call指令进行调用。
2. 将C函数嵌入到汇编代码中在这种方式下,可以使用asm关键字将一段汇编代码嵌入到C函数中。
例如:int main(){int a = 1;int b = 2;asm("mov eax, %0\n""add eax, %1\n""mov %0, eax\n": "+r" (a): "r" (b): "eax");return a;}在这个例子中,使用asm关键字将一段汇编代码嵌入到了main函数中。
该汇编代码实现了对a和b的相加操作,并将结果保存在a中。
二、优点汇编语言与C语言混合编程具有以下几个优点:1. 提高程序效率汇编语言可以直接操作硬件,因此在需要进行底层操作或者对性能要求较高的场合下,使用汇编语言可以提高程序的效率。
2. 增强程序灵活性C语言具有良好的可移植性和易读性,而汇编语言则具有更高的灵活性。
通过将两者结合起来使用,既可以充分利用C语言的优点,又可以发挥汇编语言的灵活性,从而增强程序的灵活性。
vscode c和汇编混合编译
vscode c和汇编混合编译1. 简介在软件开发过程中,我们经常需要使用多种编程语言来实现复杂的功能。
在某些情况下,我们可能需要将C语言和汇编语言混合编译,以充分利用汇编语言的高效性能和C语言的便捷性。
本文将介绍如何在VSCode中进行C和汇编混合编译的步骤和技巧。
2. 准备工作在开始混合编译之前,我们需要准备一些工具和环境:•VSCode:这是一个强大的开源代码编辑器,支持多种编程语言和插件扩展。
•C编译器:在Windows系统上,我们可以使用MinGW或者MSVC作为C语言的编译器;在Linux系统上,我们可以使用GCC。
•汇编编译器:通常情况下,C语言的编译器会集成汇编编译器。
在Windows 系统上,我们可以使用NASM或者MASM;在Linux系统上,我们可以使用GCC。
•Make工具:Make是一个用于自动化编译的工具,可以根据源文件的依赖关系自动编译生成目标文件。
确保以上工具和环境已经正确安装并配置好。
3. 创建项目首先,我们需要在VSCode中创建一个工作目录来存放我们的项目文件。
打开VSCode,点击菜单栏中的“文件”->“打开文件夹”,选择一个合适的位置来创建工作目录。
在工作目录中,我们需要创建以下文件:•main.c:C语言源文件,包含我们的主要代码。
•asm.asm:汇编语言源文件,用于实现一些高效的底层操作。
•Makefile:用于自动化编译的Makefile文件。
4. 编写C代码在main.c文件中,我们可以编写C语言的代码。
下面是一个简单的示例:#include <stdio.h>extern void asm_func();int main() {printf("Hello, World!\n");asm_func();return 0;}在这段代码中,我们通过extern关键字引入了一个名为asm_func的外部函数。
这个函数在后面的汇编语言源文件中实现。
汇编语言和C语言的混合编程
2. 在汇编中使用C定义的全局变量
#include <stdio.h> int gVar_1 = 12; extern asmDouble(void);
int main() { printf("original value of gVar_1 is: %d", gVar_1); asmDouble(); printf(" modified value of gVar_1 is: %d", gVar_1); return 0; }
内嵌汇编使用的标记是 __asm,用法如下: __asm { instruction [; instruction] … [instruction] }
void enable_IRQ(void) { void disable_IRQ(void) {
int tmp;
__asm
int tmp;
__asm
PRESERVE8 AREA Scopy ,CODE,READONLY EXPORT strcopy strcopy LDRB R2,[R1],#1 STRB R2,[R0],#1 CMP R2,0 BNE strcopy MOV pc,lr END
在汇编中调用C的函数
• 在汇编中调用C的函数,需要在汇编中IMPORT 对应 的C函数名,使用BL指令调用
return a + b + c;
}
int g( int a, int b, int c, int d, int e, int f ) { printf(“e=%d\n”, e); printf(“f=%d\n”, f); return (a+b+c+d+e+f); }
10 C语言与汇编语言的混合编程
3.混合项目(续)
模块3(文件名pulse3.asm): PUBLIC _DELAY1MS;用PUBLIC声明_DELAY1MS为其它函数调用,以 “_”为前缀的函数名,表示该函数通过寄存器传递参数 DELAY SEGMENT CODE;定义DELAY段为再定位程序段 ,DELAY是用 户任意起的段名 RSEG DELAY ;选择DELAY为当前段 _DELAY1MS: DELA:MOV R6,#12 ;2T LOP1:MOV R5,#250 ;2T DJNZ R5,$ ;4T DJNZ R6,LOP1 ;4T DJNZ R7,DELA ;R7中数值为C ;程序传递过来的参数,参数为2则延时2ms EXIT: RET END
1.混合编程的实现(续)
第二步:在Keil C51环境下,在Project窗口中包含汇编代码 的C文件上单击右键,在弹出的快捷菜单中选择“Options for…”命令,再点击选中右边的“Generate Assembler SRC File”和“Assembler SRC File”复选框,使检查框由灰色(无 效)变成黑色(有效)状态。
12
3.混合编程
模块1(文件名pulse1.c): #include<STC12C5A.h> #define uchar unsigned char sbit P1_1=P1^1; extern void delay4ms(void); //外部函数delay4ms()声明 void main() { while(1) { P1_1=0; delay4ms (); //调用模块2延时4ms P1_1=1; delay4ms (); //调用模块2延时4ms } }
4
1.混合编程的实现(续)
5
gcc 汇编和c混用
gcc 汇编和c混用gcc是一种常用的编译器,可以将C语言源代码编译成可执行的机器代码。
而汇编语言则是一种低级语言,与机器代码接近,可以直接操作硬件。
在开发过程中,有时候需要在C语言中嵌入汇编代码,以实现一些高级功能或优化性能。
在使用gcc和汇编语言混用时,主要有两种方式:内联汇编和外部汇编。
内联汇编是将汇编代码直接嵌入到C语言源代码中,而外部汇编则是将汇编代码放在单独的文件中,通过链接器和编译器进行连接。
内联汇编是一种简单方便的方式,可以直接在C语言函数中嵌入汇编代码,使用特定的语法进行标识。
例如,在C语言中使用内联汇编实现一个简单的加法运算:```cint add(int a, int b) {int result;asm("addl %%ebx, %%eax": "=a" (result): "a" (a), "b" (b));return result;}```在上面的例子中,使用了asm关键字来标识内联汇编代码,使用了占位符%%eax和%%ebx来表示寄存器,通过输入输出约束来传递参数和获取结果。
除了内联汇编,我们还可以使用外部汇编文件来与C语言混用。
首先,我们需要编写一个汇编文件,例如sum.asm,实现一个求和函数:```asmsection .textglobal sumsum:mov eax, DWORD [ebp + 8]add eax, DWORD [ebp + 12]mov DWORD [ebp + 8], eaxmov eax, DWORD [ebp + 8]ret```在上面的汇编代码中,使用了ebp寄存器来获取函数参数,并将求和结果存放到ebp+8的内存地址中。
接下来,我们需要在C语言中声明这个外部函数,并在链接时引入汇编代码:```cextern int sum(int a, int b);int main() {int a = 1;int b = 2;int result = sum(a, b);printf("The sum of %d and %d is %d\n", a, b, result);return 0;}```在上面的例子中,使用extern关键字来声明外部函数sum,并在main函数中调用该函数。
C语言和汇编语言混合编程
C/C++ 语言和汇编语言混合编程开发为了充分利用DSP芯片的硬件资源,更好发挥C/C++ 语言和汇编语言进行软件开发的各自优点,可以将两者有机的结合起来,兼顾两者优点,避免其弊端。
因此,在很多情况下,采用混合编程方法能更好地达到设计要求,完成设计任务。
C/C++语言和汇编语言混合编程的具体方法有以下几种:(1)独立编写C/C++程序和汇编程序,分开编译或汇编形成各自的目标模块,再用链接器将C/C++模块和汇编模块链接起来,这是一种灵活性较大的方法。
但用户必须自己维护各汇编模块的入口和出口代码,自己计算传递参数在堆栈中的偏移量,工作量稍大,但能做到对程序的绝对控制,也能满足软件设计结构化的要求。
这是主要讲述的方法。
(2)在C/C++程序中使用汇编程序中定义的变量和常量。
(3)在C/C++程序中直接内嵌汇编语句。
这种方法可以在C/C++程序中实现C/C++语言无法实现的硬件控制功能,如修改中断控制寄存器、中断标志寄存器等。
(4)在C/C++源程序中使用内部函数直接调用汇编语言语句。
后3种方法由于在C/C++语言中直接嵌入了汇编语言的成分,容易造成程序混乱,C/C++环境被破坏,甚至导致程序崩溃,而编程者又很难对不良结果进行预期和有效控制。
而如果采用第一种方法,只要遵循有关C/C++语言函数调用规则和寄存器规则,就能预见到程序运行的结果,保证程序正确。
下面分别讲述函数调用规则和寄存器规则。
1.函数调用规则C/C++编译器对函数调用强加了一组严格的原则。
除了特殊的运行时间支持库函数外,任何调用函数和被C/C++函数调用的函数都必须遵守这些原则。
不遵守这些原则可能破坏C/C++环境并导致程序失败。
图1说明了典型的函数调用。
在这个例子中,参数被传递到堆栈中调用者的参数块,函数再使用这些参数调用被调用函数。
注意,第一个参数是在A累加器中传递的。
这个例子还说明了汇编器对被调用函数的局部帧的分配。
局部帧包括局部变量块和局部参数块两部分,其中局部参数块是局部帧中用来传递参数到其他函数的部分。
第6章 C语言和汇编语言混合编程(C55x DSP)
第6章 C语言和汇编语言混合编程
2. 编译模式 使用C编译器,在进入汇编程序时,C55x的CPL(编译模 式位)自动被置1,相对寻址模式使用堆栈指针SP。如果在汇 编程序中需要使用相对直接寻址模式访问数据存储器,则必 须改成数据页DP直接寻址模式,这可以通过清CPL位实现。 在返回C调用程序前,CPL位必须重新置1。
序清单输入到这些文件当中,保存并将其添加到工程之中。
第6章 C语言和汇编语言混合编程
3. 编译链接工程和装载输出文件 编译链接工程,修改错误,直到无错误信息提示为止, 再装载输出文件。 4. 打开CPU寄存器视窗 在主菜单中选择View→Registers→CPU Registers命令, 打开CPU寄存器视窗,观察CPU寄存器的变化情况。
int s4; s4=sum(); s1=mac1(x,a,n); //*ar0=x[0],*ar1=a[0],t0=n,return is stored in t0
s2=mac2(x,a,n);
//*ar0=x[0],*ar1=a[0],t0=n,return is stored in ac0
第6章 C语言和汇编语言混合编程
“ _ ”。以下是C函数调用汇编子程序的例子。
第6章 C语言和汇编语言混合编程
//C源程序: extern int sum(int *); //参考一个汇编函数 int x[4]={0x1223,0x345,0x2345,0x3444}; //定义全局变量并初始化
int s;
void main() {
参数类型和寄存器安排顺序表。
第6章 C语言和汇编语言混合编程
表6-1 参数类型和寄存器安排顺序表
第6章 C语言和汇编语言混合编程
汇编语言和C语言的简单混合编程
ASM JB COPY
/*转移到C的标号*/
ASM CMP A1,’Z’
ASM JA COPY /
/*不是’A’到’Z’之间的字符原样复制*
ASM ADD A1,20H
/*是小写字母转换成大写字母*/
copy:ASM STOSB
/* C语言定义的标号*/
ASM CMP AL,0
/* C语言中字符串用0结尾 */
12
注意:直接使用Turbo C的连接程序TLINK进行连接时,用户必须指定要连接 的与存储模式一致的初始化模块和函数库文件,并且初始化模块必须是第 一个文件。上例中,Lib\c0m和Lib\cm就是在Lib目录下中型存储模式的 初始化模块c0m.obj和函数库cm.lib。 如果形成的可执行文件exampl. exe正确,它的运行结果将是:
6
}
编辑完成后,假定该文件名为,在命令行输人如下 编译命令(选项-I和-L分别指定头文件和库函数的 所在目录):
TCC –B –Iinclude –Llib example.c 生成可执行文件example.exe,程序运行后输出的结
果将是:
OLD STRING IS I’AM a good STUDENT!
/*GOOD函数返回值缺省为INT型*/
EXTERN char tempvar
/*TEMPVAR变量为CHAR型*/
经说明后,这些外部变量、过程、函数可在C程序中直接使用,函数的参数在传 递过程中要求参数个数、类型、顺序要一一对应。
和纯汇编语言多模块编程要求一样,汇编语言程序的标识符(子程序名和变量名)
以用换行符结束;一行中可以有多个汇编语句,相互间用分号分隔, 但不能跨行书写。嵌入汇编语句的分号不是注释的开始;要对语句注 释,应使用C语言的注释,如/*……*/。例如:
c语言和汇编语言混合编程
c语言和汇编语言混合编程一、混合编程的概念混合编程是指在一个项目中同时使用C语言和汇编语言来进行编程的方式。
C语言作为高级语言,具有易读易写的特点,适合用来进行程序的框架设计和算法实现;而汇编语言作为底层语言,可以直接操作计算机的硬件资源,提供了更高的灵活性和效率。
二、混合编程的使用场景1.优化性能:C语言的执行效率相对较高,但在某些特殊的场景下,需要通过使用汇编语言来对关键代码进行优化,以提升程序的性能。
2.访问底层硬件资源:汇编语言可以直接对计算机的硬件资源进行操作,如访问特定的芯片、寄存器等,从而实现对硬件资源的更直接的控制。
3.特殊算法实现:某些算法的实现需要使用到特定的汇编指令集,通过混合编程的方式可以在C语言的框架下使用汇编语言来实现这些特殊算法。
4.系统级编程:在操作系统、驱动程序等领域,混合编程可以更好地利用C语言的高级特性,同时结合汇编语言的底层操作能力,实现对系统资源的更加细致控制。
三、混合编程的优势1.灵活性:混合编程可以充分发挥C语言的灵活性,利用其高级特性来进行程序的框架设计和算法实现,同时结合汇编语言的底层操作能力,实现对硬件资源的更加细致控制。
2.效率:通过使用汇编语言对关键代码进行优化,可以提高程序的执行效率,尤其在对计算量较大的算法进行优化时,混合编程可以发挥出更大的优势。
3.可移植性:C语言作为一种通用的高级语言,具有良好的可移植性,而汇编语言的编写通常与具体的硬件平台相关,因此,在混合编程中,可以通过将汇编代码与C代码分离,实现对不同平台的兼容性。
四、混合编程的实际应用案例1.图像处理:在图像处理领域,混合编程可以利用C语言的框架设计和算法实现特定的图像处理算法,同时通过汇编语言来实现对图像像素的高效访问和处理。
2.密码学:在密码学领域,混合编程可以结合C语言的高级特性和汇编语言的底层操作能力,实现对密码算法的高效实现。
3.嵌入式系统开发:在嵌入式系统开发中,混合编程可以通过使用C语言进行系统框架设计和驱动程序开发,同时利用汇编语言来对底层硬件资源进行直接访问和控制,提高系统的性能和稳定性。
第六讲_C与汇编语言混合编程
数据栈的使用规则
•根据堆栈指针指向位置的不同 和增长方向的不同 可以分为以下4种数据栈 : FD (Full Descending) 满递减 ED (Empty Descending)空递减 FA (Full Ascending) 满递增 EA (Empty Ascending) 空递增 ATPCS规定数据栈为FD(满递减)类型,并且 对数据栈的操作是8字节对齐的。
内嵌汇编注意事项
必须小心使用物理寄存器,如R0~R3,LR和PC。 不要使用寄存器寻址变量。 使用内嵌汇编时,编译器自己会保存和恢复它可能用 到的寄存器,用户无须保存和恢复寄存器。 LDM和STM指令的寄存器列表只允许物理寄存器。 汇编语言用“,”作为操作数分隔符
三、C和ARM汇编程序间相互调用
内嵌汇编指令的语法格式
• __asm(“指令[;指令]”); • ARM C汇编器使用关键字“__asm"。如果有多 条汇编指令需要嵌入,可以用“{}”将它们 归为一条语句。如: • __asm •{ • 指令[;指令] • … • [指令] •} • 需要特别注意的是__asm是两个下划线。
内嵌的汇编指令的特点
在 C 和 ARM 汇编程序之间相互调用必须遵守 ATPCS (ARM-Thumb Procedure Call Standard)规则。 C和汇编之间的相互调用可以从以下这程序中调用C语言程序
汇编程序访问全局C变量
C函数原型: int g(int a,int b,int c,int d,int e) { return a+b+c+d+e; } ////汇编程序调用C程序g()计算5个整数i, 2*i, 3*i, 4*i, 5*i 的和。 汇编源程序: EXPORT f AREA f,CODE,READONLY IMPORT g ;声明该变量函数g( ),i在R0中 STR LR,[SP,#- 4]! ;预先保存LR ADD R1,R0,R0 ;计算2 * i(第2个参数) ADD R2,R1,R0 ;计算3*i(第3个参数) ADD R3,R1,R2 ;计算5*i(第5个参数) STR R3,[SP,#- 4]! ;将第5个参数压人堆栈 ADD R3,R1,R1 ;计算4 * i(第4个参数) BL g ;调用C程序g( ) ADD SP,SP,#4 ;调整数据栈指针,准备返回 LDR PC,[SP],#4 ;从子程序返回 END
汇编语言与C的混合编程
Turbo C 2.0在处理汇编语句时要调用TASM.EXE。如果没有 TASM汇编器,可以用Microsoft公司的MASM.EXE来替代。具 体的方法有3种: (1)用pctools或debug在tcc.exe中查找到TASM,并替 换成MASM。 (2)把MASM.EXE拷贝或改名为TASM.EXE,放到Turbo C 子目录中。 (3)在命令行输入以下内容:tcc –b –e /path/masm/exec语言源文件名,其中,-b是使用汇编开 关,-e是使用另一个汇编器开关,path是MASM或其他的汇 编器所在的目录路径。
8.1.1 嵌入汇编语句格式 在Turbo C中,C程序中嵌入汇编语言语句必须以关键 字ASM开头,其格式如下: ASM <操作码> <操作数> <;或回车换行>/*注释*/ 在使用此种格式时,应注意以下几点: (1)ASM作为关键字不能省略,带有ASM标识的汇 编语言代码可以看成是C语言的部分程序代码。当C 语言编译器遇到ASM语句时会识别并自动调用汇编语 言编译程序,将它翻译为机器码再嵌入到C语言程序 之中。 (2)操作码可以是处理器指令(如PUSH、MOV等), 也可以是伪指令(如DB、DW、EXTERN等)。
(5)嵌入的汇编语句的注释方式必须采用C语言的 注释方式,即必须用“/*”标识注释的开始,用 “*/”来标识注释的结束,绝不能像纯汇编那样使 用“;”来作为一条注释的开始。 (6)如果要在C语言程序中嵌入连续多条汇编语句, 可以在每行前面都加上“ASM”关键字,更简单的 办法是输入一个ASM关键字后,使用括号"{"和"}"将 这些汇编语句括起来。
本章要点
混合编程是指使用两种或两种以上的程序设计 语言,通过相互调用、参数传递、共享数据结 构和数据信息而形成程序的过程。采用高级语 言与汇编语言混合编程,从而充分利用各种程 序设计语言各自的优势,即程序的大部分采用 高级语言编写,以提高程序的开发效率;在某 些部分利用汇编语言编写,以提高程序的运行 效率。 Turbo C嵌入汇编方式 Turbo C模块连接方式 汇编语言在Visual C++中的应用
C与汇编的混编
//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;
?_AFUNC?BIT:
v_bflag?041: DBIT 1
; uchar tmp_vchr;
; uint tp_vint;
;
; tmp_vchr=v_achr;
; SOURCE LINE # 10
;---- Variable 'tmp_vchr?042' assigned to Register 'R5' ----
{
bit BFLAG;
uchar mav_chr;
uint mvintrslt;
mav_chr=0xd4; BFLAG=1;
mvintrslt=AFUNC(mav_chr,BFLAG);
}
//a51FUNC.c
#define uchar unsigned char
; return tmp_vchr+(tp_vint<<8);
; SOURCE LINE # 12
MOV R6,A
MOV R4,#00H
CLR A
ADD A,R5
MOV R7,A
tmp_vchr=v_achr;
tp_vint=(uint)v_bflag;
return CFUNC(tp_vint); //这里调用一个c51函数
C及汇编语言的混合编程
1
第五章 C/C++及汇编语言旳混合编程
5.1 ARM C/C++编译器 5.2 在C/C++程序中内嵌汇编指令 5.3 从汇编程序中访问C程序变量 5.4 汇编程序、C程序及C++程序相互调用 5.5 嵌入式C编程
2
5.1 ARM C/C++编译器
ARM集成开发环境中包括旳C/C++编译器。
R0就能够处理这个问题: __asm { MOV var,x ADD y,var,x/y }
内嵌汇编器探测到隐含旳寄存器冲突就会报错.
5
5.2 在C/C++程序中内嵌汇编指令
在C/C++程序中内嵌汇编指令注意事项: 不要使用寄存器替代变量.尽管有时寄存器明显相应某个变量,但也不能直接使用寄存器替代变量.
数,当参数超出4个时,还能够使用数据栈来传递参数。 2. 成果为一种32位整数时,能够经过寄存器R0返回 3. 成果为一种64位整数时,能够经过寄存器R0和R1返
回,依次类推。
10
5.4 汇编程序、C程序及C++程序相互调用
C 程序调用汇编程序: 汇编程序旳设置要遵照ATPCS 规则,确保
程序调用时参数旳正确传递。 在汇编程序中使用EXPORT 伪指令申明本
ADD R1,R0,R0 ;设置sum5 函数入口参数,R0 为参数a
ADD R2,R1,R0 ;R1 为参数b,R2 为参数c
ADD R3,R1,R2,
STR R3,[SP,# -4]! ;参数e 要经过堆栈传递
ADD R3,R1,R1
;R3 为参数d
BL sum5
C汇编的混合编程
C汇编的混合编程C和汇编的混合编程优势C和汇编很容易的混合可实现在C中⽆法实现的处理器功能使⽤新的或不⽀持的指令产⽣更⾼效的代码直接链接变量和程序确定符合程序调⽤规范输⼊/输出相关的符号编译器也可保留内嵌汇编⼤多数arm指令都可实现内嵌汇编代码可由编译器的优化器来传递ATPCS(arm/thumb程序调⽤规范)ATPCS即ARM-THUMB procedure call standard的简称。
PCS规定了应⽤程序的函数可以如何分开地写,分开地编译,最后将它们连接在⼀起,所以它实际上定义了⼀套有关过程(函数)调⽤者与被调⽤者之间的协议。
PCS强制实现如下约定:调⽤函数如何传递参数(即压栈⽅法,以何种⽅式存放参数),被调⽤函数如何获取参数,以何种⽅式传递函数返回值。
ATPCS规定了⼀些⼦程序之间调⽤的基本规则。
这些基本规则包括⼦程序调⽤过程中寄存器的使⽤规则,数据栈的使⽤规则,参数的传递规则。
为适应⼀些特定的需要,对这些基本的调⽤规则进⾏⼀些修改得到⼏种不同的⼦程序调⽤规则,这些特定的调⽤规则包括:⽀持数据栈限制检查的ATPCS。
,⽀持只读段位置⽆关的ATPCS,⽀持可读写段位置⽆关的ATPCS,⽀持ARM程序和THUMB程序混合使⽤的ATPCS,处理浮点运算的ATPCS等。
有调⽤关系的所有⼦程序必须遵守同⼀种ATPCS。
编译器或者汇编器在ELF格式的⽬标⽂件中设置相应的属性,标识⽤户选定的ATPCS类型。
对应不同类型的ATPCS规则,有相应的C语⾔库,连接器根据⽤户指定的ATPCS类型连接相应的C语⾔库。
使⽤ADS(ADS全称为ARM Developer Suite)的C语⾔编译器编译的C语⾔⼦程序满⾜⽤户指定的ATPCS类型。
⽽对于汇编语⾔程序来说,完全要依赖⽤户来保证各⼦程序满⾜选定的ATPCS类型。
具体来说,汇编语⾔⼦程序必须满⾜下⾯三个条件:在⼦程序编写时必须遵守相应的ATPCS规则;数据栈的使⽤要遵守ATPCS规则;在汇编编译器中使⽤-apcs选项。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
请教:关于C51与汇编地混合编译作者:leonyang栏目:单片机请教:关于C51与汇编地混合编译在下第一次用C51,计划用汇编写底层,用C51写应用层.但是不知道C51跟汇编之间地结构应该是怎样地,才能顺利编译.汇编地程序是不是跟C51地分别在不同地文件还是同一文件下,两种我都试过,没有成.一下就是小弟地一段实验代码,请高人指点一二,不胜感激!#define uint unsigned int#include<reg51.h>extern void getval(uint i);void main(void){uint data i=10;uint data j=0;getval(i);}PUBLIC _getvalgetvalp SEGMENT CODERSEG getvalp_getval: NOPMOV A,R6MOV B,R7RETEND报告有错如下:error C129: missing ';'before '_getval'请英雄们赐教,谢谢了!2楼:>>参与讨论作者:gaozt于 2005-5-3 17:16:00 发布:不能这样简单地混合吧!看一下keil里地帮助文件!3楼:>>参与讨论作者:leonyang于 2005-5-5 22:17:00 发布:step1. 按写普通c51程序方法,建立工程,在里面导入main.c文件和CFUNC.c文件.相关文件如下://main.c文件#include < reg51.h >#define u CHAR unsigned CHAR#define uint unsigned intextern uint AFUNC(u CHAR v_achr,bit v_bflag);void main(){bit BFLAG;u CHAR mav_chr;uint mvintrslt;mav_chr=0xd4; BFLAG=1;mvintrslt=AFUNC(mav_chr,BFLAG);}//CFUNC.c文件#define u CHAR unsigned CHAR#define uint unsigned intuint AFUNC(u CHAR v_achr,bit v_bflag){u CHAR 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.cNAME CFUNC?PR?_AFUNC?CFUNC SEGMENT CODE?BI?_AFUNC?CFUNC SEGMENT BIT OVERLAYABLEPUBLIC ?_AFUNC?BITPUBLIC _AFUNCRSEG ?BI?_AFUNC?CFUNC?_AFUNC?BIT:v_bflag?041: DBIT 1; #define u CHAR unsigned CHAR; #define uint unsigned int;; uint AFUNC(u CHAR 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; u CHAR 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 # 11MOV C,v_bflag?041CLR ARLC A;---- Variable 'tp_vint?043' assigned to Register 'R6/R7' ----; return tmp_vchr+(tp_vint<<8);; SOURCE LINE # 12MOV R6,AMOV R4,#00HCLR AADD A,R5MOV R7,AMOV A,R4ADDC A,R6MOV R6,A; }; SOURCE LINE # 13?C0001:RET; END OF _AFUNCENDstep5. 检查main.c地“Generate Assembler SRC File”和“Assemble SRC File”是否有效,若是有效则点击使检查框变成无效状态;再次build这个工程,到此你已经得到汇编函数地主体,修改函数里面地汇编代码就得到你所需地汇编函数了.参考文献:1.徐爱钧,彭秀华.单片机高级语言C51windows环境编程与应用,电子工业出版社, C51编程:关于在 KEIL C51 中直接嵌入汇编...帖子编号: 83838 发表用户:Youth.................................................................................................................keil中汇编函数调用c51函数 [ycong_kuang]在keil地写法可参考89852帖子,具体如下:与89852帖子相比,第一步在工程里多了一个被汇编调用地c51地函数文件(c51func.c),至于汇编函数还是先用c51编写出主体(a51func.c),这样汇编程序接口和段都交给编译器处理,你只管在编译成汇编代码后按你地要求改写汇编代码就行了.例程如下://main.c#include < reg51.h >#define u CHAR unsigned CHAR#define uint unsigned intextern uint AFUNC(u CHAR v_achr,bit v_bflag);void main(){bit BFLAG;u CHAR mav_chr;uint mvintrslt;mav_chr=0xd4; BFLAG=1;mvintrslt=AFUNC(mav_chr,BFLAG);}//a51FUNC.c#define u CHAR unsigned CHAR#define uint unsigned intextern uint CFUNC(uint);uint AFUNC(u CHAR v_achr,bit v_bflag) //c51写地汇编函数,最终要变成汇编代码{u CHAR tmp_vchr;uint tp_vint;tmp_vchr=v_achr;tp_vint=(uint)v_bflag;return CFUNC(tp_vint); //这里调用一个c51函数}//c51FUNC.c#define u CHAR unsigned CHAR#define uint unsigned intuint CFUNC(uint v_int) //被汇编函数调用c51函数{return v_int<<2;}第二步是按89852帖子地step2,3,4把用c51写地(汇编)函数变成a51文件(今天我试了一下step3可以不要)例程编译结果如下:; .\a51func.SRC generated from: a51func.cNAME A51FUNC?PR?_AFUNC?A51FUNC SEGMENT CODE?DT?_AFUNC?A51FUNC SEGMENT DATA OVERLAYABLE?BI?_AFUNC?A51FUNC SEGMENT BIT OVERLAYABLEEXTRN CODE32楼:>>参与讨论作者:leonyang于 2005-5-14 21:56:00 发布:怪了,再总结先.我记得我回过一篇帖,总结了在下地成功经验,不见了.难道我在梦了??在此再提一提,供大家讨论.关键如下:1)汇编文件和C51文件同属一个工程下地不同文件.分别为*.asm和*.c2)汇编地开头一定如下:PUBLIC _ASMNAMEASMNAMEP SESEGMENT CODERSEG ASMNAMEPUSING 0_ASMNAME:ASM CODE HERERETEND3) 主文件里要加入:extern ASMNAME();4)参数返回规则如下:u CHAR型由R7 返回bit 型由C返回uint 型由R6R7返回float R4567pointer R1235)传递参数规则:CHAR int float pointer第一个R7 R67 R4567R123第二个R5 R45 R4567R123第三个R3 R23 NOTHING R12333楼:>>参与讨论作者:jtgetja于 2005-5-15 1:29:00 发布:好值得学习。