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 const std::string magic_enum_include = std::string(MayaFlux::Config::MAGIC_ENUM_INCLUDE);
144 if (!magic_enum_include.empty()) {
145 m_impl->compile_flags.push_back(
"-I" + magic_enum_include);
147 std::string("Added magic_enum include path: ") + magic_enum_include);
148 } else {
150 "Could not find magic_enum include path - some features may not work");
151 }
152
153#ifdef MAYAFLUX_PLATFORM_MACOS
154
155
156 std::string sdk_path = MayaFlux::Platform::SystemConfig::get_macos_sdk_path();
157 if (!sdk_path.empty()) {
158 m_impl->compile_flags.push_back(
"-isysroot" + sdk_path);
159 LILA_DEBUG(Emitter::INTERPRETER,
"Using macOS SDK: " + sdk_path);
160 } else {
162 "Could not find macOS SDK - JIT may fail to find system headers");
163 }
164#endif
165
166 for (
const auto& path :
m_impl->include_paths) {
167 m_impl->compile_flags.push_back(
"-I" + path);
168 }
169
170#ifdef MAYAFLUX_PLATFORM_WINDOWS
171 m_impl->compile_flags.emplace_back(
"-fno-function-sections");
172 m_impl->compile_flags.emplace_back(
"-fno-data-sections");
173 m_impl->compile_flags.emplace_back(
"-fno-unique-section-names");
174 m_impl->compile_flags.emplace_back(
"-D_CRT_SECURE_NO_WARNINGS");
175#endif
176
177 std::vector<const char*> args;
178 for (
const auto& flag :
m_impl->compile_flags) {
179 args.push_back(flag.c_str());
180 }
181
182 clang::IncrementalCompilerBuilder ICB;
183 ICB.SetCompilerArgs(args);
184
185 auto CI = ICB.CreateCpp();
186 if (!CI) {
187 m_last_error =
"Failed to create CompilerInstance: " + llvm::toString(CI.takeError());
189 return false;
190 }
191
192 auto interp = clang::Interpreter::create(std::move(*CI));
193 if (!interp) {
194 m_last_error =
"Failed to create interpreter: " + llvm::toString(interp.takeError());
196 return false;
197 }
198
199 m_impl->interpreter = std::move(*interp);
200
201 LILA_INFO(Emitter::INTERPRETER,
"Clang interpreter created successfully");
202
203 if (skip_host_library_load) {
205 "Skipping MayaFluxLib load: host process already resident");
206 } else {
208 "MayaFluxLib", std::string(MayaFlux::Config::INSTALL_PREFIX));
209
210 LILA_DEBUG(Emitter::INTERPRETER,
"Loading MayaFlux library: " + lib_to_load);
211
212 if (
auto err =
m_impl->interpreter->LoadDynamicLibrary(lib_to_load.c_str())) {
214 + llvm::toString(std::move(err));
216 return false;
217 }
218
219 LILA_DEBUG(Emitter::INTERPRETER,
"Loaded " + lib_to_load);
220 }
221
222 const char* header_include = "#include \"pch.h\"\n"
223 "#include \"MayaFlux/MayaFlux.hpp\"\n";
224
225 auto result =
m_impl->interpreter->ParseAndExecute(header_include);
226
227 if (result) {
228 std::string warning = "Failed to load MayaFlux headers: " + llvm::toString(std::move(result));
229 LILA_WARN(Emitter::INTERPRETER, warning);
230 } else {
231 LILA_INFO(Emitter::INTERPRETER,
"MayaFlux headers loaded successfully");
232 }
233
234 result =
m_impl->interpreter->ParseAndExecute(
"std::cout << \"Ready for Live\" << std::flush;");
235
236 return true;
237}
std::unique_ptr< Impl > m_impl
Internal implementation details.