33#include "llvm/ADT/SmallSet.h"
34#include "llvm/Support/FileSystem.h"
35#include "llvm/Support/ThreadPool.h"
42#include "mach/machine.h"
44#include <CoreFoundation/CoreFoundation.h>
57 CFUUIDRef uuid, CFURLRef exec_url) =
nullptr;
77 return "DebugSymbols symbol locator.";
88 LLDB_LOGF(log,
"Spotlight lookup for .dSYM bundles is disabled.");
92 return_module_spec = module_spec;
103 void *handle = dlopen(
104 "/System/Library/PrivateFrameworks/DebugSymbols.framework/DebugSymbols",
105 RTLD_LAZY | RTLD_LOCAL);
108 (CFURLRef(*)(CFUUIDRef, CFURLRef))dlsym(handle,
109 "DBGCopyFullDSYMURLForUUID");
111 handle,
"DBGCopyDSYMPropertyLists");
122 llvm::ArrayRef<uint8_t> module_uuid = uuid->
GetBytes();
123 if (module_uuid.size() == 16) {
125 NULL, module_uuid[0], module_uuid[1], module_uuid[2], module_uuid[3],
126 module_uuid[4], module_uuid[5], module_uuid[6], module_uuid[7],
127 module_uuid[8], module_uuid[9], module_uuid[10], module_uuid[11],
128 module_uuid[12], module_uuid[13], module_uuid[14], module_uuid[15]));
130 if (module_uuid_ref.
get()) {
135 if (exec_fspec->
GetPath(exec_cf_path,
sizeof(exec_cf_path)))
136 exec_url.
reset(::CFURLCreateFromFileSystemRepresentation(
137 NULL, (
const UInt8 *)exec_cf_path, strlen(exec_cf_path),
142 module_uuid_ref.
get(), exec_url.
get()));
145 if (dsym_url.
get()) {
146 if (::CFURLGetFileSystemRepresentation(
147 dsym_url.
get(),
true, (UInt8 *)path,
sizeof(path) - 1)) {
149 "DebugSymbols framework returned dSYM path of %s for "
150 "UUID %s -- looking for the dSYM",
158 dsym_filespec, uuid, arch);
166 bool success =
false;
168 if (::CFURLGetFileSystemRepresentation(
169 dsym_url.
get(),
true, (UInt8 *)path,
sizeof(path) - 1)) {
171 "DebugSymbols framework returned dSYM path of %s for "
172 "UUID %s -- looking for an exec file",
179 CFDictionaryRef uuid_dict = NULL;
182 uuid_dict =
static_cast<CFDictionaryRef
>(
183 ::CFDictionaryGetValue(dict.
get(), uuid_cfstr.
get()));
192 module_sp = std::make_shared<Module>(exe_spec);
193 if (module_sp && module_sp->GetObjectFile() &&
194 module_sp->MatchesModuleSpec(exe_spec)) {
197 LLDB_LOGF(log,
"using original binary filepath %s for UUID %s",
220 "using binary from shared cache for filepath %s for "
229 if (!success && uuid_dict) {
232 uuid_dict, CFSTR(
"DBGSymbolRichExecutable")));
233 if (exec_cf_path && ::CFStringGetFileSystemRepresentation(
234 exec_cf_path, path,
sizeof(path))) {
235 LLDB_LOGF(log,
"plist bundle has exec path of %s for UUID %s",
250 if (::CFURLGetFileSystemRepresentation(
251 dsym_url.
get(),
true, (UInt8 *)path,
sizeof(path) - 1)) {
252 char *dsym_extension_pos = ::strstr(path,
".dSYM");
253 if (dsym_extension_pos) {
254 *dsym_extension_pos =
'\0';
256 "Looking for executable binary next to dSYM "
257 "bundle with name with name %s",
263 using namespace llvm::sys::fs;
264 switch (get_file_type(file_spec.
GetPath())) {
266 case file_type::directory_file:
271 if (bundle_exe_url.
get()) {
272 if (::CFURLGetFileSystemRepresentation(bundle_exe_url.
get(),
275 FileSpec bundle_exe_file_spec(path);
278 bundle_exe_file_spec, 0, 0, module_specs) &&
280 module_spec, matched_module_spec))
284 return_module_spec.
GetFileSpec() = bundle_exe_file_spec;
286 "Executable binary %s next to dSYM is "
294 case file_type::fifo_file:
295 case file_type::socket_file:
296 case file_type::file_not_found:
297 case file_type::status_error:
300 case file_type::type_unknown:
301 case file_type::regular_file:
302 case file_type::symlink_file:
303 case file_type::block_file:
304 case file_type::character_file:
308 matched_module_spec))
314 "Executable binary %s next to dSYM is "
329 return return_module_spec;
336 std::string dsym_bundle_path = dsym_bundle_fspec.
GetPath();
337 llvm::SmallString<128> buffer(dsym_bundle_path);
338 llvm::sys::path::append(buffer,
"Contents",
"Resources",
"DWARF");
341 llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> vfs =
343 llvm::vfs::recursive_directory_iterator Iter(*vfs, buffer.str(), EC);
344 llvm::vfs::recursive_directory_iterator End;
345 for (; Iter != End && !EC; Iter.increment(EC)) {
346 llvm::ErrorOr<llvm::vfs::Status>
Status = vfs->status(Iter->path());
347 if (
Status->isDirectory())
354 for (
size_t i = 0; i < module_specs.
GetSize(); ++i) {
358 if ((uuid ==
nullptr ||
378 for (
size_t i = 0; i < module_specs.
GetSize(); ++i) {
403 FileSpec dsym_directory = exec_fspec;
406 std::string dsym_filename = filename.
AsCString();
407 dsym_filename +=
".dSYM";
417 dsym_fspec = dsym_directory;
429 std::string binary_name(filename.
AsCString());
430 auto last_dot = binary_name.find_last_of(
'.');
431 if (last_dot != std::string::npos) {
432 binary_name.erase(last_dot);
433 dsym_fspec = dsym_directory;
445 FileSpec dsym_yaa_fspec = exec_fspec;
447 std::string dsym_yaa_filename = filename.
AsCString();
448 dsym_yaa_filename +=
".dSYM.yaa";
484 LLDB_LOGF(log,
"dSYM with matching UUID & arch found at %s",
506 for (
int i = 0; i < 4; i++) {
512 if (::strchr(fn,
'.') !=
nullptr) {
516 LLDB_LOGF(log,
"dSYM with matching UUID & arch found at %s",
534 LLDB_LOGF(log,
"Spotlight lookup for .dSYM bundles is disabled.");
538 return_module_spec = module_spec;
549 void *handle = dlopen(
550 "/System/Library/PrivateFrameworks/DebugSymbols.framework/DebugSymbols",
551 RTLD_LAZY | RTLD_LOCAL);
554 (CFURLRef(*)(CFUUIDRef, CFURLRef))dlsym(handle,
555 "DBGCopyFullDSYMURLForUUID");
557 handle,
"DBGCopyDSYMPropertyLists");
568 llvm::ArrayRef<uint8_t> module_uuid = uuid->
GetBytes();
569 if (module_uuid.size() == 16) {
571 NULL, module_uuid[0], module_uuid[1], module_uuid[2], module_uuid[3],
572 module_uuid[4], module_uuid[5], module_uuid[6], module_uuid[7],
573 module_uuid[8], module_uuid[9], module_uuid[10], module_uuid[11],
574 module_uuid[12], module_uuid[13], module_uuid[14], module_uuid[15]));
576 if (module_uuid_ref.
get()) {
581 if (exec_fspec->
GetPath(exec_cf_path,
sizeof(exec_cf_path)))
582 exec_url.
reset(::CFURLCreateFromFileSystemRepresentation(
583 NULL, (
const UInt8 *)exec_cf_path, strlen(exec_cf_path),
588 module_uuid_ref.
get(), exec_url.
get()));
591 if (dsym_url.
get()) {
592 if (::CFURLGetFileSystemRepresentation(
593 dsym_url.
get(),
true, (UInt8 *)path,
sizeof(path) - 1)) {
595 "DebugSymbols framework returned dSYM path of %s for "
596 "UUID %s -- looking for the dSYM",
604 dsym_filespec, uuid, arch);
612 bool success =
false;
614 if (::CFURLGetFileSystemRepresentation(
615 dsym_url.
get(),
true, (UInt8 *)path,
sizeof(path) - 1)) {
617 "DebugSymbols framework returned dSYM path of %s for "
618 "UUID %s -- looking for an exec file",
625 CFDictionaryRef uuid_dict = NULL;
628 uuid_dict =
static_cast<CFDictionaryRef
>(
629 ::CFDictionaryGetValue(dict.
get(), uuid_cfstr.
get()));
638 module_sp = std::make_shared<Module>(exe_spec);
639 if (module_sp && module_sp->GetObjectFile() &&
640 module_sp->MatchesModuleSpec(exe_spec)) {
643 LLDB_LOGF(log,
"using original binary filepath %s for UUID %s",
666 "using binary from shared cache for filepath %s for "
675 if (!success && uuid_dict) {
678 uuid_dict, CFSTR(
"DBGSymbolRichExecutable")));
679 if (exec_cf_path && ::CFStringGetFileSystemRepresentation(
680 exec_cf_path, path,
sizeof(path))) {
681 LLDB_LOGF(log,
"plist bundle has exec path of %s for UUID %s",
696 if (::CFURLGetFileSystemRepresentation(
697 dsym_url.
get(),
true, (UInt8 *)path,
sizeof(path) - 1)) {
698 char *dsym_extension_pos = ::strstr(path,
".dSYM");
699 if (dsym_extension_pos) {
700 *dsym_extension_pos =
'\0';
702 "Looking for executable binary next to dSYM "
703 "bundle with name with name %s",
709 using namespace llvm::sys::fs;
710 switch (get_file_type(file_spec.
GetPath())) {
712 case file_type::directory_file:
717 if (bundle_exe_url.
get()) {
718 if (::CFURLGetFileSystemRepresentation(bundle_exe_url.
get(),
721 FileSpec bundle_exe_file_spec(path);
724 bundle_exe_file_spec, 0, 0, module_specs) &&
726 module_spec, matched_module_spec))
730 return_module_spec.
GetFileSpec() = bundle_exe_file_spec;
732 "Executable binary %s next to dSYM is "
740 case file_type::fifo_file:
741 case file_type::socket_file:
742 case file_type::file_not_found:
743 case file_type::status_error:
746 case file_type::type_unknown:
747 case file_type::regular_file:
748 case file_type::symlink_file:
749 case file_type::block_file:
750 case file_type::character_file:
754 matched_module_spec))
760 "Executable binary %s next to dSYM is "
784 "LocateExecutableSymbolFileDsym (file = %s, arch = %s, uuid = %p)",
789 "Locating external symbol file",
809 const std::string &command) {
811 bool success =
false;
812 if (uuid_dict != NULL && CFGetTypeID(uuid_dict) == CFDictionaryGetTypeID()) {
815 CFDictionaryRef cf_dict;
817 cf_str = (
CFStringRef)CFDictionaryGetValue((CFDictionaryRef)uuid_dict,
819 if (cf_str && CFGetTypeID(cf_str) == CFStringGetTypeID()) {
821 std::string errorstr = command;
829 (CFDictionaryRef)uuid_dict, CFSTR(
"DBGSymbolRichExecutable"));
830 if (cf_str && CFGetTypeID(cf_str) == CFStringGetTypeID()) {
835 "From dsymForUUID plist: Symbol rich executable is at '%s'",
840 cf_str = (
CFStringRef)CFDictionaryGetValue((CFDictionaryRef)uuid_dict,
841 CFSTR(
"DBGDSYMPath"));
842 if (cf_str && CFGetTypeID(cf_str) == CFStringGetTypeID()) {
845 FileSpec::Style::native);
848 LLDB_LOGF(log,
"From dsymForUUID plist: dSYM is at '%s'", str.c_str());
852 std::string DBGBuildSourcePath;
853 std::string DBGSourcePath;
862 cf_dict = (CFDictionaryRef)CFDictionaryGetValue(
863 (CFDictionaryRef)uuid_dict, CFSTR(
"DBGSourcePathRemapping"));
864 if (cf_dict && CFGetTypeID(cf_dict) == CFDictionaryGetTypeID()) {
867 bool new_style_source_remapping_dictionary =
false;
868 bool do_truncate_remapping_names =
false;
869 std::string original_DBGSourcePath_value = DBGSourcePath;
870 cf_str = (
CFStringRef)CFDictionaryGetValue((CFDictionaryRef)uuid_dict,
871 CFSTR(
"DBGVersion"));
872 if (cf_str && CFGetTypeID(cf_str) == CFStringGetTypeID()) {
875 if (!version.empty() && isdigit(version[0])) {
876 int version_number = atoi(version.c_str());
877 if (version_number > 1) {
878 new_style_source_remapping_dictionary =
true;
880 if (version_number == 2) {
881 do_truncate_remapping_names =
true;
886 CFIndex kv_pair_count = CFDictionaryGetCount((CFDictionaryRef)uuid_dict);
887 if (kv_pair_count > 0) {
892 if (keys !=
nullptr && values !=
nullptr) {
893 CFDictionaryGetKeysAndValues((CFDictionaryRef)uuid_dict,
895 (
const void **)values);
897 for (CFIndex i = 0; i < kv_pair_count; i++) {
898 DBGBuildSourcePath.clear();
899 DBGSourcePath.clear();
900 if (keys[i] && CFGetTypeID(keys[i]) == CFStringGetTypeID()) {
903 if (values[i] && CFGetTypeID(values[i]) == CFStringGetTypeID()) {
906 if (!DBGBuildSourcePath.empty() && !DBGSourcePath.empty()) {
911 if (new_style_source_remapping_dictionary &&
912 !original_DBGSourcePath_value.empty()) {
913 DBGSourcePath = original_DBGSourcePath_value;
915 if (DBGSourcePath[0] ==
'~') {
916 FileSpec resolved_source_path(DBGSourcePath.c_str());
918 DBGSourcePath = resolved_source_path.
GetPath();
925 DBGSourcePath,
true);
926 if (do_truncate_remapping_names) {
927 FileSpec build_path(DBGBuildSourcePath.c_str());
928 FileSpec source_path(DBGSourcePath.c_str());
948 cf_str = (
CFStringRef)CFDictionaryGetValue((CFDictionaryRef)uuid_dict,
949 CFSTR(
"DBGBuildSourcePath"));
950 if (cf_str && CFGetTypeID(cf_str) == CFStringGetTypeID()) {
954 cf_str = (
CFStringRef)CFDictionaryGetValue((CFDictionaryRef)uuid_dict,
955 CFSTR(
"DBGSourcePath"));
956 if (cf_str && CFGetTypeID(cf_str) == CFStringGetTypeID()) {
960 if (!DBGBuildSourcePath.empty() && !DBGSourcePath.empty()) {
961 if (DBGSourcePath[0] ==
'~') {
962 FileSpec resolved_source_path(DBGSourcePath.c_str());
964 DBGSourcePath = resolved_source_path.
GetPath();
967 DBGSourcePath,
true);
976 static std::once_flag g_once_flag;
977 static std::string g_dbgshell_command;
978 std::call_once(g_once_flag, [&]() {
979 CFTypeRef defaults_setting = CFPreferencesCopyAppValue(
980 CFSTR(
"DBGShellCommands"), CFSTR(
"com.apple.DebugSymbols"));
981 if (defaults_setting &&
982 CFGetTypeID(defaults_setting) == CFStringGetTypeID()) {
984 if (CFStringGetCString((
CFStringRef)defaults_setting, buffer,
985 sizeof(buffer), kCFStringEncodingUTF8)) {
986 g_dbgshell_command = buffer;
989 if (defaults_setting) {
990 CFRelease(defaults_setting);
993 return g_dbgshell_command;
1002 if (
const char *dsymForUUID_env =
1003 getenv(
"LLDB_APPLE_DSYMFORUUID_EXECUTABLE")) {
1004 FileSpec dsymForUUID_executable(dsymForUUID_env);
1007 return dsymForUUID_executable;
1010 static std::once_flag g_once_flag;
1011 static FileSpec g_dsymForUUID_executable;
1012 std::call_once(g_once_flag, [&]() {
1015 if (!dbgshell_command.empty()) {
1016 g_dsymForUUID_executable =
FileSpec(dbgshell_command);
1024 g_dsymForUUID_executable =
FileSpec(
"/usr/local/bin/dsymForUUID");
1030 g_dsymForUUID_executable = {};
1032 return g_dsymForUUID_executable;
1037 bool copy_executable) {
1048 if (!force_lookup && dbgshell_command.empty())
1058 if (!dsymForUUID_exe_spec)
1062 const std::string dsymForUUID_exe_path = dsymForUUID_exe_spec.
GetPath();
1063 const std::string uuid_str = uuid_ptr ? uuid_ptr->
GetAsString() :
"";
1065 std::string lookup_arg = uuid_str;
1066 if (lookup_arg.empty())
1067 lookup_arg = file_spec_ptr ? file_spec_ptr->
GetPath() :
"";
1068 if (lookup_arg.empty())
1072 command << dsymForUUID_exe_path <<
" --ignoreNegativeCache ";
1073 if (copy_executable)
1074 command <<
"--copyExecutable ";
1075 command << lookup_arg;
1078 std::string lookup_desc;
1079 if (uuid_ptr && file_spec_ptr)
1085 else if (file_spec_ptr)
1089 LLDB_LOG(log,
"Calling {0} for {1} to find dSYM: {2}", dsymForUUID_exe_path,
1092 Progress progress(
"Downloading symbol file for", lookup_desc);
1095 int exit_status = -1;
1097 std::string command_output;
1098 std::string error_output;
1106 std::chrono::seconds(
1110 if (
error.Fail() || exit_status != 0 || command_output.empty()) {
1112 "'%s' failed (exit status: %d, error: '%s', stdout: '%s', "
1115 command_output.c_str(), error_output.c_str());
1120 CFDataCreateWithBytesNoCopy(NULL, (
const UInt8 *)command_output.data(),
1121 command_output.size(), kCFAllocatorNull));
1124 (CFDictionaryRef)::CFPropertyListCreateWithData(
1125 NULL, data.
get(), kCFPropertyListImmutable, NULL, NULL));
1128 LLDB_LOGF(log,
"'%s' failed: output is not a valid plist",
1130 LLDB_LOGF(log,
"Response:\n%s\n", command_output.c_str());
1134 if (CFGetTypeID(plist.
get()) != CFDictionaryGetTypeID()) {
1135 LLDB_LOGF(log,
"'%s' failed: output plist is not a valid CFDictionary",
1140 if (!uuid_str.empty()) {
1142 CFDictionaryRef uuid_dict =
1143 (CFDictionaryRef)CFDictionaryGetValue(plist.
get(), uuid_cfstr.
get());
1148 if (
const CFIndex num_values = ::CFDictionaryGetCount(plist.
get())) {
1149 std::vector<CFStringRef> keys(num_values, NULL);
1150 std::vector<CFDictionaryRef> values(num_values, NULL);
1151 ::CFDictionaryGetKeysAndValues(plist.
get(), NULL,
1152 (
const void **)&values[0]);
1153 if (num_values == 1) {
1158 for (CFIndex i = 0; i < num_values; ++i) {
1164 module_spec = curr_module_spec;
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
#define LLDB_LOGF(log,...)
#define LLDB_PLUGIN_DEFINE(PluginName)
static llvm::StringRef GetDbgShellCommand()
It's expensive to check for the DBGShellCommands defaults setting.
static CFDictionaryRef(* g_dlsym_DBGCopyDSYMPropertyLists)(CFURLRef dsym_url)
static bool LocateDSYMInVincinityOfExecutable(const ModuleSpec &module_spec, FileSpec &dsym_fspec)
static int LocateMacOSXFilesUsingDebugSymbols(const ModuleSpec &module_spec, ModuleSpec &return_module_spec)
static bool FileAtPathContainsArchAndUUID(const FileSpec &file_fspec, const ArchSpec *arch, const lldb_private::UUID *uuid)
static bool LookForDsymNextToExecutablePath(const ModuleSpec &mod_spec, const FileSpec &exec_fspec, FileSpec &dsym_fspec)
static CFURLRef(* g_dlsym_DBGCopyFullDSYMURLForUUID)(CFUUIDRef uuid, CFURLRef exec_url)
static FileSpec GetDsymForUUIDExecutable()
Get the dsymForUUID executable and cache the result so we don't end up stat'ing the binary over and o...
static bool GetModuleSpecInfoFromUUIDDictionary(CFDictionaryRef uuid_dict, ModuleSpec &module_spec, Status &error, const std::string &command)
#define LLDB_SCOPED_TIMERF(...)
CFURLRef CopyExecutableURL() const
static const char * FileSystemRepresentation(CFStringRef cf_str, std::string &str)
An architecture specification class.
bool IsCompatibleMatch(const ArchSpec &rhs) const
Shorthand for IsMatch(rhs, CompatibleMatch).
const char * GetArchitectureName() const
Returns a static string representing the current architecture.
A uniqued constant string class.
std::string GetString() const
Get the string value as a std::string.
const char * AsCString(const char *value_if_empty=nullptr) const
Get the string value as a C string.
void SetFile(llvm::StringRef path, Style style)
Change the file specified with a new path.
void AppendPathComponent(llvm::StringRef component)
const ConstString & GetFilename() const
Filename string const get accessor.
bool RemoveLastPathComponent()
Removes the last path component by replacing the current path with its parent.
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
void Clear()
Clears the object state.
ConstString GetPathAsConstString(bool denormalize=true) const
Get the full path as a ConstString.
llvm::IntrusiveRefCntPtr< llvm::vfs::FileSystem > GetVirtualFileSystem()
static FileSystem & Instance()
void Resolve(llvm::SmallVectorImpl< char > &path, bool force_make_absolute=false)
Resolve path to make it canonical.
static Status RunShellCommand(llvm::StringRef command, const FileSpec &working_dir, int *status_ptr, int *signo_ptr, std::string *command_output, std::string *error_output, const Timeout< std::micro > &timeout, bool run_in_shell=true)
Run a shell command.
lldb::SymbolSharedCacheUse GetSharedCacheBinaryLoading() const
static ModuleListProperties & GetGlobalModuleListProperties()
bool GetModuleSpecAtIndex(size_t i, ModuleSpec &module_spec) const
bool FindMatchingModuleSpec(const ModuleSpec &module_spec, ModuleSpec &match_module_spec) const
PathMappingList & GetSourceMappingList() const
ArchSpec & GetArchitecture()
FileSpec * GetFileSpecPtr()
FileSpec & GetSymbolFileSpec()
ArchSpec * GetArchitecturePtr()
static size_t GetModuleSpecifications(const FileSpec &file, lldb::offset_t file_offset, lldb::offset_t file_size, ModuleSpecList &specs, lldb::DataExtractorSP=lldb::DataExtractorSP())
void Append(llvm::StringRef path, llvm::StringRef replacement, bool notify)
static bool DownloadObjectAndSymbolFile(ModuleSpec &module_spec, Status &error, bool force_lookup=true, bool copy_executable=true)
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
static FileSpec FindSymbolFileInBundle(const FileSpec &dsym_bundle_fspec, const UUID *uuid, const ArchSpec *arch)
A Progress indicator helper class.
const char * GetData() const
llvm::StringRef GetString() const
SymbolLocatorDebugSymbols()
static llvm::StringRef GetPluginNameStatic()
static std::optional< ModuleSpec > LocateExecutableObjectFile(const ModuleSpec &module_spec)
static lldb_private::SymbolLocator * CreateInstance()
static llvm::StringRef GetPluginDescriptionStatic()
static std::optional< FileSpec > LocateExecutableSymbolFile(const ModuleSpec &module_spec, const FileSpecList &default_search_paths)
static bool DownloadObjectAndSymbolFile(ModuleSpec &module_spec, Status &error, bool force_lookup, bool copy_executable)
static std::optional< FileSpec > FindSymbolFileInBundle(const FileSpec &dsym_bundle_fspec, const UUID *uuid, const ArchSpec *arch)
Represents UUID's of various sizes.
llvm::ArrayRef< uint8_t > GetBytes() const
std::string GetAsString(llvm::StringRef separator="-") const
#define UNUSED_IF_ASSERT_DISABLED(x)
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
std::shared_ptr< lldb_private::Module > ModuleSP
const UUID & GetUUID() const