GCCG++ C C++ 内嵌汇编和调用汇编函数的方法(x86,ARM自己对照改)

2024-06-16 07:08

本文主要是介绍GCCG++ C C++ 内嵌汇编和调用汇编函数的方法(x86,ARM自己对照改),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

#PS:要转载请注明出处,本人版权所有

#PS:这个只是 《 我自己 》理解,如果和你的

#原则相冲突,请谅解,勿喷

测试环境:
Linux 4.8.0-36-generic #36~16.04.1-Ubuntu SMP Sun Feb 5 09:39:57 UTC 2017 x86_64 x86_64 x86_64 GNU/Linux
gcc:
Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/5/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v –with-pkgversion=’Ubuntu 5.4.0-6ubuntu1~16.04.4’ –with-bugurl=file:///usr/share/doc/gcc-5/README.Bugs –enable-languages=c,ada,c++,java,go,d,fortran,objc,obj-c++ –prefix=/usr –program-suffix=-5 –enable-shared –enable-linker-build-id –libexecdir=/usr/lib –without-included-gettext –enable-threads=posix –libdir=/usr/lib –enable-nls –with-sysroot=/ –enable-clocale=gnu –enable-libstdcxx-debug –enable-libstdcxx-time=yes –with-default-libstdcxx-abi=new –enable-gnu-unique-object –disable-vtable-verify –enable-libmpx –enable-plugin –with-system-zlib –disable-browser-plugin –enable-java-awt=gtk –enable-gtk-cairo –with-java-home=/usr/lib/jvm/java-1.5.0-gcj-5-amd64/jre –enable-java-home –with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-5-amd64 –with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-5-amd64 –with-arch-directory=amd64 –with-ecj-jar=/usr/share/java/eclipse-ecj.jar –enable-objc-gc –enable-multiarch –disable-werror –with-arch-32=i686 –with-abi=m64 –with-multilib-list=m32,m64,mx32 –enable-multilib –with-tune=generic –enable-checking=release –build=x86_64-linux-gnu –host=x86_64-linux-gnu –target=x86_64-linux-gnu
Thread model: posix
gcc version 5.4.0 20160609 (Ubuntu 5.4.0-6ubuntu1~16.04.4)

1 内嵌汇编格式和编译器息息相关,对于不同的编译器,有不同的内嵌规则,这里以gcc为例。(此外,基本的一些知识点,我这里就不重复说了,有需要可以去百度相关的东西,此文的面向读者为了解一些汇编知识的人,比如知道一些基本的寄存器等等)
2 gcc内嵌汇编格式,基本内嵌汇编寄存器等引用为%,带C&&C++的寄存器等引用为%%

__asm__ [__volatile__] ("instruction list");//基本内嵌汇编
__asm__ [__volatile__]("instruction list":Output:Input:Clobber/Modify);//带C&C++相关内容的内嵌汇编

3 以c=a+b为例

  • 在代码段中内嵌汇编

    根据第二节的第二种格式,一一对应,就知道,我把a给了ebx,b给了eax,求和后放入eax,把eax传给变量ret

int user_add1(int a, int b){int ret;__asm__ __volatile__("movl %2, %%eax;movl %1, %%ebx;addl %%ebx, %%eax;movl %%eax, %0"       :"=m"(ret):"m"(a),"m"(b):"eax","ebx","memory");return ret;
}
  • 在代码中调用汇编子程序
    这里先开辟一个栈帧,然后读到两个传入参数a,b在栈中的位置,并放入寄存器,并对寄存器进行操作求和,函数返回值放在eax中
//xxx.c文件中
c=user_add(a+b);
//xxx.s文件中
.text
.globl  user_add
.type   user_add, @function
user_add:.cfi_startprocpushq   %rbp.cfi_def_cfa_offset 16.cfi_offset 6, -16movq    %rsp, %rbp.cfi_def_cfa_register 6movl    %edi, -20(%rbp)movl    %esi, -24(%rbp)addl    %esi, %edimovl    %edi, -4(%rbp)movl    -4(%rbp), %eaxpopq    %rbp.cfi_def_cfa 7, 8ret.cfi_endproc
  • 在内嵌汇编代码中调用汇编子程序

由于调用的时候,我先吧参数传给了eax,ebx,通过call,我直接对eax,ebx求和,得到结果

//xxx.c文件中int ret;__asm__ __volatile__("movl %2, %%eax;movl %1, %%ebx;call user_add2"     :"=a"(ret):"m"(a),"m"(b):"ebx","memory");//xxx.s文件中
.text
.globl  user_add2
.type   user_add2, @function
user_add2:addl %ebx,%eaxret

4 总结与分析

1 首先这是x86下面的汇编指令格式,如果是ARM或者MIPS等平台,请根据各自的指令格式进行修改。

2 这里需要注意的是,从这三种方式来看,前两种方式里面,我们必须要注意,在通过c的方式调用函数时,其参数的存放位置在哪里,参数的存放顺序,同时我们还必须知道gcc的默认调用约定是什么。这是极其重要的

第三种方式的调用的最简单粗暴的,但是可能会隐含一些问题,如果gcc没有帮你很好的保护现场,那么可能会出现程序崩溃的情况。

5 源代码和测试

//t.c
#include <stdio.h>
extern int  user_add(int a, int b);
extern int  user_add2(int a, int b);
int user_add1(int a, int b){int ret;__asm__ __volatile__("movl %2, %%eax;movl %1, %%ebx;addl %%ebx, %%eax;movl %%eax, %0"       :"=m"(ret):"m"(a),"m"(b):"eax","ebx","memory");return ret;
}
int main(int argc, char *argv[]){int a = 6;int b = 3;printf("user_add1 sum(a+b)=%d\n",user_add1(a,b));printf("user_add sum(a+b)=%d\n",user_add(a,b));int ret;__asm__ __volatile__("movl %2, %%eax;movl %1, %%ebx;call user_add2"     :"=a"(ret):"m"(a),"m"(b):"ebx","memory");printf("user_add2 sum(a+b)=%d\n",ret);return 0;
}//t1.s.text
.globl  user_add
.type   user_add, @function
user_add:.cfi_startprocpushq   %rbp.cfi_def_cfa_offset 16.cfi_offset 6, -16movq    %rsp, %rbp.cfi_def_cfa_register 6movl    %edi, -20(%rbp)movl    %esi, -24(%rbp)addl    %esi, %edimovl    %edi, -4(%rbp)movl    -4(%rbp), %eaxpopq    %rbp.cfi_def_cfa 7, 8ret.cfi_endproc.text
.globl  user_add2
.type   user_add2, @function
user_add2:addl %ebx,%eaxret

编译及运行效果:
这里写图片描述

#PS:请尊重原创,不喜勿喷

#PS:要转载请注明出处,本人版权所有.

有问题请留言,看到后我会第一时间回复

这篇关于GCCG++ C C++ 内嵌汇编和调用汇编函数的方法(x86,ARM自己对照改)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1065778

相关文章

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

C++对象布局及多态实现探索之内存布局(整理的很多链接)

本文通过观察对象的内存布局,跟踪函数调用的汇编代码。分析了C++对象内存的布局情况,虚函数的执行方式,以及虚继承,等等 文章链接:http://dev.yesky.com/254/2191254.shtml      论C/C++函数间动态内存的传递 (2005-07-30)   当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道。 文章链接:http://dev.yesky

C++的模板(八):子系统

平常所见的大部分模板代码,模板所传的参数类型,到了模板里面,或实例化为对象,或嵌入模板内部结构中,或在模板内又派生了子类。不管怎样,最终他们在模板内,直接或间接,都实例化成对象了。 但这不是唯一的用法。试想一下。如果在模板内限制调用参数类型的构造函数会发生什么?参数类的对象在模板内无法构造。他们只能从模板的成员函数传入。模板不保存这些对象或者只保存他们的指针。因为构造函数被分离,这些指针在模板外

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

[word] word设置上标快捷键 #学习方法#其他#媒体

word设置上标快捷键 办公中,少不了使用word,这个是大家必备的软件,今天给大家分享word设置上标快捷键,希望在办公中能帮到您! 1、添加上标 在录入一些公式,或者是化学产品时,需要添加上标内容,按下快捷键Ctrl+shift++就能将需要的内容设置为上标符号。 word设置上标快捷键的方法就是以上内容了,需要的小伙伴都可以试一试呢!

C++工程编译链接错误汇总VisualStudio

目录 一些小的知识点 make工具 可以使用windows下的事件查看器崩溃的地方 dumpbin工具查看dll是32位还是64位的 _MSC_VER .cc 和.cpp 【VC++目录中的包含目录】 vs 【C/C++常规中的附加包含目录】——头文件所在目录如何怎么添加,添加了以后搜索头文件就会到这些个路径下搜索了 include<> 和 include"" WinMain 和

大学湖北中医药大学法医学试题及答案,分享几个实用搜题和学习工具 #微信#学习方法#职场发展

今天分享拥有拍照搜题、文字搜题、语音搜题、多重搜题等搜题模式,可以快速查找问题解析,加深对题目答案的理解。 1.快练题 这是一个网站 找题的网站海量题库,在线搜题,快速刷题~为您提供百万优质题库,直接搜索题库名称,支持多种刷题模式:顺序练习、语音听题、本地搜题、顺序阅读、模拟考试、组卷考试、赶快下载吧! 2.彩虹搜题 这是个老公众号了 支持手写输入,截图搜题,详细步骤,解题必备

C/C++的编译和链接过程

目录 从源文件生成可执行文件(书中第2章) 1.Preprocessing预处理——预处理器cpp 2.Compilation编译——编译器cll ps:vs中优化选项设置 3.Assembly汇编——汇编器as ps:vs中汇编输出文件设置 4.Linking链接——链接器ld 符号 模块,库 链接过程——链接器 链接过程 1.简单链接的例子 2.链接过程 3.地址和

C++必修:模版的入门到实践

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:C++学习 贝蒂的主页:Betty’s blog 1. 泛型编程 首先让我们来思考一个问题,如何实现一个交换函数? void swap(int& x, int& y){int tmp = x;x = y;y = tmp;} 相信大家很快就能写出上面这段代码,但是如果要求这个交换函数支持字符型

电脑不小心删除的文件怎么恢复?4个必备恢复方法!

“刚刚在对电脑里的某些垃圾文件进行清理时,我一不小心误删了比较重要的数据。这些误删的数据还有机会恢复吗?希望大家帮帮我,非常感谢!” 在这个数字化飞速发展的时代,电脑早已成为我们日常生活和工作中不可或缺的一部分。然而,就像生活中的小插曲一样,有时我们可能会在不经意间犯下一些小错误,比如不小心删除了重要的文件。 当那份文件消失在眼前,仿佛被时间吞噬,我们不禁会心生焦虑。但别担心,就像每个问题