180814287SRaphael Isemann //===-- PlatformAndroid.cpp -----------------------------------------------===// 200e305d2STamas Berghammer // 32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 600e305d2STamas Berghammer // 700e305d2STamas Berghammer //===----------------------------------------------------------------------===// 800e305d2STamas Berghammer 9ec3f92a5STamas Berghammer #include "lldb/Core/Module.h" 1000e305d2STamas Berghammer #include "lldb/Core/PluginManager.h" 11ec3f92a5STamas Berghammer #include "lldb/Core/Section.h" 123cb132a0STamas Berghammer #include "lldb/Core/ValueObject.h" 131c6a1ea9STamas Berghammer #include "lldb/Host/HostInfo.h" 146f9e6901SZachary Turner #include "lldb/Utility/Log.h" 15d821c997SPavel Labath #include "lldb/Utility/Scalar.h" 165f7e583bSPavel Labath #include "lldb/Utility/UriParser.h" 1700e305d2STamas Berghammer 186f001068SOleksiy Vyalov #include "AdbClient.h" 1900e305d2STamas Berghammer #include "PlatformAndroid.h" 2000e305d2STamas Berghammer #include "PlatformAndroidRemoteGDBServer.h" 21f7a5851dSNitesh Jain #include "lldb/Target/Target.h" 2200e305d2STamas Berghammer 2300e305d2STamas Berghammer using namespace lldb; 2400e305d2STamas Berghammer using namespace lldb_private; 25db264a6dSTamas Berghammer using namespace lldb_private::platform_android; 26ce255ff2SPavel Labath using namespace std::chrono; 2700e305d2STamas Berghammer 28bba9ba8dSJonas Devlieghere LLDB_PLUGIN_DEFINE(PlatformAndroid) 29fbb4d1e4SJonas Devlieghere 3000e305d2STamas Berghammer static uint32_t g_initialize_count = 0; 31b9c1b51eSKate Stone static const unsigned int g_android_default_cache_size = 32b9c1b51eSKate Stone 2048; // Fits inside 4k adb packet. 3300e305d2STamas Berghammer 34b9c1b51eSKate Stone void PlatformAndroid::Initialize() { 353c4f89d7STamas Berghammer PlatformLinux::Initialize(); 363c4f89d7STamas Berghammer 37b9c1b51eSKate Stone if (g_initialize_count++ == 0) { 381c6a1ea9STamas Berghammer #if defined(__ANDROID__) 391c6a1ea9STamas Berghammer PlatformSP default_platform_sp(new PlatformAndroid(true)); 401c6a1ea9STamas Berghammer default_platform_sp->SetSystemArchitecture(HostInfo::GetArchitecture()); 411c6a1ea9STamas Berghammer Platform::SetHostPlatform(default_platform_sp); 421c6a1ea9STamas Berghammer #endif 43b9c1b51eSKate Stone PluginManager::RegisterPlugin( 44b9c1b51eSKate Stone PlatformAndroid::GetPluginNameStatic(false), 451c6a1ea9STamas Berghammer PlatformAndroid::GetPluginDescriptionStatic(false), 4600e305d2STamas Berghammer PlatformAndroid::CreateInstance); 4700e305d2STamas Berghammer } 4800e305d2STamas Berghammer } 4900e305d2STamas Berghammer 50b9c1b51eSKate Stone void PlatformAndroid::Terminate() { 51b9c1b51eSKate Stone if (g_initialize_count > 0) { 52b9c1b51eSKate Stone if (--g_initialize_count == 0) { 5300e305d2STamas Berghammer PluginManager::UnregisterPlugin(PlatformAndroid::CreateInstance); 5400e305d2STamas Berghammer } 5500e305d2STamas Berghammer } 563c4f89d7STamas Berghammer 573c4f89d7STamas Berghammer PlatformLinux::Terminate(); 5800e305d2STamas Berghammer } 5900e305d2STamas Berghammer 60b9c1b51eSKate Stone PlatformSP PlatformAndroid::CreateInstance(bool force, const ArchSpec *arch) { 61db264a6dSTamas Berghammer Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM)); 62b9c1b51eSKate Stone if (log) { 6300e305d2STamas Berghammer const char *arch_name; 6400e305d2STamas Berghammer if (arch && arch->GetArchitectureName()) 6500e305d2STamas Berghammer arch_name = arch->GetArchitectureName(); 6600e305d2STamas Berghammer else 6700e305d2STamas Berghammer arch_name = "<null>"; 6800e305d2STamas Berghammer 69b9c1b51eSKate Stone const char *triple_cstr = 70b9c1b51eSKate Stone arch ? arch->GetTriple().getTriple().c_str() : "<null>"; 7100e305d2STamas Berghammer 7263e5fb76SJonas Devlieghere LLDB_LOGF(log, "PlatformAndroid::%s(force=%s, arch={%s,%s})", __FUNCTION__, 73b9c1b51eSKate Stone force ? "true" : "false", arch_name, triple_cstr); 7400e305d2STamas Berghammer } 7500e305d2STamas Berghammer 7600e305d2STamas Berghammer bool create = force; 77a6682a41SJonas Devlieghere if (!create && arch && arch->IsValid()) { 7800e305d2STamas Berghammer const llvm::Triple &triple = arch->GetTriple(); 79b9c1b51eSKate Stone switch (triple.getVendor()) { 8000e305d2STamas Berghammer case llvm::Triple::PC: 8100e305d2STamas Berghammer create = true; 8200e305d2STamas Berghammer break; 8300e305d2STamas Berghammer 8400e305d2STamas Berghammer #if defined(__ANDROID__) 85852e88adSEric Christopher // Only accept "unknown" for the vendor if the host is android and if 8605097246SAdrian Prantl // "unknown" wasn't specified (it was just returned because it was NOT 87852e88adSEric Christopher // specified). 8800e305d2STamas Berghammer case llvm::Triple::VendorType::UnknownVendor: 8900e305d2STamas Berghammer create = !arch->TripleVendorWasSpecified(); 9000e305d2STamas Berghammer break; 9100e305d2STamas Berghammer #endif 9200e305d2STamas Berghammer default: 9300e305d2STamas Berghammer break; 9400e305d2STamas Berghammer } 9500e305d2STamas Berghammer 96b9c1b51eSKate Stone if (create) { 9783c49e8eSEric Christopher switch (triple.getEnvironment()) { 981c6a1ea9STamas Berghammer case llvm::Triple::Android: 9900e305d2STamas Berghammer break; 10000e305d2STamas Berghammer 10100e305d2STamas Berghammer #if defined(__ANDROID__) 10205097246SAdrian Prantl // Only accept "unknown" for the OS if the host is android and it 10305097246SAdrian Prantl // "unknown" wasn't specified (it was just returned because it was NOT 10405097246SAdrian Prantl // specified) 10583c49e8eSEric Christopher case llvm::Triple::EnvironmentType::UnknownEnvironment: 10683c49e8eSEric Christopher create = !arch->TripleEnvironmentWasSpecified(); 10700e305d2STamas Berghammer break; 10800e305d2STamas Berghammer #endif 10900e305d2STamas Berghammer default: 11000e305d2STamas Berghammer create = false; 11100e305d2STamas Berghammer break; 11200e305d2STamas Berghammer } 11300e305d2STamas Berghammer } 11400e305d2STamas Berghammer } 11500e305d2STamas Berghammer 116b9c1b51eSKate Stone if (create) { 11763e5fb76SJonas Devlieghere LLDB_LOGF(log, "PlatformAndroid::%s() creating remote-android platform", 118b9c1b51eSKate Stone __FUNCTION__); 1191c6a1ea9STamas Berghammer return PlatformSP(new PlatformAndroid(false)); 12000e305d2STamas Berghammer } 12100e305d2STamas Berghammer 12263e5fb76SJonas Devlieghere LLDB_LOGF( 12363e5fb76SJonas Devlieghere log, "PlatformAndroid::%s() aborting creation of remote-android platform", 124b9c1b51eSKate Stone __FUNCTION__); 12500e305d2STamas Berghammer 12600e305d2STamas Berghammer return PlatformSP(); 12700e305d2STamas Berghammer } 12800e305d2STamas Berghammer 129b9c1b51eSKate Stone PlatformAndroid::PlatformAndroid(bool is_host) 130b9c1b51eSKate Stone : PlatformLinux(is_host), m_sdk_version(0) {} 13100e305d2STamas Berghammer 132*a458ef4fSPavel Labath llvm::StringRef PlatformAndroid::GetPluginDescriptionStatic(bool is_host) { 1331c6a1ea9STamas Berghammer if (is_host) 1341c6a1ea9STamas Berghammer return "Local Android user platform plug-in."; 13500e305d2STamas Berghammer return "Remote Android user platform plug-in."; 13600e305d2STamas Berghammer } 13700e305d2STamas Berghammer 13897206d57SZachary Turner Status PlatformAndroid::ConnectRemote(Args &args) { 1396f001068SOleksiy Vyalov m_device_id.clear(); 1406f001068SOleksiy Vyalov 141a3939e15SPavel Labath if (IsHost()) 142a3939e15SPavel Labath return Status("can't connect to the host platform, always connected"); 1431c6a1ea9STamas Berghammer 14400e305d2STamas Berghammer if (!m_remote_platform_sp) 14500e305d2STamas Berghammer m_remote_platform_sp = PlatformSP(new PlatformAndroidRemoteGDBServer()); 1466f001068SOleksiy Vyalov 1473ea689b3SChaoren Lin const char *url = args.GetArgumentAtIndex(0); 1483ea689b3SChaoren Lin if (!url) 14997206d57SZachary Turner return Status("URL is null."); 1500e5a4147SMichał Górny llvm::Optional<URI> parsed_url = URI::Parse(url); 1510e5a4147SMichał Górny if (!parsed_url) 15297206d57SZachary Turner return Status("Invalid URL: %s", url); 1530e5a4147SMichał Górny if (parsed_url->hostname != "localhost") 1540e5a4147SMichał Górny m_device_id = parsed_url->hostname.str(); 1553ea689b3SChaoren Lin 1566f001068SOleksiy Vyalov auto error = PlatformLinux::ConnectRemote(args); 157b9c1b51eSKate Stone if (error.Success()) { 158d6a143fcSOleksiy Vyalov AdbClient adb; 159d6a143fcSOleksiy Vyalov error = AdbClient::CreateByDeviceID(m_device_id, adb); 1606f001068SOleksiy Vyalov if (error.Fail()) 1616f001068SOleksiy Vyalov return error; 1626f001068SOleksiy Vyalov 163d6a143fcSOleksiy Vyalov m_device_id = adb.GetDeviceID(); 1646f001068SOleksiy Vyalov } 1656f001068SOleksiy Vyalov return error; 1666f001068SOleksiy Vyalov } 1676f001068SOleksiy Vyalov 16897206d57SZachary Turner Status PlatformAndroid::GetFile(const FileSpec &source, 169b9c1b51eSKate Stone const FileSpec &destination) { 170291f59c5SOleksiy Vyalov if (IsHost() || !m_remote_platform_sp) 1710b5ebef7SOleksiy Vyalov return PlatformLinux::GetFile(source, destination); 172291f59c5SOleksiy Vyalov 1738f3be7a3SJonas Devlieghere FileSpec source_spec(source.GetPath(false), FileSpec::Style::posix); 174372e9067SChaoren Lin if (source_spec.IsRelative()) 175b9c1b51eSKate Stone source_spec = GetRemoteWorkingDirectory().CopyByAppendingPathComponent( 176b9c1b51eSKate Stone source_spec.GetCString(false)); 177291f59c5SOleksiy Vyalov 17897206d57SZachary Turner Status error; 179d6a143fcSOleksiy Vyalov auto sync_service = GetSyncService(error); 180d6a143fcSOleksiy Vyalov if (error.Fail()) 181d6a143fcSOleksiy Vyalov return error; 182d6a143fcSOleksiy Vyalov 183c6ac2e1eSOleksiy Vyalov uint32_t mode = 0, size = 0, mtime = 0; 184c6ac2e1eSOleksiy Vyalov error = sync_service->Stat(source_spec, mode, size, mtime); 185c6ac2e1eSOleksiy Vyalov if (error.Fail()) 186c6ac2e1eSOleksiy Vyalov return error; 187c6ac2e1eSOleksiy Vyalov 188c6ac2e1eSOleksiy Vyalov if (mode != 0) 189d6a143fcSOleksiy Vyalov return sync_service->PullFile(source_spec, destination); 190c6ac2e1eSOleksiy Vyalov 191c6ac2e1eSOleksiy Vyalov auto source_file = source_spec.GetCString(false); 192c6ac2e1eSOleksiy Vyalov 193c6ac2e1eSOleksiy Vyalov Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM)); 19463e5fb76SJonas Devlieghere LLDB_LOGF(log, "Got mode == 0 on '%s': try to get file via 'shell cat'", 195b9c1b51eSKate Stone source_file); 196c6ac2e1eSOleksiy Vyalov 197c6ac2e1eSOleksiy Vyalov if (strchr(source_file, '\'') != nullptr) 19897206d57SZachary Turner return Status("Doesn't support single-quotes in filenames"); 199c6ac2e1eSOleksiy Vyalov 20005097246SAdrian Prantl // mode == 0 can signify that adbd cannot access the file due security 20105097246SAdrian Prantl // constraints - try "cat ..." as a fallback. 202c6ac2e1eSOleksiy Vyalov AdbClient adb(m_device_id); 203c6ac2e1eSOleksiy Vyalov 204c6ac2e1eSOleksiy Vyalov char cmd[PATH_MAX]; 205c6ac2e1eSOleksiy Vyalov snprintf(cmd, sizeof(cmd), "cat '%s'", source_file); 206c6ac2e1eSOleksiy Vyalov 207ce255ff2SPavel Labath return adb.ShellToFile(cmd, minutes(1), destination); 2080b5ebef7SOleksiy Vyalov } 2090b5ebef7SOleksiy Vyalov 21097206d57SZachary Turner Status PlatformAndroid::PutFile(const FileSpec &source, 211b9c1b51eSKate Stone const FileSpec &destination, uint32_t uid, 212b9c1b51eSKate Stone uint32_t gid) { 21317701b59SOleksiy Vyalov if (IsHost() || !m_remote_platform_sp) 21417701b59SOleksiy Vyalov return PlatformLinux::PutFile(source, destination, uid, gid); 21517701b59SOleksiy Vyalov 2168f3be7a3SJonas Devlieghere FileSpec destination_spec(destination.GetPath(false), FileSpec::Style::posix); 217372e9067SChaoren Lin if (destination_spec.IsRelative()) 218b9c1b51eSKate Stone destination_spec = GetRemoteWorkingDirectory().CopyByAppendingPathComponent( 219b9c1b51eSKate Stone destination_spec.GetCString(false)); 22017701b59SOleksiy Vyalov 22162efb1f6SRobert Flack // TODO: Set correct uid and gid on remote file. 22297206d57SZachary Turner Status error; 223d6a143fcSOleksiy Vyalov auto sync_service = GetSyncService(error); 224d6a143fcSOleksiy Vyalov if (error.Fail()) 225d6a143fcSOleksiy Vyalov return error; 226d6a143fcSOleksiy Vyalov return sync_service->PushFile(source, destination_spec); 22762efb1f6SRobert Flack } 22862efb1f6SRobert Flack 229b9c1b51eSKate Stone const char *PlatformAndroid::GetCacheHostname() { return m_device_id.c_str(); } 23009e9079dSOleksiy Vyalov 23197206d57SZachary Turner Status PlatformAndroid::DownloadModuleSlice(const FileSpec &src_file_spec, 23209e9079dSOleksiy Vyalov const uint64_t src_offset, 23309e9079dSOleksiy Vyalov const uint64_t src_size, 234b9c1b51eSKate Stone const FileSpec &dst_file_spec) { 23509e9079dSOleksiy Vyalov if (src_offset != 0) 23697206d57SZachary Turner return Status("Invalid offset - %" PRIu64, src_offset); 23709e9079dSOleksiy Vyalov 2380b5ebef7SOleksiy Vyalov return GetFile(src_file_spec, dst_file_spec); 23909e9079dSOleksiy Vyalov } 2401d0d90b9STamas Berghammer 24197206d57SZachary Turner Status PlatformAndroid::DisconnectRemote() { 24297206d57SZachary Turner Status error = PlatformLinux::DisconnectRemote(); 243b9c1b51eSKate Stone if (error.Success()) { 2441d0d90b9STamas Berghammer m_device_id.clear(); 2451d0d90b9STamas Berghammer m_sdk_version = 0; 2461d0d90b9STamas Berghammer } 2471d0d90b9STamas Berghammer return error; 2481d0d90b9STamas Berghammer } 2491d0d90b9STamas Berghammer 250b9c1b51eSKate Stone uint32_t PlatformAndroid::GetDefaultMemoryCacheLineSize() { 251f2991488SPavel Labath return g_android_default_cache_size; 252f2991488SPavel Labath } 253f2991488SPavel Labath 254b9c1b51eSKate Stone uint32_t PlatformAndroid::GetSdkVersion() { 2551d0d90b9STamas Berghammer if (!IsConnected()) 2561d0d90b9STamas Berghammer return 0; 2571d0d90b9STamas Berghammer 2581d0d90b9STamas Berghammer if (m_sdk_version != 0) 2591d0d90b9STamas Berghammer return m_sdk_version; 2601d0d90b9STamas Berghammer 2611d0d90b9STamas Berghammer std::string version_string; 262d6a143fcSOleksiy Vyalov AdbClient adb(m_device_id); 26397206d57SZachary Turner Status error = 264ce255ff2SPavel Labath adb.Shell("getprop ro.build.version.sdk", seconds(5), &version_string); 2653e8947beSTamas Berghammer version_string = llvm::StringRef(version_string).trim().str(); 2663e8947beSTamas Berghammer 267b9c1b51eSKate Stone if (error.Fail() || version_string.empty()) { 2681d0d90b9STamas Berghammer Log *log = GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM); 26963e5fb76SJonas Devlieghere LLDB_LOGF(log, "Get SDK version failed. (error: %s, output: %s)", 2703e8947beSTamas Berghammer error.AsCString(), version_string.c_str()); 2711d0d90b9STamas Berghammer return 0; 2721d0d90b9STamas Berghammer } 2731d0d90b9STamas Berghammer 2743a6ba367SMichał Górny // FIXME: improve error handling 2753a6ba367SMichał Górny llvm::to_integer(version_string, m_sdk_version); 2761d0d90b9STamas Berghammer return m_sdk_version; 2771d0d90b9STamas Berghammer } 278ec3f92a5STamas Berghammer 27997206d57SZachary Turner Status PlatformAndroid::DownloadSymbolFile(const lldb::ModuleSP &module_sp, 280b9c1b51eSKate Stone const FileSpec &dst_file_spec) { 281ec3f92a5STamas Berghammer // For oat file we can try to fetch additional debug info from the device 282dba6503aSTamas Berghammer ConstString extension = module_sp->GetFileSpec().GetFileNameExtension(); 28305cfdb0eSRaphael Isemann if (extension != ".oat" && extension != ".odex") 28497206d57SZachary Turner return Status( 285b9c1b51eSKate Stone "Symbol file downloading only supported for oat and odex files"); 286ec3f92a5STamas Berghammer 287ec3f92a5STamas Berghammer // If we have no information about the platform file we can't execute oatdump 288ec3f92a5STamas Berghammer if (!module_sp->GetPlatformFileSpec()) 28997206d57SZachary Turner return Status("No platform file specified"); 290ec3f92a5STamas Berghammer 291ec3f92a5STamas Berghammer // Symbolizer isn't available before SDK version 23 292ec3f92a5STamas Berghammer if (GetSdkVersion() < 23) 29397206d57SZachary Turner return Status("Symbol file generation only supported on SDK 23+"); 294ec3f92a5STamas Berghammer 295ec3f92a5STamas Berghammer // If we already have symtab then we don't have to try and generate one 296b9c1b51eSKate Stone if (module_sp->GetSectionList()->FindSectionByName(ConstString(".symtab")) != 297b9c1b51eSKate Stone nullptr) 29897206d57SZachary Turner return Status("Symtab already available in the module"); 299ec3f92a5STamas Berghammer 300d6a143fcSOleksiy Vyalov AdbClient adb(m_device_id); 3019d8dde8cSTamas Berghammer std::string tmpdir; 30297206d57SZachary Turner Status error = adb.Shell("mktemp --directory --tmpdir /data/local/tmp", 303ce255ff2SPavel Labath seconds(5), &tmpdir); 3049d8dde8cSTamas Berghammer if (error.Fail() || tmpdir.empty()) 30597206d57SZachary Turner return Status("Failed to generate temporary directory on the device (%s)", 306b9c1b51eSKate Stone error.AsCString()); 307539cd118STamas Berghammer tmpdir = llvm::StringRef(tmpdir).trim().str(); 308ec3f92a5STamas Berghammer 309ec3f92a5STamas Berghammer // Create file remover for the temporary directory created on the device 310b9c1b51eSKate Stone std::unique_ptr<std::string, std::function<void(std::string *)>> 3113bc714b2SZachary Turner tmpdir_remover(&tmpdir, [&adb](std::string *s) { 312ec3f92a5STamas Berghammer StreamString command; 313ec3f92a5STamas Berghammer command.Printf("rm -rf %s", s->c_str()); 31497206d57SZachary Turner Status error = adb.Shell(command.GetData(), seconds(5), nullptr); 315ec3f92a5STamas Berghammer 316ec3f92a5STamas Berghammer Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM)); 317dd778ca6SOleksiy Vyalov if (log && error.Fail()) 31863e5fb76SJonas Devlieghere LLDB_LOGF(log, "Failed to remove temp directory: %s", error.AsCString()); 319b9c1b51eSKate Stone }); 320ec3f92a5STamas Berghammer 3218f3be7a3SJonas Devlieghere FileSpec symfile_platform_filespec(tmpdir); 322ec3f92a5STamas Berghammer symfile_platform_filespec.AppendPathComponent("symbolized.oat"); 323ec3f92a5STamas Berghammer 324ec3f92a5STamas Berghammer // Execute oatdump on the remote device to generate a file with symtab 3259d8dde8cSTamas Berghammer StreamString command; 326ec3f92a5STamas Berghammer command.Printf("oatdump --symbolize=%s --output=%s", 327ec3f92a5STamas Berghammer module_sp->GetPlatformFileSpec().GetCString(false), 328ec3f92a5STamas Berghammer symfile_platform_filespec.GetCString(false)); 329ce255ff2SPavel Labath error = adb.Shell(command.GetData(), minutes(1), nullptr); 3309d8dde8cSTamas Berghammer if (error.Fail()) 33197206d57SZachary Turner return Status("Oatdump failed: %s", error.AsCString()); 332ec3f92a5STamas Berghammer 333ec3f92a5STamas Berghammer // Download the symbolfile from the remote device 334ec3f92a5STamas Berghammer return GetFile(symfile_platform_filespec, dst_file_spec); 335ec3f92a5STamas Berghammer } 3363c47151dSTamas Berghammer 337b9c1b51eSKate Stone bool PlatformAndroid::GetRemoteOSVersion() { 3382272c481SPavel Labath m_os_version = llvm::VersionTuple(GetSdkVersion()); 3392272c481SPavel Labath return !m_os_version.empty(); 3403c47151dSTamas Berghammer } 3413cb132a0STamas Berghammer 342f7a5851dSNitesh Jain llvm::StringRef 343f7a5851dSNitesh Jain PlatformAndroid::GetLibdlFunctionDeclarations(lldb_private::Process *process) { 344f7a5851dSNitesh Jain SymbolContextList matching_symbols; 345f7a5851dSNitesh Jain std::vector<const char *> dl_open_names = { "__dl_dlopen", "dlopen" }; 346f7a5851dSNitesh Jain const char *dl_open_name = nullptr; 347f7a5851dSNitesh Jain Target &target = process->GetTarget(); 348f7a5851dSNitesh Jain for (auto name: dl_open_names) { 3491ad655e2SAdrian Prantl target.GetImages().FindFunctionSymbols( 3501ad655e2SAdrian Prantl ConstString(name), eFunctionNameTypeFull, matching_symbols); 3511ad655e2SAdrian Prantl if (matching_symbols.GetSize()) { 352f7a5851dSNitesh Jain dl_open_name = name; 353f7a5851dSNitesh Jain break; 354f7a5851dSNitesh Jain } 355f7a5851dSNitesh Jain } 356d37c9469SPavel Labath // Older platform versions have the dl function symbols mangled 357f7a5851dSNitesh Jain if (dl_open_name == dl_open_names[0]) 358fcdde63fSTamas Berghammer return R"( 359fcdde63fSTamas Berghammer extern "C" void* dlopen(const char*, int) asm("__dl_dlopen"); 360fcdde63fSTamas Berghammer extern "C" void* dlsym(void*, const char*) asm("__dl_dlsym"); 361fcdde63fSTamas Berghammer extern "C" int dlclose(void*) asm("__dl_dlclose"); 362fcdde63fSTamas Berghammer extern "C" char* dlerror(void) asm("__dl_dlerror"); 3633cb132a0STamas Berghammer )"; 364d37c9469SPavel Labath 365f7a5851dSNitesh Jain return PlatformPOSIX::GetLibdlFunctionDeclarations(process); 3663cb132a0STamas Berghammer } 367d6a143fcSOleksiy Vyalov 36897206d57SZachary Turner AdbClient::SyncService *PlatformAndroid::GetSyncService(Status &error) { 369d6a143fcSOleksiy Vyalov if (m_adb_sync_svc && m_adb_sync_svc->IsConnected()) 370d6a143fcSOleksiy Vyalov return m_adb_sync_svc.get(); 371d6a143fcSOleksiy Vyalov 372d6a143fcSOleksiy Vyalov AdbClient adb(m_device_id); 373d6a143fcSOleksiy Vyalov m_adb_sync_svc = adb.GetSyncService(error); 374d6a143fcSOleksiy Vyalov return (error.Success()) ? m_adb_sync_svc.get() : nullptr; 375d6a143fcSOleksiy Vyalov } 376