因此,如果您在启用断言的情况下编译 LLVM,错误会更加清晰,并且它实际上会告诉您需要做什么:
x: .../src/llvm/lib/CodeGen/LLVMTargetMachine.cpp:63:
void llvm::LLVMTargetMachine::initAsmInfo():
Assertion `TmpAsmInfo && "MCAsmInfo not initialized. "
"Make sure you include the correct TargetSelect.h"
"and that InitializeAllTargetMCs() is being invoked!"' failed.
(我添加了一些换行符,因为它打印为单行长行)。
添加完需要的内容后InitializeAllTargetMCs()
在。。。之初main
,我又遇到了一个错误。看看我的编译器的目标文件生成,我“猜测”这是另一个问题InitializeAll*
称呼。稍微测试了一下,结果发现你还需要InitializeAllAsmPrinters();
- 考虑到您想要生成汇编代码,这是有道理的。
我不完全确定如何“查看”代码的结果,但将这两个添加到main
使其运行完成,而不是断言、错误退出或崩溃——这通常是朝着正确方向迈出的良好一步。
所以这就是main
看起来像“我的”代码:
int main() {
constexpr auto testCodeFileName = "test.cpp";
constexpr auto testCode = "int test() { return 2+2; }";
InitializeAllTargetMCs();
InitializeAllAsmPrinters();
// Prepare compilation arguments
vector<const char *> args;
args.push_back(testCodeFileName);
// Prepare DiagnosticEngine
DiagnosticOptions DiagOpts;
TextDiagnosticPrinter *textDiagPrinter =
new clang::TextDiagnosticPrinter(errs(),
&DiagOpts);
IntrusiveRefCntPtr<clang::DiagnosticIDs> pDiagIDs;
DiagnosticsEngine *pDiagnosticsEngine =
new DiagnosticsEngine(pDiagIDs,
&DiagOpts,
textDiagPrinter);
// Initialize CompilerInvocation
CompilerInvocation *CI = new CompilerInvocation();
CompilerInvocation::CreateFromArgs(*CI, &args[0], &args[0] + args.size(), *pDiagnosticsEngine);
// Map code filename to a memoryBuffer
StringRef testCodeData(testCode);
unique_ptr<MemoryBuffer> buffer = MemoryBuffer::getMemBufferCopy(testCodeData);
CI->getPreprocessorOpts().addRemappedFile(testCodeFileName, buffer.get());
// Create and initialize CompilerInstance
CompilerInstance Clang;
Clang.setInvocation(CI);
Clang.createDiagnostics();
// Set target (I guess I can initialize only the BPF target, but I don't know how)
InitializeAllTargets();
const std::shared_ptr<clang::TargetOptions> targetOptions = std::make_shared<clang::TargetOptions>();
targetOptions->Triple = string("bpf");
TargetInfo *pTargetInfo = TargetInfo::CreateTargetInfo(*pDiagnosticsEngine,targetOptions);
Clang.setTarget(pTargetInfo);
// Create and execute action
// CodeGenAction *compilerAction = new EmitLLVMOnlyAction();
CodeGenAction *compilerAction = new EmitAssemblyAction();
Clang.ExecuteAction(*compilerAction);
buffer.release();
}
我强烈建议,如果您想使用 clang&LLVM 进行开发,请构建 Clang&LLVM 的调试版本 - 这将有助于追踪“原因”,并尽早发现问题以及更明显的问题。使用-DCMAKE_BUILD_TYPE=Debug
with cmake
以获得那种味道。
我用于构建 LLVM 和 Clang 的完整脚本:
export CC=clang
export CXX=clang++
cmake -DCMAKE_BUILD_TYPE=Debug -DCMAKE_INSTALL_PREFIX=/usr/local/llvm-debug -DLLVM_TAR
GETS_TO_BUILD=X86 ../llvm
[我使用的是 3.8 的较晚预发布版本来测试这一点,但我非常怀疑它在这方面与 3.7.1 有很大不同]