1a7a03d64SLang Hames //===----- LLJITDumpObjects.cpp - How to dump JIT'd objects with LLJIT ----===//
2a7a03d64SLang Hames //
3a7a03d64SLang Hames // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4a7a03d64SLang Hames // See https://llvm.org/LICENSE.txt for license information.
5a7a03d64SLang Hames // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6a7a03d64SLang Hames //
7a7a03d64SLang Hames //===----------------------------------------------------------------------===//
8a7a03d64SLang Hames ///
9a7a03d64SLang Hames /// \file
10a7a03d64SLang Hames ///
11a7a03d64SLang Hames /// This example demonstrates how to use LLJIT to:
12a7a03d64SLang Hames ///   - Add absolute symbol definitions.
13a7a03d64SLang Hames ///   - Run static constructors for a JITDylib.
14a7a03d64SLang Hames ///   - Run static destructors for a JITDylib.
15a7a03d64SLang Hames ///
16a7a03d64SLang Hames /// This example does not call any functions (e.g. main or equivalent)  between
17a7a03d64SLang Hames /// running the static constructors and running the static destructors.
18a7a03d64SLang Hames ///
19a7a03d64SLang Hames //===----------------------------------------------------------------------===//
20a7a03d64SLang Hames 
21a7a03d64SLang Hames #include "llvm/ADT/StringMap.h"
22a7a03d64SLang Hames #include "llvm/ExecutionEngine/Orc/LLJIT.h"
23a7a03d64SLang Hames #include "llvm/Support/InitLLVM.h"
24a7a03d64SLang Hames #include "llvm/Support/TargetSelect.h"
25a7a03d64SLang Hames #include "llvm/Support/raw_ostream.h"
26a7a03d64SLang Hames 
27a7a03d64SLang Hames #include "../ExampleModules.h"
28a7a03d64SLang Hames 
29a7a03d64SLang Hames using namespace llvm;
30a7a03d64SLang Hames using namespace llvm::orc;
31a7a03d64SLang Hames 
32a7a03d64SLang Hames ExitOnError ExitOnErr;
33a7a03d64SLang Hames 
34a7a03d64SLang Hames const llvm::StringRef ModuleWithInitializer =
35a7a03d64SLang Hames     R"(
36a7a03d64SLang Hames 
37a7a03d64SLang Hames   @InitializersRunFlag = external global i32
38a7a03d64SLang Hames   @DeinitializersRunFlag = external global i32
39a7a03d64SLang Hames 
40a7a03d64SLang Hames   declare i32 @__cxa_atexit(void (i8*)*, i8*, i8*)
41a7a03d64SLang Hames   @__dso_handle = external hidden global i8
42a7a03d64SLang Hames 
43a7a03d64SLang Hames   @llvm.global_ctors =
44a7a03d64SLang Hames     appending global [1 x { i32, void ()*, i8* }]
45a7a03d64SLang Hames       [{ i32, void ()*, i8* } { i32 65535, void ()* @init_func, i8* null }]
46a7a03d64SLang Hames 
47a7a03d64SLang Hames   define internal void @init_func() {
48a7a03d64SLang Hames   entry:
49a7a03d64SLang Hames     store i32 1, i32* @InitializersRunFlag
50a7a03d64SLang Hames     %0 = call i32 @__cxa_atexit(void (i8*)* @deinit_func, i8* null,
51a7a03d64SLang Hames                                 i8* @__dso_handle)
52a7a03d64SLang Hames     ret void
53a7a03d64SLang Hames   }
54a7a03d64SLang Hames 
55a7a03d64SLang Hames   define internal void @deinit_func(i8* %0) {
56a7a03d64SLang Hames     store i32 1, i32* @DeinitializersRunFlag
57a7a03d64SLang Hames     ret void
58a7a03d64SLang Hames   }
59a7a03d64SLang Hames 
60a7a03d64SLang Hames )";
61a7a03d64SLang Hames 
main(int argc,char * argv[])62a7a03d64SLang Hames int main(int argc, char *argv[]) {
63a7a03d64SLang Hames   // Initialize LLVM.
64a7a03d64SLang Hames   InitLLVM X(argc, argv);
65a7a03d64SLang Hames 
66a7a03d64SLang Hames   InitializeNativeTarget();
67a7a03d64SLang Hames   InitializeNativeTargetAsmPrinter();
68a7a03d64SLang Hames 
69a7a03d64SLang Hames   cl::ParseCommandLineOptions(argc, argv, "LLJITWithInitializers");
70a7a03d64SLang Hames   ExitOnErr.setBanner(std::string(argv[0]) + ": ");
71a7a03d64SLang Hames 
72a7a03d64SLang Hames   auto J = ExitOnErr(LLJITBuilder().create());
73a7a03d64SLang Hames   auto M = ExitOnErr(parseExampleModule(ModuleWithInitializer, "M"));
74a7a03d64SLang Hames 
75a7a03d64SLang Hames   // Load the module.
76a7a03d64SLang Hames   ExitOnErr(J->addIRModule(std::move(M)));
77a7a03d64SLang Hames 
78a7a03d64SLang Hames   int32_t InitializersRunFlag = 0;
79a7a03d64SLang Hames   int32_t DeinitializersRunFlag = 0;
80a7a03d64SLang Hames 
81*91d1f417SLang Hames   ExitOnErr(J->getMainJITDylib().define(absoluteSymbols(
82a7a03d64SLang Hames       {{J->mangleAndIntern("InitializersRunFlag"),
83a7a03d64SLang Hames         JITEvaluatedSymbol::fromPointer(&InitializersRunFlag)},
84a7a03d64SLang Hames        {J->mangleAndIntern("DeinitializersRunFlag"),
85a7a03d64SLang Hames         JITEvaluatedSymbol::fromPointer(&DeinitializersRunFlag)}})));
86a7a03d64SLang Hames 
87a7a03d64SLang Hames   // Run static initializers.
88a7a03d64SLang Hames   ExitOnErr(J->initialize(J->getMainJITDylib()));
89a7a03d64SLang Hames 
90a7a03d64SLang Hames   // Run deinitializers.
91a7a03d64SLang Hames   ExitOnErr(J->deinitialize(J->getMainJITDylib()));
92a7a03d64SLang Hames 
93a7a03d64SLang Hames   outs() << "InitializerRanFlag = " << InitializersRunFlag << "\n"
94a7a03d64SLang Hames          << "DeinitializersRunFlag = " << DeinitializersRunFlag << "\n";
95a7a03d64SLang Hames 
96a7a03d64SLang Hames   return 0;
97a7a03d64SLang Hames }
98