如果您在“alloc.c”中有以下代码:
typedef __typeof__(sizeof(int)) size_t;
extern void *calloc (size_t __nmemb, size_t __size)
  __attribute__ ((__nothrow__ )) __attribute__ ((__malloc__)) ;
extern void free (void *__ptr) __attribute__ ((__nothrow__ ));
struct _astruct {
  int l;
};
int main() {
  struct _astruct *f = (struct _astruct*)calloc(1, sizeof(struct _astruct));
  f->l = 99;
  int x = f->l;
  free(f);
  return x;
}
(我知道这不是声明 calloc/free 的首选方式,但它是为了简化下面的输出。)
然后你clang -O3 -S -emit-llvm alloc.c用 Clang/LLVM 3.3 运行“”,你会得到:
; ModuleID = 'alloc.c'
target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64-S128"
target triple = "x86_64-unknown-linux-gnu"
; Function Attrs: nounwind readnone uwtable
define i32 @main() #0 {
entry:
  ret i32 99
}
attributes #0 = { nounwind readnone uwtable "less-precise-fpmad"="false" "no-frame-pointer-elim"="false" "no-frame-pointer-elim-non-leaf"="false" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "unsafe-fp-math"="false" "use-soft-float"="false" }
您可以看到它完全消除了对 calloc 的调用。惊人的。但是,一般来说,编译器不能消除对它没有源的函数的调用(可能有副作用)。因此,Clang 似乎有 calloc/free 的特殊情况代码。如果您将代码中所有出现的“calloc”更改为“_calloc”,则“分配”会返回,这一事实支持这一点。
所以我的问题是,有没有办法告诉 Clang/LLVM 某个特定函数是“仅分配”函数?也许我可以在我自己的一个函数的定义上添加一些注释,这将允许 LLVM 优化分配,就像它可以用于 calloc/malloc 一样。