内核中likely和unlikely宏定义

在内核代码中经常会看到unlikely和likely的踪影。他们实际上是定义在 linux/compiler.h 中的两个宏。

  #define likely(x)    __builtin_expect(!!(x), 1)

  #define unlikely(x)  __builtin_expect(!!(x), 0)

  这里的__built_expect()函数是gcc的內建函数。

  至于为什么要在内核代码中使用这两个宏,主要的目的是为了进行代码的优化,提高系统执行速度。

  比如 :

  if (likely(a>b)) {

  fun1();

  }

  if (unlikely(a<b)) {

  fun2();

  }

  这里就是程序员可以确定 a>b 在程序执行过程中出现的可能相比较大,因此使用了likely()告诉编译器将fun1()函数的二进制代码紧跟在前面程序的后面,这样就cache在预取数据时就可以将fun1()函数的二进制代码拿到cache中。这样,也就增加了cache的命中率。

  同样的,unlikely()的作用就是告诉编译器,a<b 的可能性很小所以这里在编译时,将fun2()的二进制代码尽量不要和前边的编译在一块。

  我们不用对likely和unlikely感到迷惑,需要知道的就是 if(likely(a>b)) 和 if(a>b)在功能上是等价的,同样 if(unlikely(a<b)) 和 if(a<b) 的功能也是一样的。不一样的只是他们声称的二进制代码有所不同,这一点我们也可以从他们的汇编代码中看到。

  比如下面的代码:

  #include <stdio.h>

  #define unlikely(x) __builtin_expect(!!(x), 0)

  #define likely(x) __builtin_expect(!!(x), 1)

  int main()

  {

  int a=2,b=4;

  if(unlikely(a<b)) {

  printf("in the unlikely,is not your expecting!\n");

  } else {

  printf("in the unlikely, is your expecting\n");

  }

  if(likely(a<b)) {

  printf("in the likely, is your expecting\n");

  }

  return 0;

  }

  执行结果:

  in the unlikely,is not your expecting!

  in the likely, is your expecting

  总之,likely和unlikely的功能就是增加cache的命中率,提高系统执行速度。

时间: 2024-10-11 08:23:03

内核中likely和unlikely宏定义的相关文章

linux内核头文件kdev_t.h 宏定义解析

kdev_t.h 宏定义解析 这个header file反正不多,直接原因是--遇到了,就搞定它! dev_t 类型的变量定义在linux/types.h 用来保存设备编号--包括主设备号和次设备号.dev_t 是一个32位的数,其中12位用来表示设备号,其余20位用来表示次设备号. 始终不要对这32位是高12位是主设备号还是低2位是主设备号做出假定,不利于代码的可移植性,始终记得使用宏定义来处理dev_t ! 都在这里了: #define MINORBITS 20 //次设备号的占位数目 #d

swift中 类似OC的宏定义

OC 中调试打印 /** *  自定义Log */ #ifdef DEBUG #define JYLog(...)        NSLog(__VA_ARGS__) #else #define JYLog(...) #endif // swift中我写了这么一个类,进行调试输出 调用方法  Debug.Log(error) import UIKit class Debug: NSObject{ // TODO: 为调试修改 private static let debug = true cla

VC中预处理指令与宏定义详解

刚接触到MFC编程的人往往会被MFC 向导生成的各种宏定义和预处理指令所吓倒,但是预处理和宏定义又是C语言的一个强大工具.使用它们可以进行简单的源代码控制,版本控制,预警或者完成一些特殊的功能. 一个经典的例子 使用预处理与宏定义最经典的例子莫过于加在一个头文件中以避免头文件被两次编译.试想这种的情况,有一个文件headerfile.h 它被包含在headerfile1.h中,同时在headerfile2.h 中也被包含了,现在有一个CPP文件,implement.cpp 包含了headerfi

C中的预编译宏定义

文章来自 http://www.uml.org.cn/c++/200902104.asp 在将一个C源程序转换为可执行程序的过程中, 编译预处理是最初的步骤. 这一步骤是由预处理器(preprocessor)来完成的. 在源流程序被编译器处理之前, 预处理器首先对源程序中的"宏(macro)"进行处理. C初学者可能对预处理器没什么概念, 这是情有可原的: 一般的C编译器都将预处理, 汇编, 编译, 连接过程集成到一起了. 编译预处理往往在后台运行. 在有的C编译器中, 这些过程统统由

linux中offsetof与container_of宏定义

linux内核中offsetof与container_of的宏定义 #define offsetof(TYPE, MEMBER)    ((size_t) &((TYPE *)0)->MEMBER) /** * container_of - cast a member of a structure out to the containing structure * @ptr:        the pointer to the member. * @type:       the type

C中变量名通过宏定义转换成char*类型字符串

#include <iostream> p>define name_to_str(name)(#name) int main() { int sum = 10; std::cout<<name_to_str(sum)<<std::endl;//调用宏输出char*型“sum” system("pause"); return 0; }

【转】linux内核中writesb(), writesw(), writesl() 宏函数

writesb(), writesw(), writesl() 宏函数 功能 : writesb()    I/O 上写入 8 位数据流数据 (1字节) writesw()   I/O  上写入 16 位流数据 (2字节) writesl()    I/O 上写入 32 位流数据 ( 4字节 ) 原型 : 引用 #include <asm/io.h> void   writesb (unsigned short ioaddr , void * addr , unsigned long coun

Xcode 工程中添加预编译宏定义

Build Settings —— > Preprocessor Macros. 使用例子: 1 #ifndef APP_STORE_DISTRIBUTION 2 //your method 3 #if DEBUG 4 //your method 5 #else 6 //your method 7 #endif 8 //your method 9 #endif

Linux内核中大小端判定宏

#include <stdio.h> static union{ char c[4];unsigned long mylong;} endian_test = { {'l','?','?','b'} }; #define ENDIANNESS ( (char) endian_test.mylong ) int main() { printf("%c",ENDIANNESS); } 略显简洁