09. C语言内嵌汇编代码

C语言函数内可以自定义一段汇编代码,在GCC编译器中使用 asm 或 asm 关键词定义一段汇编代码,并可选添加volatile关键字,表示不要让编译器优化这段汇编代码。

内嵌汇编代码格式如下:

cpp 复制代码
__asm__
(
    "汇编代码"
    :输出描述
    :输入描述
    :修改描述
);

汇编代码部分

汇编代码部分是一个字符串,嵌入的汇编代码使用此字符串存储,多条汇编代码之间使用;符号隔开,字符串可以换行存储,换行存储方式如下:

cpp 复制代码
__asm__
(
    "mov eax,ecx; \
    add eax,ebx;"
);

或者:

cpp 复制代码
__asm__
(
    "mov eax,ecx;"
    "add eax,ebx;"
);

注:

1.汇编代码中16进制数据只能使用0x前缀,不能使用h后缀。

2.指定内存数据长度时,使用 word prt 关键词,不能省略 prt。

3.内嵌汇编代码默认使用AT&T语法,若使用inter语法则需要在编译时添加如下参数:-masm=intel,本文使用inter汇编语法。

输出描述部分

若汇编代码执行完毕后需要将寄存器、内存单元中的数据保存在C语言代码定义的局部变量中,则需要在输出描述部分定义寄存器或内存单元与局部变量的绑定关系,绑定代码格式如下:

cpp 复制代码
#include <stdio.h>
int main()
{
    int i;

    __asm__
    (
        "mov eax,1; \
        mov ebx,2; \
        add eax,ebx;"
        
        :"=a"(i)          //输出描述部分,"=a"表示输出eax寄存器,(i)表示使用变量i接收输出数据
    );
    
    printf("%d\n", i);
    return 0;
}

输出描述部分使用简写代码表示要输出的寄存器或内存单元,常用代码如下:

a,表示ax系寄存器

b,表示bx系寄存器

c,表示cx系寄存器

d,表示dx系寄存器

S,表示si系寄存器

D,表示di系寄存器

r,表示自动分配的寄存器

m,表示自动分配的内存单元

g,表示自动分配的寄存器或内存单元

绑定的局部变量可以使用指针指定。

cpp 复制代码
#include <stdio.h>
int main()
{
    int i;
    int *p1 = &i;
    
    __asm__
    (
        "mov eax,1; \
        mov ebx,2; \
        add eax,ebx;"
        
        :"=a"(*p1)
    );
    
    printf("%d\n", i);
    return 0;
}

输入描述部分

输入描述用于将局部变量与指定的寄存器或内存单元绑定,绑定的变量会首先复制到对应的寄存器或内存单元,然后再执行汇编代码。

cpp 复制代码
#include <stdio.h>
int main()
{
    int i1, i2, i3;
    
    printf("输入两个整数\n");
    scanf("%d%d", &i1, &i2);
    
    __asm__
    (
        "add eax,ebx;"
        
        :"=a"(i3)            //输出描述部分,eax写入i3
        
        :"a"(i1), "b"(i2)    //输入描述部分,i1写入eax,i2写入ebx
    );
    
    printf("两数相加结果为:%d\n", i3);
    return 0;
}

修改描述部分

修改描述用于告知编译器哪些寄存器、内存单元被汇编代码修改过,让编译器在编译代码时对这些寄存器或内存单元进行保护,当然我们也可以手动进行保护,在使用寄存器之前首先将其入栈存储,在汇编代码末尾处还原寄存器。

修改描述注意事项:

1.若被修改的寄存器在输出描述、输入描述中记录过,则无需在修改描述中重复指定,编译器会自动处理。

2.若修改了内存单元,则应该在此部分定义"memory"。

3.若修改了标志寄存器,则应该在此部分定义"c"。

cpp 复制代码
#include <stdio.h>
int main()
{
	int i1, i2, i3;
	
	printf("输入两个整数\n");
	scanf("%d%d", &i1, &i2);
	
	__asm__
	(
		"add eax,ebx; \
		mov edx,3; \
		imul ecx,edx"        //ecx与edx相乘仅做示例,没有实际作用
		
		:"=a"(i3)            //输出描述部分,eax写入i3
		
		:"a"(i1), "b"(i2)    //输入描述部分,i1写入eax,i2写入ebx
		
		:"ecx", "edx"        //修改描述部分,告知编译器ecx、edx被修改过,并且没有在输入输出描述中记录
	);
	
	printf("两数相加结果为:%d\n", i3);
	return 0;
}

编译器自动分配寄存器、内存单元

在汇编代码中存储数据时可以让编译器自动分配寄存器或内存,汇编代码使用"%数字"的方式调用编译器自动分配的寄存器或内存,比如:%0、%1、%2,这些名称称为占位符,占位符可以不按顺序定义。

占位符可以与输入输出描述中的局部变量绑定,绑定顺序为局部变量在输入输出描述中出现的顺序,%0绑定第一个变量、%1第二个、%2第三个。

cpp 复制代码
#include <stdio.h>
int main()
{
    int i1, i2, i3;
    
    printf("输入两个整数\n");
    scanf("%d%d", &i1, &i2);
    
    __asm__
    (
        "add %1,%2; \
        mov %0,%1"
        
        :"=m"(i3)            //输出描述部分,%0写入i3,这里使用m,表示让编译器自动分配内存单元存储绑定的i3
        
        :"r"(i1), "r"(i2)    //输入描述部分,i1写入%1,i2写入%2,这里使用r,表示让编译器自动分配寄存器存储绑定的i1、i2
    );
    
    printf("两数相加结果为:%d\n", i3);
    return 0;
}

上面汇编代码中三个变量出现顺序是i3、i1、i2,%0绑定i3,%1绑定i1,%2绑定i2。

使用全局变量

在汇编代码中调用全局变量无需进行任何设置,直接使用变量名即可,编译器会自动转换为对应的内存地址。

cpp 复制代码
#include <stdio.h>
int i1, i2;
int main()
{
    printf("输入两个整数\n");
    scanf("%d%d", &i1, &i2);
    
    __asm__
    (
        "push rax; \
        mov eax,i1; \
        add eax,i2; \
        mov i1,eax; \
        pop rax;"
    );
    
    printf("两数相加结果为:%d\n", i1);
    return 0;
}
相关推荐
Xxxx. .Xxxx2 小时前
C语言程序设计实验与习题指导 (第4版 )课后题-第二章+第三章
java·c语言·开发语言
DogDaoDao4 小时前
Windows 环境下 vscode 配置 C/C++ 环境
c语言·c++·windows·vscode·gcc·mingw-w64
Beginner_bml4 小时前
结构体---C语言
c语言·开发语言·数据结构
巭犇6 小时前
c语言中define使用方法
c语言·开发语言
奇点 ♡7 小时前
【线程】线程的控制
linux·运维·c语言·开发语言·c++·visual studio code
铁匠匠匠9 小时前
【C总集篇】第八章 数组和指针
c语言·开发语言·数据结构·经验分享·笔记·学习·算法
王哈哈嘻嘻噜噜9 小时前
c语言中“函数指针”
java·c语言·数据结构
别耗尽11 小时前
错题集锦之C语言
c语言
六点半88811 小时前
【C/C++】速通涉及string类的经典编程题
c语言·开发语言·c++·算法
Jack黄从零学c++11 小时前
自制网络连接工具(支持tcpudp,客户端服务端)
linux·c语言·开发语言·网络协议·tcp/ip·udp·信息与通信