blob: 715ca0a3dee01eef1b365f04c6aa1f99e24525fd [file] [log] [blame]
Wang Nan00b86692016-11-26 07:03:34 +00001/*
2 * llvm C frontend for perf. Support dynamically compile C file
3 *
4 * Inspired by clang example code:
5 * http://llvm.org/svn/llvm-project/cfe/trunk/examples/clang-interpreter/main.cpp
6 *
7 * Copyright (C) 2016 Wang Nan <wangnan0@huawei.com>
8 * Copyright (C) 2016 Huawei Inc.
9 */
10
11#include "clang/CodeGen/CodeGenAction.h"
12#include "clang/Frontend/CompilerInvocation.h"
13#include "clang/Frontend/CompilerInstance.h"
14#include "clang/Frontend/TextDiagnosticPrinter.h"
15#include "clang/Tooling/Tooling.h"
16#include "llvm/IR/Module.h"
17#include "llvm/Option/Option.h"
Wang Nan77dfa842016-11-26 07:03:35 +000018#include "llvm/Support/FileSystem.h"
Wang Nan00b86692016-11-26 07:03:34 +000019#include "llvm/Support/ManagedStatic.h"
20#include <memory>
21
22#include "clang.h"
23#include "clang-c.h"
24
25namespace perf {
26
27static std::unique_ptr<llvm::LLVMContext> LLVMCtx;
28
29using namespace clang;
30
Wang Nan00b86692016-11-26 07:03:34 +000031static CompilerInvocation *
Wang Nana9495fe2016-11-26 07:03:36 +000032createCompilerInvocation(llvm::opt::ArgStringList CFlags, StringRef& Path,
33 DiagnosticsEngine& Diags)
Wang Nan00b86692016-11-26 07:03:34 +000034{
35 llvm::opt::ArgStringList CCArgs {
36 "-cc1",
37 "-triple", "bpf-pc-linux",
38 "-fsyntax-only",
39 "-ferror-limit", "19",
40 "-fmessage-length", "127",
41 "-O2",
42 "-nostdsysteminc",
43 "-nobuiltininc",
44 "-vectorize-loops",
45 "-vectorize-slp",
46 "-Wno-unused-value",
47 "-Wno-pointer-sign",
48 "-x", "c"};
Wang Nana9495fe2016-11-26 07:03:36 +000049
50 CCArgs.append(CFlags.begin(), CFlags.end());
Wang Nan00b86692016-11-26 07:03:34 +000051 CompilerInvocation *CI = tooling::newInvocation(&Diags, CCArgs);
52
53 FrontendOptions& Opts = CI->getFrontendOpts();
54 Opts.Inputs.clear();
55 Opts.Inputs.emplace_back(Path, IK_C);
56 return CI;
57}
58
Wang Nan77dfa842016-11-26 07:03:35 +000059static std::unique_ptr<llvm::Module>
Wang Nana9495fe2016-11-26 07:03:36 +000060getModuleFromSource(llvm::opt::ArgStringList CFlags,
61 StringRef Path, IntrusiveRefCntPtr<vfs::FileSystem> VFS)
Wang Nan00b86692016-11-26 07:03:34 +000062{
63 CompilerInstance Clang;
64 Clang.createDiagnostics();
65
Wang Nan00b86692016-11-26 07:03:34 +000066 Clang.setVirtualFileSystem(&*VFS);
67
68 IntrusiveRefCntPtr<CompilerInvocation> CI =
Wang Nana9495fe2016-11-26 07:03:36 +000069 createCompilerInvocation(std::move(CFlags), Path,
70 Clang.getDiagnostics());
Wang Nan00b86692016-11-26 07:03:34 +000071 Clang.setInvocation(&*CI);
72
73 std::unique_ptr<CodeGenAction> Act(new EmitLLVMOnlyAction(&*LLVMCtx));
74 if (!Clang.ExecuteAction(*Act))
75 return std::unique_ptr<llvm::Module>(nullptr);
76
77 return Act->takeModule();
78}
79
Wang Nan77dfa842016-11-26 07:03:35 +000080std::unique_ptr<llvm::Module>
Wang Nana9495fe2016-11-26 07:03:36 +000081getModuleFromSource(llvm::opt::ArgStringList CFlags,
82 StringRef Name, StringRef Content)
Wang Nan77dfa842016-11-26 07:03:35 +000083{
84 using namespace vfs;
85
86 llvm::IntrusiveRefCntPtr<OverlayFileSystem> OverlayFS(
87 new OverlayFileSystem(getRealFileSystem()));
88 llvm::IntrusiveRefCntPtr<InMemoryFileSystem> MemFS(
89 new InMemoryFileSystem(true));
90
91 /*
92 * pushOverlay helps setting working dir for MemFS. Must call
93 * before addFile.
94 */
95 OverlayFS->pushOverlay(MemFS);
96 MemFS->addFile(Twine(Name), 0, llvm::MemoryBuffer::getMemBuffer(Content));
97
Wang Nana9495fe2016-11-26 07:03:36 +000098 return getModuleFromSource(std::move(CFlags), Name, OverlayFS);
Wang Nan77dfa842016-11-26 07:03:35 +000099}
100
101std::unique_ptr<llvm::Module>
Wang Nana9495fe2016-11-26 07:03:36 +0000102getModuleFromSource(llvm::opt::ArgStringList CFlags, StringRef Path)
Wang Nan77dfa842016-11-26 07:03:35 +0000103{
104 IntrusiveRefCntPtr<vfs::FileSystem> VFS(vfs::getRealFileSystem());
Wang Nana9495fe2016-11-26 07:03:36 +0000105 return getModuleFromSource(std::move(CFlags), Path, VFS);
Wang Nan77dfa842016-11-26 07:03:35 +0000106}
107
Wang Nan00b86692016-11-26 07:03:34 +0000108}
109
110extern "C" {
111void perf_clang__init(void)
112{
113 perf::LLVMCtx.reset(new llvm::LLVMContext());
114}
115
116void perf_clang__cleanup(void)
117{
118 perf::LLVMCtx.reset(nullptr);
119 llvm::llvm_shutdown();
120}
121}