在阅读GStreamer代码中有两个调用很频繁的宏:
#define G_LIKELY(expr) (__builtin_expect (_G_BOOLEAN_EXPR(expr), 1))
#define G_UNLIKELY(expr) (__builtin_expect (_G_BOOLEAN_EXPR(expr), 0))
其中 _G_BOOLEAN_EXPR宏的定义如下:
#define _G_BOOLEAN_EXPR(expr) /
__extension__ ({ /
int _g_boolean_var_; /
if (expr) /
_g_boolean_var_ = 1; /
else /
_g_boolean_var_ = 0; /
_g_boolean_var_; /
})
其目的无非就是将!=0的数转化为1。而函数__builtin_expect查阅GCC手册,发现其定义如下:
long __builtin_expect (long exp, long c) [Built-in Function]
You may use __builtin_expect to provide the compiler with branch prediction
information. In general, you should prefer to use actual profile feedback for this
(‘-fprofile-arcs’), as programmers are notoriously bad at predicting how their
programs actually perform. However, there are applications in which this data is
hard to collect.
The return value is the value of exp, which should be an integral expression. The
value of c must be a compile-time constant. The semantics of the built-in are that it
is expected that exp == c. For example:
if (__builtin_expect (x, 0))
foo ();
would indicate that we do not expect to call foo, since we expect x to be zero. Since
you are limited to integral expressions for exp, you should use constructions such as
if (__builtin_expect (ptr != NULL, 1))
error ();
when testing pointer or floating-point values.
大致是说,由于大部分程式员在分支预测方面做得非常糟糕,所以GCC提供了这个内建函数来帮助程式员处理分支预测,优化程式。其第一个参数exp为一个整型表达式,这个内建函数的返回值也是这个exp,而c为一个编译期常量,这个函数的语义是:你期望exp表达式的值等于常量c,从而GCC为你优化程式,将符合这个条件的分支放在合适的地方。
因为这个程式只提供了整型表达式,所以如果你要优化其他类型的表达式,能采用指针的形式。
说到底__builtin_expect函数就是为了优化可能性大的分支程式。