ARM体系下的GCC内联汇编

发布时间: 2024-08-30
来源: 电子工程世界

在操作系统级的编程中,有时候,C语言并不能完全的使用硬件的功能,这时候就需要嵌入一些汇编代码来实现功能。 有两种方式可以使C语言和assemly语言一起工作,一种是两种语言分开写成两个文件,链接的时候链接成一个文件;另一种就是在C语言中嵌入汇编代码。下面简单介绍一下如何在GCC中嵌入汇编代码。


GCC规定了一个内联汇编的语法,不同硬件平台上的GCC内联汇编几乎都是这样的:


asm(

    汇编指令列表

    :输出运算符列表

    :输入运算符列表

    :被更改的资源列表

};

在GCC中插入汇编代码,需要以asm关键字开头,中间四个部分用”:”分隔,如果你嵌入的汇编没有输入输出,或者更改资源,后面三项是可以省略的。


下面以一个实例来说明这个语法:


 1  void test(void)

 2  {

 3      int tmp;

 4      // some code

 5      __asm__(

 6         ' mov r1,%0nt'

 7         : 

 8         : 'r' (tmp)

 9         : 'r1'

10      );

11      

12   }

13   


以上代码的意思就是,将tmp变量的值赋给r1寄存器,%0代表出现在输入运算符列表和输出运算符列表中的第一个值,%1,%2依次类推。由于我们自己的汇编代码改变了r1的值,所以我们要通知GCC编译器,r1的值被我们改变了, 在 “r” (tmp) 表达式中,tmp代表C语言输入到汇编中的变量,”r”代表tmp会通过一个寄存器传递。可以使用的符号有以下几种:


表1 GCC4内联汇编操作符节选

操作符 含义
r 通用寄存器R0~R15
m 一个有效内存地址
l 数据处理指令中的立即数
X 被修饰的操作符只能作为输出

上面一个代码是将C语言的值传递到汇编代码中,也可以将汇编代码输出的结果传给C代码:


 1     void test(void)

 2     {

 3         int tmp;

 4         __asm__(

 5             'mov %0, #1nt'

 6             : '=r' (tmp)

 7             :

 8         ); 

 9      }

10      

这段代码的意思是,将立即数1赋给变量tmp。 与上面不同的是,输入运算符列表移到了输出运算符列表,”r”前面也多了一个等于号。这个等号被称为约束修饰符,以下是几种修饰符的含义列表:


表2  GCC4中内联汇编修饰符

修饰符 说明
被修饰的操作符是只读的
= 被修饰的操作符只写
+ 被修饰的操作符具有可读写的属性
& 被修饰的操作符只能作为输出


文章来源于: 电子工程世界 原文链接

本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。