1 //===-- ModuleCache.cpp ---------------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "lldb/Target/ModuleCache.h"
11 #include "lldb/Core/Module.h"
12 #include "lldb/Core/ModuleList.h"
13 #include "lldb/Core/ModuleSpec.h"
14 #include "lldb/Host/File.h"
15 #include "lldb/Host/LockFile.h"
16 #include "lldb/Utility/LLDBLog.h"
17 #include "lldb/Utility/Log.h"
18 #include "llvm/Support/FileSystem.h"
19 #include "llvm/Support/FileUtilities.h"
26 using namespace lldb_private
;
30 const char *kModulesSubdir
= ".cache";
31 const char *kLockDirName
= ".lock";
32 const char *kTempFileName
= ".temp";
33 const char *kTempSymFileName
= ".symtemp";
34 const char *kSymFileExtension
= ".sym";
35 const char *kFSIllegalChars
= "\\/:*?\"<>|";
37 std::string
GetEscapedHostname(const char *hostname
) {
38 if (hostname
== nullptr)
40 std::string
result(hostname
);
41 size_t size
= result
.size();
42 for (size_t i
= 0; i
< size
; ++i
) {
43 if ((result
[i
] >= 1 && result
[i
] <= 31) ||
44 strchr(kFSIllegalChars
, result
[i
]) != nullptr)
53 std::unique_ptr
<lldb_private::LockFile
> m_lock
;
57 ModuleLock(const FileSpec
&root_dir_spec
, const UUID
&uuid
, Status
&error
);
61 static FileSpec
JoinPath(const FileSpec
&path1
, const char *path2
) {
62 FileSpec
result_spec(path1
);
63 result_spec
.AppendPathComponent(path2
);
67 static Status
MakeDirectory(const FileSpec
&dir_path
) {
68 namespace fs
= llvm::sys::fs
;
70 return fs::create_directories(dir_path
.GetPath(), true, fs::perms::owner_all
);
73 FileSpec
GetModuleDirectory(const FileSpec
&root_dir_spec
, const UUID
&uuid
) {
74 const auto modules_dir_spec
= JoinPath(root_dir_spec
, kModulesSubdir
);
75 return JoinPath(modules_dir_spec
, uuid
.GetAsString().c_str());
78 FileSpec
GetSymbolFileSpec(const FileSpec
&module_file_spec
) {
79 return FileSpec(module_file_spec
.GetPath() + kSymFileExtension
);
82 void DeleteExistingModule(const FileSpec
&root_dir_spec
,
83 const FileSpec
&sysroot_module_path_spec
) {
84 Log
*log
= GetLog(LLDBLog::Modules
);
88 std::make_shared
<Module
>(ModuleSpec(sysroot_module_path_spec
));
89 module_uuid
= module_sp
->GetUUID();
92 if (!module_uuid
.IsValid())
96 ModuleLock
lock(root_dir_spec
, module_uuid
, error
);
98 LLDB_LOGF(log
, "Failed to lock module %s: %s",
99 module_uuid
.GetAsString().c_str(), error
.AsCString());
102 namespace fs
= llvm::sys::fs
;
104 if (status(sysroot_module_path_spec
.GetPath(), st
))
107 if (st
.getLinkCount() > 2) // module is referred by other hosts.
110 const auto module_spec_dir
= GetModuleDirectory(root_dir_spec
, module_uuid
);
111 llvm::sys::fs::remove_directories(module_spec_dir
.GetPath());
115 void DecrementRefExistingModule(const FileSpec
&root_dir_spec
,
116 const FileSpec
&sysroot_module_path_spec
) {
117 // Remove $platform/.cache/$uuid folder if nobody else references it.
118 DeleteExistingModule(root_dir_spec
, sysroot_module_path_spec
);
120 // Remove sysroot link.
121 llvm::sys::fs::remove(sysroot_module_path_spec
.GetPath());
123 FileSpec symfile_spec
= GetSymbolFileSpec(sysroot_module_path_spec
);
124 llvm::sys::fs::remove(symfile_spec
.GetPath());
127 Status
CreateHostSysRootModuleLink(const FileSpec
&root_dir_spec
,
128 const char *hostname
,
129 const FileSpec
&platform_module_spec
,
130 const FileSpec
&local_module_spec
,
131 bool delete_existing
) {
132 const auto sysroot_module_path_spec
=
133 JoinPath(JoinPath(root_dir_spec
, hostname
),
134 platform_module_spec
.GetPath().c_str());
135 if (FileSystem::Instance().Exists(sysroot_module_path_spec
)) {
136 if (!delete_existing
)
139 DecrementRefExistingModule(root_dir_spec
, sysroot_module_path_spec
);
142 const auto error
= MakeDirectory(
143 FileSpec(sysroot_module_path_spec
.GetDirectory().AsCString()));
147 return llvm::sys::fs::create_hard_link(local_module_spec
.GetPath(),
148 sysroot_module_path_spec
.GetPath());
153 ModuleLock::ModuleLock(const FileSpec
&root_dir_spec
, const UUID
&uuid
,
155 const auto lock_dir_spec
= JoinPath(root_dir_spec
, kLockDirName
);
156 error
= MakeDirectory(lock_dir_spec
);
160 m_file_spec
= JoinPath(lock_dir_spec
, uuid
.GetAsString().c_str());
162 auto file
= FileSystem::Instance().Open(
163 m_file_spec
, File::eOpenOptionWriteOnly
| File::eOpenOptionCanCreate
|
164 File::eOpenOptionCloseOnExec
);
166 m_file_up
= std::move(file
.get());
169 error
= Status(file
.takeError());
173 m_lock
= std::make_unique
<lldb_private::LockFile
>(m_file_up
->GetDescriptor());
174 error
= m_lock
->WriteLock(0, 1);
176 error
.SetErrorStringWithFormat("Failed to lock file: %s",
180 void ModuleLock::Delete() {
186 llvm::sys::fs::remove(m_file_spec
.GetPath());
189 /////////////////////////////////////////////////////////////////////////
191 Status
ModuleCache::Put(const FileSpec
&root_dir_spec
, const char *hostname
,
192 const ModuleSpec
&module_spec
, const FileSpec
&tmp_file
,
193 const FileSpec
&target_file
) {
194 const auto module_spec_dir
=
195 GetModuleDirectory(root_dir_spec
, module_spec
.GetUUID());
196 const auto module_file_path
=
197 JoinPath(module_spec_dir
, target_file
.GetFilename().AsCString());
199 const auto tmp_file_path
= tmp_file
.GetPath();
200 const auto err_code
=
201 llvm::sys::fs::rename(tmp_file_path
, module_file_path
.GetPath());
203 return Status("Failed to rename file %s to %s: %s", tmp_file_path
.c_str(),
204 module_file_path
.GetPath().c_str(),
205 err_code
.message().c_str());
207 const auto error
= CreateHostSysRootModuleLink(
208 root_dir_spec
, hostname
, target_file
, module_file_path
, true);
210 return Status("Failed to create link to %s: %s",
211 module_file_path
.GetPath().c_str(), error
.AsCString());
215 Status
ModuleCache::Get(const FileSpec
&root_dir_spec
, const char *hostname
,
216 const ModuleSpec
&module_spec
,
217 ModuleSP
&cached_module_sp
, bool *did_create_ptr
) {
219 m_loaded_modules
.find(module_spec
.GetUUID().GetAsString());
220 if (find_it
!= m_loaded_modules
.end()) {
221 cached_module_sp
= (*find_it
).second
.lock();
222 if (cached_module_sp
)
224 m_loaded_modules
.erase(find_it
);
227 const auto module_spec_dir
=
228 GetModuleDirectory(root_dir_spec
, module_spec
.GetUUID());
229 const auto module_file_path
= JoinPath(
230 module_spec_dir
, module_spec
.GetFileSpec().GetFilename().AsCString());
232 if (!FileSystem::Instance().Exists(module_file_path
))
233 return Status("Module %s not found", module_file_path
.GetPath().c_str());
234 if (FileSystem::Instance().GetByteSize(module_file_path
) !=
235 module_spec
.GetObjectSize())
236 return Status("Module %s has invalid file size",
237 module_file_path
.GetPath().c_str());
239 // We may have already cached module but downloaded from an another host - in
240 // this case let's create a link to it.
241 auto error
= CreateHostSysRootModuleLink(root_dir_spec
, hostname
,
242 module_spec
.GetFileSpec(),
243 module_file_path
, false);
245 return Status("Failed to create link to %s: %s",
246 module_file_path
.GetPath().c_str(), error
.AsCString());
248 auto cached_module_spec(module_spec
);
249 cached_module_spec
.GetUUID().Clear(); // Clear UUID since it may contain md5
250 // content hash instead of real UUID.
251 cached_module_spec
.GetFileSpec() = module_file_path
;
252 cached_module_spec
.GetPlatformFileSpec() = module_spec
.GetFileSpec();
254 error
= ModuleList::GetSharedModule(cached_module_spec
, cached_module_sp
,
255 nullptr, nullptr, did_create_ptr
, false);
259 FileSpec symfile_spec
= GetSymbolFileSpec(cached_module_sp
->GetFileSpec());
260 if (FileSystem::Instance().Exists(symfile_spec
))
261 cached_module_sp
->SetSymbolFileFileSpec(symfile_spec
);
263 m_loaded_modules
.insert(
264 std::make_pair(module_spec
.GetUUID().GetAsString(), cached_module_sp
));
269 Status
ModuleCache::GetAndPut(const FileSpec
&root_dir_spec
,
270 const char *hostname
,
271 const ModuleSpec
&module_spec
,
272 const ModuleDownloader
&module_downloader
,
273 const SymfileDownloader
&symfile_downloader
,
274 lldb::ModuleSP
&cached_module_sp
,
275 bool *did_create_ptr
) {
276 const auto module_spec_dir
=
277 GetModuleDirectory(root_dir_spec
, module_spec
.GetUUID());
278 auto error
= MakeDirectory(module_spec_dir
);
282 ModuleLock
lock(root_dir_spec
, module_spec
.GetUUID(), error
);
284 return Status("Failed to lock module %s: %s",
285 module_spec
.GetUUID().GetAsString().c_str(),
288 const auto escaped_hostname(GetEscapedHostname(hostname
));
289 // Check local cache for a module.
290 error
= Get(root_dir_spec
, escaped_hostname
.c_str(), module_spec
,
291 cached_module_sp
, did_create_ptr
);
295 const auto tmp_download_file_spec
= JoinPath(module_spec_dir
, kTempFileName
);
296 error
= module_downloader(module_spec
, tmp_download_file_spec
);
297 llvm::FileRemover
tmp_file_remover(tmp_download_file_spec
.GetPath());
299 return Status("Failed to download module: %s", error
.AsCString());
301 // Put downloaded file into local module cache.
302 error
= Put(root_dir_spec
, escaped_hostname
.c_str(), module_spec
,
303 tmp_download_file_spec
, module_spec
.GetFileSpec());
305 return Status("Failed to put module into cache: %s", error
.AsCString());
307 tmp_file_remover
.releaseFile();
308 error
= Get(root_dir_spec
, escaped_hostname
.c_str(), module_spec
,
309 cached_module_sp
, did_create_ptr
);
313 // Fetching a symbol file for the module
314 const auto tmp_download_sym_file_spec
=
315 JoinPath(module_spec_dir
, kTempSymFileName
);
316 error
= symfile_downloader(cached_module_sp
, tmp_download_sym_file_spec
);
317 llvm::FileRemover
tmp_symfile_remover(tmp_download_sym_file_spec
.GetPath());
319 // Failed to download a symfile but fetching the module was successful. The
320 // module might contain the necessary symbols and the debugging is also
321 // possible without a symfile.
324 error
= Put(root_dir_spec
, escaped_hostname
.c_str(), module_spec
,
325 tmp_download_sym_file_spec
,
326 GetSymbolFileSpec(module_spec
.GetFileSpec()));
328 return Status("Failed to put symbol file into cache: %s",
331 tmp_symfile_remover
.releaseFile();
333 FileSpec symfile_spec
= GetSymbolFileSpec(cached_module_sp
->GetFileSpec());
334 cached_module_sp
->SetSymbolFileFileSpec(symfile_spec
);