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