perf clang: Compile BPF script using builtin clang support
authorWang Nan <wangnan0@huawei.com>
Sat, 26 Nov 2016 07:03:39 +0000 (07:03 +0000)
committerArnaldo Carvalho de Melo <acme@redhat.com>
Mon, 5 Dec 2016 18:51:45 +0000 (15:51 -0300)
After this patch, perf utilizes builtin clang support to build BPF
script, no longer depend on external clang, but fallbacking to it
if for some reason the builtin compiling framework fails.

Test:

  $ type clang
  -bash: type: clang: not found
  $ cat ~/.perfconfig
  $ echo '#define LINUX_VERSION_CODE 0x040700' > ./test.c
  $ cat ./tools/perf/tests/bpf-script-example.c >> ./test.c
  $ ./perf record -v --dry-run -e ./test.c 2>&1 | grep builtin
  bpf: successfull builtin compilation
  $

Can't pass cflags so unable to include kernel headers now. Will be fixed
by following commits.

Committer notes:

Make sure '-v' comes before the '-e ./test.c' in the command line otherwise the
'verbose' variable will not be set when the bpf event is parsed and thus the
pr_debug indicating a 'successfull builtin compilation' will not be output, as
the debug level (1) will be less than what 'verbose' has at that point (0).

Signed-off-by: Wang Nan <wangnan0@huawei.com>
Tested-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Alexei Starovoitov <ast@fb.com>
Cc: He Kuang <hekuang@huawei.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Joe Stringer <joe@ovn.org>
Cc: Zefan Li <lizefan@huawei.com>
Cc: pi3orama@163.com
Link: http://lkml.kernel.org/r/20161126070354.141764-16-wangnan0@huawei.com
[ Spell check/reflow successfull pr_debug string ]
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
tools/perf/util/bpf-loader.c
tools/perf/util/c++/clang-c.h
tools/perf/util/c++/clang.cpp

index cf16b94115b5f0069d340cca3b0aeb4d5aca8a84..36c861103291982516b7463422544ed279fe18ed 100644 (file)
 #include "debug.h"
 #include "bpf-loader.h"
 #include "bpf-prologue.h"
-#include "llvm-utils.h"
 #include "probe-event.h"
 #include "probe-finder.h" // for MAX_PROBES
 #include "parse-events.h"
 #include "llvm-utils.h"
+#include "c++/clang-c.h"
 
 #define DEFINE_PRINT_FN(name, level) \
 static int libbpf_##name(const char *fmt, ...) \
@@ -86,9 +86,16 @@ struct bpf_object *bpf__prepare_load(const char *filename, bool source)
                void *obj_buf;
                size_t obj_buf_sz;
 
-               err = llvm__compile_bpf(filename, &obj_buf, &obj_buf_sz);
-               if (err)
-                       return ERR_PTR(-BPF_LOADER_ERRNO__COMPILE);
+               perf_clang__init();
+               err = perf_clang__compile_bpf(filename, &obj_buf, &obj_buf_sz);
+               perf_clang__cleanup();
+               if (err) {
+                       pr_warning("bpf: builtin compilation failed: %d, try external compiler\n", err);
+                       err = llvm__compile_bpf(filename, &obj_buf, &obj_buf_sz);
+                       if (err)
+                               return ERR_PTR(-BPF_LOADER_ERRNO__COMPILE);
+               } else
+                       pr_debug("bpf: successfull builtin compilation\n");
                obj = bpf_object__open_buffer(obj_buf, obj_buf_sz, filename);
 
                if (!IS_ERR(obj) && llvm_param.dump_obj)
index 22b3936d1f09d025ac87eef1f807f4078cd65edd..0eadd792ab1f0b99a784d14eb71fae99a6f1214f 100644 (file)
@@ -1,16 +1,42 @@
 #ifndef PERF_UTIL_CLANG_C_H
 #define PERF_UTIL_CLANG_C_H
 
+#include <stddef.h>    /* for size_t */
+#include <util-cxx.h>  /* for __maybe_unused */
+
 #ifdef __cplusplus
 extern "C" {
 #endif
 
+#ifdef HAVE_LIBCLANGLLVM_SUPPORT
 extern void perf_clang__init(void);
 extern void perf_clang__cleanup(void);
 
 extern int test__clang_to_IR(void);
 extern int test__clang_to_obj(void);
 
+extern int perf_clang__compile_bpf(const char *filename,
+                                  void **p_obj_buf,
+                                  size_t *p_obj_buf_sz);
+#else
+
+
+static inline void perf_clang__init(void) { }
+static inline void perf_clang__cleanup(void) { }
+
+static inline int test__clang_to_IR(void) { return -1; }
+static inline int test__clang_to_obj(void) { return -1;}
+
+static inline int
+perf_clang__compile_bpf(const char *filename __maybe_unused,
+                       void **p_obj_buf __maybe_unused,
+                       size_t *p_obj_buf_sz __maybe_unused)
+{
+       return -ENOTSUP;
+}
+
+#endif
+
 #ifdef __cplusplus
 }
 #endif
index 2a1a75df204fb65347ebe5f3130eda387ea97afe..1e974152cac2d3cb0d57156c5d2a9f4c67ed826d 100644 (file)
@@ -163,4 +163,33 @@ void perf_clang__cleanup(void)
        perf::LLVMCtx.reset(nullptr);
        llvm::llvm_shutdown();
 }
+
+int perf_clang__compile_bpf(const char *filename,
+                           void **p_obj_buf,
+                           size_t *p_obj_buf_sz)
+{
+       using namespace perf;
+
+       if (!p_obj_buf || !p_obj_buf_sz)
+               return -EINVAL;
+
+       llvm::opt::ArgStringList CFlags;
+       auto M = getModuleFromSource(std::move(CFlags), filename);
+       if (!M)
+               return  -EINVAL;
+       auto O = getBPFObjectFromModule(&*M);
+       if (!O)
+               return -EINVAL;
+
+       size_t size = O->size_in_bytes();
+       void *buffer;
+
+       buffer = malloc(size);
+       if (!buffer)
+               return -ENOMEM;
+       memcpy(buffer, O->data(), size);
+       *p_obj_buf = buffer;
+       *p_obj_buf_sz = size;
+       return 0;
+}
 }