MayaFlux 0.4.0
Digital-First Multimedia Processing Framework
Loading...
Searching...
No Matches

◆ initialize()

bool Lila::ClangInterpreter::initialize ( bool  skip_host_library_load = false)

Initialize the Clang interpreter.

Parameters
skip_host_library_loadIf true, do not call LoadDynamicLibrary on MayaFluxLib. Pass true when the host process has already loaded MayaFluxLib (attach scenario). Pass false for standalone use (lila_server).
Returns
true on success.

Definition at line 62 of file ClangInterpreter.cpp.

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;
89 LILA_DEBUG(Emitter::INTERPRETER,
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";
94 LILA_DEBUG(Emitter::INTERPRETER,
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";
99 LILA_ERROR(Emitter::INTERPRETER, m_last_error);
100 return false;
101 }
102
103 m_impl->compile_flags.push_back("-I" + pch_dir);
104
105 const std::string& resource_dir = MayaFlux::Platform::SystemConfig::get_clang_resource_dir();
106 if (!resource_dir.empty()) {
107 m_impl->compile_flags.push_back("-resource-dir=" + resource_dir);
108 LILA_DEBUG(Emitter::INTERPRETER,
109 std::string("Using clang resource dir: ") + resource_dir);
110 } else {
111 m_impl->compile_flags.emplace_back("-resource-dir=/usr/lib/clang/21");
112 LILA_WARN(Emitter::INTERPRETER,
113 "Using default clang resource dir: /usr/lib/clang/21");
114 }
115
116 const auto& system_includes = MayaFlux::Platform::SystemConfig::get_system_includes();
117 for (const auto& include : system_includes) {
118 m_impl->compile_flags.push_back("-isystem" + include);
119 }
120
121#ifndef MAYAFLUX_PLATFORM_WINDOWS
122 const auto& eigen_include = MayaFlux::Platform::SystemConfig::get_dep_includes("eigen");
123 if (!eigen_include.empty()) {
124 m_impl->compile_flags.push_back("-isystem" + eigen_include);
125 LILA_DEBUG(Emitter::INTERPRETER,
126 std::string("Added Eigen include path: ") + eigen_include);
127 } else {
128 LILA_WARN(Emitter::INTERPRETER,
129 "Could not find Eigen include path - some features may not work");
130 }
131
132 const auto& freetype_include = MayaFlux::Platform::SystemConfig::get_dep_includes("freetype2");
133 if (!freetype_include.empty()) {
134 m_impl->compile_flags.push_back("-isystem" + freetype_include);
135 LILA_DEBUG(Emitter::INTERPRETER,
136 std::string("Added FreeType include path: ") + freetype_include);
137 } else {
138 LILA_WARN(Emitter::INTERPRETER,
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);
146 LILA_DEBUG(Emitter::INTERPRETER,
147 std::string("Added magic_enum include path: ") + magic_enum_include);
148 } else {
149 LILA_WARN(Emitter::INTERPRETER,
150 "Could not find magic_enum include path - some features may not work");
151 }
152
153#ifdef MAYAFLUX_PLATFORM_MACOS
154 // CRITICAL: JIT uses Homebrew LLVM but needs macOS SDK for system headers
155 // Homebrew LLVM's libc++ requires pthread.h, sched.h, time.h, etc. from SDK
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 {
161 LILA_WARN(Emitter::INTERPRETER,
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());
188 LILA_ERROR(Emitter::INTERPRETER, m_last_error);
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());
195 LILA_ERROR(Emitter::INTERPRETER, m_last_error);
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) {
204 LILA_DEBUG(Emitter::INTERPRETER,
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())) {
213 m_last_error = "Failed to load " + lib_to_load + ": "
214 + llvm::toString(std::move(err));
215 LILA_ERROR(Emitter::INTERPRETER, m_last_error);
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}
#define LILA_WARN(emitter, msg)
#define LILA_ERROR(emitter, msg)
#define LILA_DEBUG(emitter, msg)
#define LILA_INFO(emitter, msg)
std::unique_ptr< Impl > m_impl
Internal implementation details.
static const std::string & get_clang_resource_dir()
static const std::string & find_dep_library(const std::string &library_name, const std::string &prefix="")
static const std::string & get_dep_includes(const std::string &library_name)
static const std::vector< std::string > & get_system_includes()

References MayaFlux::Platform::SystemConfig::find_dep_library(), MayaFlux::Platform::SystemConfig::get_clang_resource_dir(), MayaFlux::Platform::SystemConfig::get_dep_includes(), MayaFlux::Platform::SystemConfig::get_system_includes(), LILA_DEBUG, LILA_ERROR, LILA_INFO, and LILA_WARN.

+ Here is the call graph for this function: