Initialize the Clang interpreter.
63{
64 LILA_INFO(Emitter::INTERPRETER,
"Initializing Clang interpreter");
65
66#ifdef MAYAFLUX_PLATFORM_WINDOWS
67 llvm::sys::DynamicLibrary::LoadLibraryPermanently("msvcp140.dll");
68 llvm::sys::DynamicLibrary::LoadLibraryPermanently("vcruntime140.dll");
69 llvm::sys::DynamicLibrary::LoadLibraryPermanently("ucrtbase.dll");
70#endif
71
72 llvm::InitializeNativeTarget();
73 llvm::InitializeNativeTargetAsmPrinter();
74 llvm::InitializeNativeTargetAsmParser();
75
76 m_impl->compile_flags.clear();
77 m_impl->compile_flags.emplace_back(
"-std=c++23");
78 m_impl->compile_flags.emplace_back(
"-DMAYASIMPLE");
79
80#ifdef MAYAFLUX_PLATFORM_LINUX
81 m_impl->compile_flags.emplace_back(
"-mcmodel=large");
82 m_impl->compile_flags.emplace_back(
"-fPIC");
83 m_impl->compile_flags.emplace_back(
"-fPIE");
84#endif
85
86 std::string pch_dir;
87 if (std::filesystem::exists(MayaFlux::Config::PCH_RUNTIME_PATH)) {
88 pch_dir = MayaFlux::Config::RUNTIME_DATA_DIR;
90 std::string("Using installed PCH from: ") + std::string(MayaFlux::Config::PCH_RUNTIME_PATH));
91
92 } else if (std::filesystem::exists(MayaFlux::Config::PCH_SOURCE_PATH)) {
93 pch_dir = std::string(MayaFlux::Config::SOURCE_DIR) + "/cmake";
95 std::string("Using source PCH from: ") + std::string(MayaFlux::Config::PCH_SOURCE_PATH));
96
97 } else {
98 m_last_error =
"Cannot find pch.h in runtime or source locations";
100 return false;
101 }
102
103 m_impl->compile_flags.push_back(
"-I" + pch_dir);
104
106 if (!resource_dir.empty()) {
107 m_impl->compile_flags.push_back(
"-resource-dir=" + resource_dir);
109 std::string("Using clang resource dir: ") + resource_dir);
110 } else {
111 m_impl->compile_flags.emplace_back(
"-resource-dir=/usr/lib/clang/21");
113 "Using default clang resource dir: /usr/lib/clang/21");
114 }
115
117 for (const auto& include : system_includes) {
118 m_impl->compile_flags.push_back(
"-isystem" + include);
119 }
120
121#ifndef MAYAFLUX_PLATFORM_WINDOWS
123 if (!eigen_include.empty()) {
124 m_impl->compile_flags.push_back(
"-isystem" + eigen_include);
126 std::string("Added Eigen include path: ") + eigen_include);
127 } else {
129 "Could not find Eigen include path - some features may not work");
130 }
131
133 if (!freetype_include.empty()) {
134 m_impl->compile_flags.push_back(
"-isystem" + freetype_include);
136 std::string("Added FreeType include path: ") + freetype_include);
137 } else {
139 "Could not find FreeType include path - some features may not work");
140 }
141#endif
142
143#ifdef MAYAFLUX_PLATFORM_MACOS
144
145
146 std::string sdk_path = MayaFlux::Platform::SystemConfig::get_macos_sdk_path();
147 if (!sdk_path.empty()) {
148 m_impl->compile_flags.push_back(
"-isysroot" + sdk_path);
149 LILA_DEBUG(Emitter::INTERPRETER,
"Using macOS SDK: " + sdk_path);
150 } else {
152 "Could not find macOS SDK - JIT may fail to find system headers");
153 }
154#endif
155
156 for (
const auto& path :
m_impl->include_paths) {
157 m_impl->compile_flags.push_back(
"-I" + path);
158 }
159
160#ifdef MAYAFLUX_PLATFORM_WINDOWS
161 m_impl->compile_flags.emplace_back(
"-fno-function-sections");
162 m_impl->compile_flags.emplace_back(
"-fno-data-sections");
163 m_impl->compile_flags.emplace_back(
"-fno-unique-section-names");
164#endif
165
166 std::vector<const char*> args;
167 for (
const auto& flag :
m_impl->compile_flags) {
168 args.push_back(flag.c_str());
169 }
170
171 clang::IncrementalCompilerBuilder ICB;
172 ICB.SetCompilerArgs(args);
173
174 auto CI = ICB.CreateCpp();
175 if (!CI) {
176 m_last_error =
"Failed to create CompilerInstance: " + llvm::toString(CI.takeError());
178 return false;
179 }
180
181 auto interp = clang::Interpreter::create(std::move(*CI));
182 if (!interp) {
183 m_last_error =
"Failed to create interpreter: " + llvm::toString(interp.takeError());
185 return false;
186 }
187
188 m_impl->interpreter = std::move(*interp);
189
190 LILA_INFO(Emitter::INTERPRETER,
"Clang interpreter created successfully");
191
192 if (skip_host_library_load) {
194 "Skipping MayaFluxLib load: host process already resident");
195 } else {
197 "MayaFluxLib", std::string(MayaFlux::Config::INSTALL_PREFIX));
198
199 LILA_DEBUG(Emitter::INTERPRETER,
"Loading MayaFlux library: " + lib_to_load);
200
201 if (
auto err =
m_impl->interpreter->LoadDynamicLibrary(lib_to_load.c_str())) {
203 + llvm::toString(std::move(err));
205 return false;
206 }
207
208 LILA_DEBUG(Emitter::INTERPRETER,
"Loaded " + lib_to_load);
209 }
210
211 const char* header_include = skip_host_library_load
212 ? "#include \"pch.h\"\n"
213 "#include \"MayaFlux/MayaFlux.hpp\"\n"
214 : "#include \"pch.h\"\n"
215 "#include \"Lila/LiveAid.hpp\"\n";
216
217 auto result =
m_impl->interpreter->ParseAndExecute(header_include);
218
219 if (result) {
220 std::string warning = "Failed to load MayaFlux headers: " + llvm::toString(std::move(result));
221 LILA_WARN(Emitter::INTERPRETER, warning);
222 } else {
223 LILA_INFO(Emitter::INTERPRETER,
"MayaFlux headers loaded successfully");
224 }
225
226 result =
m_impl->interpreter->ParseAndExecute(
"std::cout << \"Ready for Live\" << std::flush;");
227
228 return true;
229}
std::unique_ptr< Impl > m_impl
Internal implementation details.