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

相关文章

使用Python获取JS加载的数据的多种实现方法

《使用Python获取JS加载的数据的多种实现方法》在当今的互联网时代,网页数据的动态加载已经成为一种常见的技术手段,许多现代网站通过JavaScript(JS)动态加载内容,这使得传统的静态网页爬取... 目录引言一、动态 网页与js加载数据的原理二、python爬取JS加载数据的方法(一)分析网络请求1

MySQL查看表的最后一个ID的常见方法

《MySQL查看表的最后一个ID的常见方法》在使用MySQL数据库时,我们经常会遇到需要查看表中最后一个id值的场景,无论是为了调试、数据分析还是其他用途,了解如何快速获取最后一个id都是非常实用的技... 目录背景介绍方法一:使用MAX()函数示例代码解释适用场景方法二:按id降序排序并取第一条示例代码解

Python中合并列表(list)的六种方法小结

《Python中合并列表(list)的六种方法小结》本文主要介绍了Python中合并列表(list)的六种方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录一、直接用 + 合并列表二、用 extend() js方法三、用 zip() 函数交叉合并四、用

Java 中的跨域问题解决方法

《Java中的跨域问题解决方法》跨域问题本质上是浏览器的一种安全机制,与Java本身无关,但Java后端开发者需要理解其来源以便正确解决,下面给大家介绍Java中的跨域问题解决方法,感兴趣的朋友一起... 目录1、Java 中跨域问题的来源1.1. 浏览器同源策略(Same-Origin Policy)1.

Java Stream.reduce()方法操作实际案例讲解

《JavaStream.reduce()方法操作实际案例讲解》reduce是JavaStreamAPI中的一个核心操作,用于将流中的元素组合起来产生单个结果,:本文主要介绍JavaStream.... 目录一、reduce的基本概念1. 什么是reduce操作2. reduce方法的三种形式二、reduce

MybatisX快速生成增删改查的方法示例

《MybatisX快速生成增删改查的方法示例》MybatisX是基于IDEA的MyBatis/MyBatis-Plus开发插件,本文主要介绍了MybatisX快速生成增删改查的方法示例,文中通过示例代... 目录1 安装2 基本功能2.1 XML跳转2.2 代码生成2.2.1 生成.xml中的sql语句头2

python3 pip终端出现错误解决的方法详解

《python3pip终端出现错误解决的方法详解》这篇文章主要为大家详细介绍了python3pip如果在终端出现错误该如何解决,文中的示例方法讲解详细,感兴趣的小伙伴可以跟随小编一起了解一下... 目录前言一、查看是否已安装pip二、查看是否添加至环境变量1.查看环境变量是http://www.cppcns

Linux给磁盘扩容(LVM方式)的方法实现

《Linux给磁盘扩容(LVM方式)的方法实现》本文主要介绍了Linux给磁盘扩容(LVM方式)的方法实现,涵盖PV/VG/LV概念及操作步骤,具有一定的参考价值,感兴趣的可以了解一下... 目录1 概念2 实战2.1 相关基础命令2.2 开始给LVM扩容2.3 总结最近测试性能,在本地打数据时,发现磁盘空

C++ HTTP框架推荐(特点及优势)

《C++HTTP框架推荐(特点及优势)》:本文主要介绍C++HTTP框架推荐的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Crow2. Drogon3. Pistache4. cpp-httplib5. Beast (Boos

使用Python实现调用API获取图片存储到本地的方法

《使用Python实现调用API获取图片存储到本地的方法》开发一个自动化工具,用于从JSON数据源中提取图像ID,通过调用指定API获取未经压缩的原始图像文件,并确保下载结果与Postman等工具直接... 目录使用python实现调用API获取图片存储到本地1、项目概述2、核心功能3、环境准备4、代码实现