static inline Linux内联函数

转载 2012年03月24日 16:35:02

asmlinkage/FASTCALL()/fastcall

这三个宏指定了函数参数的传递方式。asmlinkage修饰的函数,其参数通过堆栈传递。FASTCALL()/fastcall,此二者实际上是一样的作用。在Intel i386架构中,它们所修饰的函数,其前三个参数分别通过通用寄存器EAX,ECX和EDX来传递。
它们定义于include/asm-i386/linkage.h:
#define asmlinkage CPP_ASMLINKAGE __attribute__((regparm(0)))
#define FASTCALL(x) x __attribute__((regparm(3)))
#define fastcall __attribute__((regparm(3)))
此处利用的是函数属性regparm。

static inline内联函数

内联函数有些类似于宏。内联函数的代码会被直接嵌入在它被调用的地方2,调用几次就嵌入几次,没有使用call指令。这样省去了函数调用时的一些额外开销,比如保存和恢复函数返回地址等,可以加快速度。不过调用次数多的话,会使可执行文件变大,这样会降低速度。相比起宏来说,内核开发者一般更喜欢使用内联函数。因为内联函数没有长度限制,格式限制。编译器还可以检查函数调用方式,以防止其被误用。
static inline的内联函数,一般情况下不会产生函数本身的代码,而是全部被嵌入在被调用的地方。如果不加static,则表示该函数有可能会被其他编译单元所调用,所以一定会产生函数本身的代码。所以加了static,一般可令可执行文件变小。内核里一般见不到只用inline的情况,而都是使用static inline。
有时候也可以见到static __inline__。gcc编译参数“-ansi”或者“-std=c89”等会禁用一些关键字,比如“asm”,“inline”等。为了在这种情况下使用内联函数,gcc提供了“__inline__”关键字。不过内核使用的标准是gnu89,所以“inline”和“__inline__”在内核中都可以使用。现在“inline”已经成了C99标准的关键字。gcc未来的缺省标准也将变成基于C99的gnu99。所以现在写内核代码时,推荐使用“inline”关键字,而不要用“__line__”。

Linux内核中的许多规范需要经过反复查找和阅读才能发现其最终的意义和目的。本篇文章搜集并澄清C语言用法中几个含糊不清,容易混淆的地方。

(1) asmlinkage

asmlinkage 告诉编译器要使用局部堆栈来传递参数,而宏FASTCALL通知编译程序使用通用寄存器来传递参数。

以下是来自 include/linux/linkage.h 的代码(2.4.0):

#ifdef __cplusplus

#define CPP_ASMLINKAGE extern "C"

#else

#define CPP_ASMLINKAGE

#endif

#if defined __i386__

#define asmlinkage CPP_ASMLINKAGE __attribute__((regparm(0)))

#elif defined __ia64__

#define asmlinkage CPP_ASMLINKAGE __attribute__((syscall_linkage))

#else

#define asmlinkage CPP_ASMLINKAGE

#endif

(2) UL

UL常被用在数值常数后,标明该常数为"unsigned long"类型,UL(L代表long)负责告诉编译程序将这一数值当作long型数值来处理。因此,使用UL很有必要,它能够保证特定体系结构内的数据不会溢出其数据类型所规定的范围。涉及很大的数或长的位掩码时,使用UL有助于编写出与体系结构无关的代码。

Linux代码中有这样的例子,如:

include/linux/kernel.h

===============

#define INT_MAX ((int)(~0U>>1))

#define INT_MIN (-INT_MAX - 1)

#define UINT_MAX (~0U)

#define LONG_MAX ((long)(~0UL>>1))

#define LONG_MIN (-LONG_MAX - 1)

#define ULONG_MAX (~0UL)

(3)inline

关键字inline表明要优化函数的可执行代码,这可以通过将函数的代码合并到调用程序的代码中来实现。Linux内核使用的inline函数大多被定义为static 类型。一个"static inline"函数促使编译程序尝试着将其代码插入到所有调用它的程序中。

这一合并能够免除函数调用的任何开销,#define语句也可以排除额外的函数调用。

另外,使用inline会增加二进制映像的大小,而这会降低访问CPU高速缓存的速度,所以不能在所有的函数定义中使用它。

(4)const 和 volatile

const不一定只代表常数,有时它表示"只读"的意思。例如,"const int * x" 中x是一个指向const整数的指针,因此,可以修改指针,但不能修改这个整数;而在"int const * x"中,x却是一个指向整数的const指针,因而这个整数可以改变,但指针x却不能改变。

关键字volatile表明变量无需警告就可以被修改,它通知编译程序每次使用该变量时都要重新加载其值,而不是存储并访问一个副本。中断处理,硬件寄存器,以及并发进程之间共享的变量都是被标记为volatile的典型例子.

include/asm-i386/spinlock.h

===================

typedef struct {

volatile unsigned int lock;

#if SPINLOCK_DEBUG

unsigned magic;

#endif

} spinlock_t;

linux下内联函数实现浅析

首先我们来看看内联函数的相关解释: 1.引入目的:为了解决程序中函数调用的效率问题 2.  优点:节省了函数调用时间,也就是说没有call指令(也就没有相关参数的压栈(push),跳转(jmp),返回...

linux c inline

转自:http://hi.baidu.com/beibeiboo/blog/item/cbea8c03191c027e3812bb72.html 看kernel,随处可见inline.说实话,还没写...

关于c中的inline

在c中,为了解决一些频繁调用的小函数大量消耗栈空间或是叫栈内存的问题,特别的引入了inline修饰符,表示为内联函数。栈空间就是指放置程式的局部数据也就是函数内数据的内存空间,在系统下,栈空间是有限的...

linux驱动中使用static关键字的重要性

linux内核十分宠大,代码量超过百万行。对于C语言的函数和全局变量的作用空间都是全局的,在另外一个文件中,使用extern关键字就可以实现对于其他文件中的全局变量和函数的访问。因此,一旦源码中函数名...
  • armeasy
  • armeasy
  • 2010年11月17日 17:31
  • 2226

linux下编译静态(static)库

configure的时候要指定 --enable-static  ,因为很多平台没有基础库,指定该选项将免去一些不必要的麻烦。 #./configure --enable-static ...

Linux下which、whereis、locate、find 命令的区别

本文转自http://blog.chinaunix.net/uid-20554039-id-3035417.html 我们经常在linux要查找某个文件,但不知道放在哪里了,可以使用下面的一些命令来...

linux下使用xml

程序中用了libxml的库,但是不知道库函数具体如何是用,那么下面的参考手册能解决你的大部分问题,简明易懂,程序表达效果。 http://www.xmlsoft.org/tutorial/index....

static inline func 内联函数

static inline function是干嘛的? 如果你的.m文件需要频繁调用一个函数,可以用static inline来声明,这相当于把函数体当做一个大号的宏定义.不过这也不是百分之...

C++内联函数(Inline)介绍

C++内联函数(Inline)介绍2001-11-12 09:34作者:杜瑾出处:yesky责任编辑:方舟  介绍内联函数之前,有必要介绍一下预处理宏。内联函数的功能和预处理宏的功能相似。相信大家都用...

c++-inline 内联函数

为什么要使用内联函数: 当程序调用函数时,系统要建立栈空间,保存
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:static inline Linux内联函数
举报原因:
原因补充:

(最多只允许输入30个字)