100e305d2STamas Berghammer //===-- PlatformAndroid.cpp -------------------------------------*- C++ -*-===//
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"
141d0d90b9STamas Berghammer #include "lldb/Host/StringConvert.h"
156f9e6901SZachary Turner #include "lldb/Utility/Log.h"
16d821c997SPavel Labath #include "lldb/Utility/Scalar.h"
175f7e583bSPavel Labath #include "lldb/Utility/UriParser.h"
1800e305d2STamas Berghammer 
196f001068SOleksiy Vyalov #include "AdbClient.h"
2000e305d2STamas Berghammer #include "PlatformAndroid.h"
2100e305d2STamas Berghammer #include "PlatformAndroidRemoteGDBServer.h"
22f7a5851dSNitesh Jain #include "lldb/Target/Target.h"
2300e305d2STamas Berghammer 
2400e305d2STamas Berghammer using namespace lldb;
2500e305d2STamas Berghammer using namespace lldb_private;
26db264a6dSTamas Berghammer using namespace lldb_private::platform_android;
27ce255ff2SPavel Labath using namespace std::chrono;
2800e305d2STamas Berghammer 
2900e305d2STamas Berghammer static uint32_t g_initialize_count = 0;
30b9c1b51eSKate Stone static const unsigned int g_android_default_cache_size =
31b9c1b51eSKate Stone     2048; // Fits inside 4k adb packet.
3200e305d2STamas Berghammer 
33b9c1b51eSKate Stone void PlatformAndroid::Initialize() {
343c4f89d7STamas Berghammer   PlatformLinux::Initialize();
353c4f89d7STamas Berghammer 
36b9c1b51eSKate Stone   if (g_initialize_count++ == 0) {
371c6a1ea9STamas Berghammer #if defined(__ANDROID__)
381c6a1ea9STamas Berghammer     PlatformSP default_platform_sp(new PlatformAndroid(true));
391c6a1ea9STamas Berghammer     default_platform_sp->SetSystemArchitecture(HostInfo::GetArchitecture());
401c6a1ea9STamas Berghammer     Platform::SetHostPlatform(default_platform_sp);
411c6a1ea9STamas Berghammer #endif
42b9c1b51eSKate Stone     PluginManager::RegisterPlugin(
43b9c1b51eSKate Stone         PlatformAndroid::GetPluginNameStatic(false),
441c6a1ea9STamas Berghammer         PlatformAndroid::GetPluginDescriptionStatic(false),
4500e305d2STamas Berghammer         PlatformAndroid::CreateInstance);
4600e305d2STamas Berghammer   }
4700e305d2STamas Berghammer }
4800e305d2STamas Berghammer 
49b9c1b51eSKate Stone void PlatformAndroid::Terminate() {
50b9c1b51eSKate Stone   if (g_initialize_count > 0) {
51b9c1b51eSKate Stone     if (--g_initialize_count == 0) {
5200e305d2STamas Berghammer       PluginManager::UnregisterPlugin(PlatformAndroid::CreateInstance);
5300e305d2STamas Berghammer     }
5400e305d2STamas Berghammer   }
553c4f89d7STamas Berghammer 
563c4f89d7STamas Berghammer   PlatformLinux::Terminate();
5700e305d2STamas Berghammer }
5800e305d2STamas Berghammer 
59b9c1b51eSKate Stone PlatformSP PlatformAndroid::CreateInstance(bool force, const ArchSpec *arch) {
60db264a6dSTamas Berghammer   Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM));
61b9c1b51eSKate Stone   if (log) {
6200e305d2STamas Berghammer     const char *arch_name;
6300e305d2STamas Berghammer     if (arch && arch->GetArchitectureName())
6400e305d2STamas Berghammer       arch_name = arch->GetArchitectureName();
6500e305d2STamas Berghammer     else
6600e305d2STamas Berghammer       arch_name = "<null>";
6700e305d2STamas Berghammer 
68b9c1b51eSKate Stone     const char *triple_cstr =
69b9c1b51eSKate Stone         arch ? arch->GetTriple().getTriple().c_str() : "<null>";
7000e305d2STamas Berghammer 
71b9c1b51eSKate Stone     log->Printf("PlatformAndroid::%s(force=%s, arch={%s,%s})", __FUNCTION__,
72b9c1b51eSKate Stone                 force ? "true" : "false", arch_name, triple_cstr);
7300e305d2STamas Berghammer   }
7400e305d2STamas Berghammer 
7500e305d2STamas Berghammer   bool create = force;
76a6682a41SJonas Devlieghere   if (!create && arch && arch->IsValid()) {
7700e305d2STamas Berghammer     const llvm::Triple &triple = arch->GetTriple();
78b9c1b51eSKate Stone     switch (triple.getVendor()) {
7900e305d2STamas Berghammer     case llvm::Triple::PC:
8000e305d2STamas Berghammer       create = true;
8100e305d2STamas Berghammer       break;
8200e305d2STamas Berghammer 
8300e305d2STamas Berghammer #if defined(__ANDROID__)
84852e88adSEric Christopher     // Only accept "unknown" for the vendor if the host is android and if
8505097246SAdrian Prantl     // "unknown" wasn't specified (it was just returned because it was NOT
86852e88adSEric Christopher     // specified).
8700e305d2STamas Berghammer     case llvm::Triple::VendorType::UnknownVendor:
8800e305d2STamas Berghammer       create = !arch->TripleVendorWasSpecified();
8900e305d2STamas Berghammer       break;
9000e305d2STamas Berghammer #endif
9100e305d2STamas Berghammer     default:
9200e305d2STamas Berghammer       break;
9300e305d2STamas Berghammer     }
9400e305d2STamas Berghammer 
95b9c1b51eSKate Stone     if (create) {
9683c49e8eSEric Christopher       switch (triple.getEnvironment()) {
971c6a1ea9STamas Berghammer       case llvm::Triple::Android:
9800e305d2STamas Berghammer         break;
9900e305d2STamas Berghammer 
10000e305d2STamas Berghammer #if defined(__ANDROID__)
10105097246SAdrian Prantl       // Only accept "unknown" for the OS if the host is android and it
10205097246SAdrian Prantl       // "unknown" wasn't specified (it was just returned because it was NOT
10305097246SAdrian Prantl       // specified)
10483c49e8eSEric Christopher       case llvm::Triple::EnvironmentType::UnknownEnvironment:
10583c49e8eSEric Christopher         create = !arch->TripleEnvironmentWasSpecified();
10600e305d2STamas Berghammer         break;
10700e305d2STamas Berghammer #endif
10800e305d2STamas Berghammer       default:
10900e305d2STamas Berghammer         create = false;
11000e305d2STamas Berghammer         break;
11100e305d2STamas Berghammer       }
11200e305d2STamas Berghammer     }
11300e305d2STamas Berghammer   }
11400e305d2STamas Berghammer 
115b9c1b51eSKate Stone   if (create) {
11600e305d2STamas Berghammer     if (log)
117b9c1b51eSKate Stone       log->Printf("PlatformAndroid::%s() creating remote-android platform",
118b9c1b51eSKate Stone                   __FUNCTION__);
1191c6a1ea9STamas Berghammer     return PlatformSP(new PlatformAndroid(false));
12000e305d2STamas Berghammer   }
12100e305d2STamas Berghammer 
12200e305d2STamas Berghammer   if (log)
123b9c1b51eSKate Stone     log->Printf(
124b9c1b51eSKate Stone         "PlatformAndroid::%s() aborting creation of remote-android platform",
125b9c1b51eSKate Stone         __FUNCTION__);
12600e305d2STamas Berghammer 
12700e305d2STamas Berghammer   return PlatformSP();
12800e305d2STamas Berghammer }
12900e305d2STamas Berghammer 
130b9c1b51eSKate Stone PlatformAndroid::PlatformAndroid(bool is_host)
131b9c1b51eSKate Stone     : PlatformLinux(is_host), m_sdk_version(0) {}
13200e305d2STamas Berghammer 
133b9c1b51eSKate Stone PlatformAndroid::~PlatformAndroid() {}
13400e305d2STamas Berghammer 
135b9c1b51eSKate Stone ConstString PlatformAndroid::GetPluginNameStatic(bool is_host) {
136b9c1b51eSKate Stone   if (is_host) {
1371c6a1ea9STamas Berghammer     static ConstString g_host_name(Platform::GetHostPlatformName());
1381c6a1ea9STamas Berghammer     return g_host_name;
139b9c1b51eSKate Stone   } else {
14000e305d2STamas Berghammer     static ConstString g_remote_name("remote-android");
14100e305d2STamas Berghammer     return g_remote_name;
14200e305d2STamas Berghammer   }
1431c6a1ea9STamas Berghammer }
14400e305d2STamas Berghammer 
145b9c1b51eSKate Stone const char *PlatformAndroid::GetPluginDescriptionStatic(bool is_host) {
1461c6a1ea9STamas Berghammer   if (is_host)
1471c6a1ea9STamas Berghammer     return "Local Android user platform plug-in.";
1481c6a1ea9STamas Berghammer   else
14900e305d2STamas Berghammer     return "Remote Android user platform plug-in.";
15000e305d2STamas Berghammer }
15100e305d2STamas Berghammer 
152b9c1b51eSKate Stone ConstString PlatformAndroid::GetPluginName() {
1531c6a1ea9STamas Berghammer   return GetPluginNameStatic(IsHost());
15400e305d2STamas Berghammer }
15500e305d2STamas Berghammer 
15697206d57SZachary Turner Status PlatformAndroid::ConnectRemote(Args &args) {
1576f001068SOleksiy Vyalov   m_device_id.clear();
1586f001068SOleksiy Vyalov 
159b9c1b51eSKate Stone   if (IsHost()) {
16097206d57SZachary Turner     return Status("can't connect to the host platform '%s', always connected",
161b9c1b51eSKate Stone                   GetPluginName().GetCString());
1621c6a1ea9STamas Berghammer   }
1631c6a1ea9STamas Berghammer 
16400e305d2STamas Berghammer   if (!m_remote_platform_sp)
16500e305d2STamas Berghammer     m_remote_platform_sp = PlatformSP(new PlatformAndroidRemoteGDBServer());
1666f001068SOleksiy Vyalov 
1673ea689b3SChaoren Lin   int port;
168245f7fdcSZachary Turner   llvm::StringRef scheme, host, path;
1693ea689b3SChaoren Lin   const char *url = args.GetArgumentAtIndex(0);
1703ea689b3SChaoren Lin   if (!url)
17197206d57SZachary Turner     return Status("URL is null.");
1723ea689b3SChaoren Lin   if (!UriParser::Parse(url, scheme, host, port, path))
17397206d57SZachary Turner     return Status("Invalid URL: %s", url);
174a29d6475SOleksiy Vyalov   if (host != "localhost")
1753ea689b3SChaoren Lin     m_device_id = host;
1763ea689b3SChaoren Lin 
1776f001068SOleksiy Vyalov   auto error = PlatformLinux::ConnectRemote(args);
178b9c1b51eSKate Stone   if (error.Success()) {
179d6a143fcSOleksiy Vyalov     AdbClient adb;
180d6a143fcSOleksiy Vyalov     error = AdbClient::CreateByDeviceID(m_device_id, adb);
1816f001068SOleksiy Vyalov     if (error.Fail())
1826f001068SOleksiy Vyalov       return error;
1836f001068SOleksiy Vyalov 
184d6a143fcSOleksiy Vyalov     m_device_id = adb.GetDeviceID();
1856f001068SOleksiy Vyalov   }
1866f001068SOleksiy Vyalov   return error;
1876f001068SOleksiy Vyalov }
1886f001068SOleksiy Vyalov 
18997206d57SZachary Turner Status PlatformAndroid::GetFile(const FileSpec &source,
190b9c1b51eSKate Stone                                 const FileSpec &destination) {
191291f59c5SOleksiy Vyalov   if (IsHost() || !m_remote_platform_sp)
1920b5ebef7SOleksiy Vyalov     return PlatformLinux::GetFile(source, destination);
193291f59c5SOleksiy Vyalov 
1948f3be7a3SJonas Devlieghere   FileSpec source_spec(source.GetPath(false), FileSpec::Style::posix);
195372e9067SChaoren Lin   if (source_spec.IsRelative())
196b9c1b51eSKate Stone     source_spec = GetRemoteWorkingDirectory().CopyByAppendingPathComponent(
197b9c1b51eSKate Stone         source_spec.GetCString(false));
198291f59c5SOleksiy Vyalov 
19997206d57SZachary Turner   Status error;
200d6a143fcSOleksiy Vyalov   auto sync_service = GetSyncService(error);
201d6a143fcSOleksiy Vyalov   if (error.Fail())
202d6a143fcSOleksiy Vyalov     return error;
203d6a143fcSOleksiy Vyalov 
204c6ac2e1eSOleksiy Vyalov   uint32_t mode = 0, size = 0, mtime = 0;
205c6ac2e1eSOleksiy Vyalov   error = sync_service->Stat(source_spec, mode, size, mtime);
206c6ac2e1eSOleksiy Vyalov   if (error.Fail())
207c6ac2e1eSOleksiy Vyalov     return error;
208c6ac2e1eSOleksiy Vyalov 
209c6ac2e1eSOleksiy Vyalov   if (mode != 0)
210d6a143fcSOleksiy Vyalov     return sync_service->PullFile(source_spec, destination);
211c6ac2e1eSOleksiy Vyalov 
212c6ac2e1eSOleksiy Vyalov   auto source_file = source_spec.GetCString(false);
213c6ac2e1eSOleksiy Vyalov 
214c6ac2e1eSOleksiy Vyalov   Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM));
215c6ac2e1eSOleksiy Vyalov   if (log)
216b9c1b51eSKate Stone     log->Printf("Got mode == 0 on '%s': try to get file via 'shell cat'",
217b9c1b51eSKate Stone                 source_file);
218c6ac2e1eSOleksiy Vyalov 
219c6ac2e1eSOleksiy Vyalov   if (strchr(source_file, '\'') != nullptr)
22097206d57SZachary Turner     return Status("Doesn't support single-quotes in filenames");
221c6ac2e1eSOleksiy Vyalov 
22205097246SAdrian Prantl   // mode == 0 can signify that adbd cannot access the file due security
22305097246SAdrian Prantl   // constraints - try "cat ..." as a fallback.
224c6ac2e1eSOleksiy Vyalov   AdbClient adb(m_device_id);
225c6ac2e1eSOleksiy Vyalov 
226c6ac2e1eSOleksiy Vyalov   char cmd[PATH_MAX];
227c6ac2e1eSOleksiy Vyalov   snprintf(cmd, sizeof(cmd), "cat '%s'", source_file);
228c6ac2e1eSOleksiy Vyalov 
229ce255ff2SPavel Labath   return adb.ShellToFile(cmd, minutes(1), destination);
2300b5ebef7SOleksiy Vyalov }
2310b5ebef7SOleksiy Vyalov 
23297206d57SZachary Turner Status PlatformAndroid::PutFile(const FileSpec &source,
233b9c1b51eSKate Stone                                 const FileSpec &destination, uint32_t uid,
234b9c1b51eSKate Stone                                 uint32_t gid) {
23517701b59SOleksiy Vyalov   if (IsHost() || !m_remote_platform_sp)
23617701b59SOleksiy Vyalov     return PlatformLinux::PutFile(source, destination, uid, gid);
23717701b59SOleksiy Vyalov 
2388f3be7a3SJonas Devlieghere   FileSpec destination_spec(destination.GetPath(false), FileSpec::Style::posix);
239372e9067SChaoren Lin   if (destination_spec.IsRelative())
240b9c1b51eSKate Stone     destination_spec = GetRemoteWorkingDirectory().CopyByAppendingPathComponent(
241b9c1b51eSKate Stone         destination_spec.GetCString(false));
24217701b59SOleksiy Vyalov 
24362efb1f6SRobert Flack   // TODO: Set correct uid and gid on remote file.
24497206d57SZachary Turner   Status error;
245d6a143fcSOleksiy Vyalov   auto sync_service = GetSyncService(error);
246d6a143fcSOleksiy Vyalov   if (error.Fail())
247d6a143fcSOleksiy Vyalov     return error;
248d6a143fcSOleksiy Vyalov   return sync_service->PushFile(source, destination_spec);
24962efb1f6SRobert Flack }
25062efb1f6SRobert Flack 
251b9c1b51eSKate Stone const char *PlatformAndroid::GetCacheHostname() { return m_device_id.c_str(); }
25209e9079dSOleksiy Vyalov 
25397206d57SZachary Turner Status PlatformAndroid::DownloadModuleSlice(const FileSpec &src_file_spec,
25409e9079dSOleksiy Vyalov                                             const uint64_t src_offset,
25509e9079dSOleksiy Vyalov                                             const uint64_t src_size,
256b9c1b51eSKate Stone                                             const FileSpec &dst_file_spec) {
25709e9079dSOleksiy Vyalov   if (src_offset != 0)
25897206d57SZachary Turner     return Status("Invalid offset - %" PRIu64, src_offset);
25909e9079dSOleksiy Vyalov 
2600b5ebef7SOleksiy Vyalov   return GetFile(src_file_spec, dst_file_spec);
26109e9079dSOleksiy Vyalov }
2621d0d90b9STamas Berghammer 
26397206d57SZachary Turner Status PlatformAndroid::DisconnectRemote() {
26497206d57SZachary Turner   Status error = PlatformLinux::DisconnectRemote();
265b9c1b51eSKate Stone   if (error.Success()) {
2661d0d90b9STamas Berghammer     m_device_id.clear();
2671d0d90b9STamas Berghammer     m_sdk_version = 0;
2681d0d90b9STamas Berghammer   }
2691d0d90b9STamas Berghammer   return error;
2701d0d90b9STamas Berghammer }
2711d0d90b9STamas Berghammer 
272b9c1b51eSKate Stone uint32_t PlatformAndroid::GetDefaultMemoryCacheLineSize() {
273f2991488SPavel Labath   return g_android_default_cache_size;
274f2991488SPavel Labath }
275f2991488SPavel Labath 
276b9c1b51eSKate Stone uint32_t PlatformAndroid::GetSdkVersion() {
2771d0d90b9STamas Berghammer   if (!IsConnected())
2781d0d90b9STamas Berghammer     return 0;
2791d0d90b9STamas Berghammer 
2801d0d90b9STamas Berghammer   if (m_sdk_version != 0)
2811d0d90b9STamas Berghammer     return m_sdk_version;
2821d0d90b9STamas Berghammer 
2831d0d90b9STamas Berghammer   std::string version_string;
284d6a143fcSOleksiy Vyalov   AdbClient adb(m_device_id);
28597206d57SZachary Turner   Status error =
286ce255ff2SPavel Labath       adb.Shell("getprop ro.build.version.sdk", seconds(5), &version_string);
2873e8947beSTamas Berghammer   version_string = llvm::StringRef(version_string).trim().str();
2883e8947beSTamas Berghammer 
289b9c1b51eSKate Stone   if (error.Fail() || version_string.empty()) {
2901d0d90b9STamas Berghammer     Log *log = GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM);
2911d0d90b9STamas Berghammer     if (log)
2923e8947beSTamas Berghammer       log->Printf("Get SDK version failed. (error: %s, output: %s)",
2933e8947beSTamas Berghammer                   error.AsCString(), version_string.c_str());
2941d0d90b9STamas Berghammer     return 0;
2951d0d90b9STamas Berghammer   }
2961d0d90b9STamas Berghammer 
2971d0d90b9STamas Berghammer   m_sdk_version = StringConvert::ToUInt32(version_string.c_str());
2981d0d90b9STamas Berghammer   return m_sdk_version;
2991d0d90b9STamas Berghammer }
300ec3f92a5STamas Berghammer 
30197206d57SZachary Turner Status PlatformAndroid::DownloadSymbolFile(const lldb::ModuleSP &module_sp,
302b9c1b51eSKate Stone                                            const FileSpec &dst_file_spec) {
303ec3f92a5STamas Berghammer   // For oat file we can try to fetch additional debug info from the device
304dba6503aSTamas Berghammer   ConstString extension = module_sp->GetFileSpec().GetFileNameExtension();
305*05cfdb0eSRaphael Isemann   if (extension != ".oat" && extension != ".odex")
30697206d57SZachary Turner     return Status(
307b9c1b51eSKate Stone         "Symbol file downloading only supported for oat and odex files");
308ec3f92a5STamas Berghammer 
309ec3f92a5STamas Berghammer   // If we have no information about the platform file we can't execute oatdump
310ec3f92a5STamas Berghammer   if (!module_sp->GetPlatformFileSpec())
31197206d57SZachary Turner     return Status("No platform file specified");
312ec3f92a5STamas Berghammer 
313ec3f92a5STamas Berghammer   // Symbolizer isn't available before SDK version 23
314ec3f92a5STamas Berghammer   if (GetSdkVersion() < 23)
31597206d57SZachary Turner     return Status("Symbol file generation only supported on SDK 23+");
316ec3f92a5STamas Berghammer 
317ec3f92a5STamas Berghammer   // If we already have symtab then we don't have to try and generate one
318b9c1b51eSKate Stone   if (module_sp->GetSectionList()->FindSectionByName(ConstString(".symtab")) !=
319b9c1b51eSKate Stone       nullptr)
32097206d57SZachary Turner     return Status("Symtab already available in the module");
321ec3f92a5STamas Berghammer 
322d6a143fcSOleksiy Vyalov   AdbClient adb(m_device_id);
3239d8dde8cSTamas Berghammer   std::string tmpdir;
32497206d57SZachary Turner   Status error = adb.Shell("mktemp --directory --tmpdir /data/local/tmp",
325ce255ff2SPavel Labath                            seconds(5), &tmpdir);
3269d8dde8cSTamas Berghammer   if (error.Fail() || tmpdir.empty())
32797206d57SZachary Turner     return Status("Failed to generate temporary directory on the device (%s)",
328b9c1b51eSKate Stone                   error.AsCString());
329539cd118STamas Berghammer   tmpdir = llvm::StringRef(tmpdir).trim().str();
330ec3f92a5STamas Berghammer 
331ec3f92a5STamas Berghammer   // Create file remover for the temporary directory created on the device
332b9c1b51eSKate Stone   std::unique_ptr<std::string, std::function<void(std::string *)>>
3333bc714b2SZachary Turner   tmpdir_remover(&tmpdir, [&adb](std::string *s) {
334ec3f92a5STamas Berghammer     StreamString command;
335ec3f92a5STamas Berghammer     command.Printf("rm -rf %s", s->c_str());
33697206d57SZachary Turner     Status error = adb.Shell(command.GetData(), seconds(5), nullptr);
337ec3f92a5STamas Berghammer 
338ec3f92a5STamas Berghammer     Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM));
339dd778ca6SOleksiy Vyalov     if (log && error.Fail())
340ec3f92a5STamas Berghammer       log->Printf("Failed to remove temp directory: %s", error.AsCString());
341b9c1b51eSKate Stone   });
342ec3f92a5STamas Berghammer 
3438f3be7a3SJonas Devlieghere   FileSpec symfile_platform_filespec(tmpdir);
344ec3f92a5STamas Berghammer   symfile_platform_filespec.AppendPathComponent("symbolized.oat");
345ec3f92a5STamas Berghammer 
346ec3f92a5STamas Berghammer   // Execute oatdump on the remote device to generate a file with symtab
3479d8dde8cSTamas Berghammer   StreamString command;
348ec3f92a5STamas Berghammer   command.Printf("oatdump --symbolize=%s --output=%s",
349ec3f92a5STamas Berghammer                  module_sp->GetPlatformFileSpec().GetCString(false),
350ec3f92a5STamas Berghammer                  symfile_platform_filespec.GetCString(false));
351ce255ff2SPavel Labath   error = adb.Shell(command.GetData(), minutes(1), nullptr);
3529d8dde8cSTamas Berghammer   if (error.Fail())
35397206d57SZachary Turner     return Status("Oatdump failed: %s", error.AsCString());
354ec3f92a5STamas Berghammer 
355ec3f92a5STamas Berghammer   // Download the symbolfile from the remote device
356ec3f92a5STamas Berghammer   return GetFile(symfile_platform_filespec, dst_file_spec);
357ec3f92a5STamas Berghammer }
3583c47151dSTamas Berghammer 
359b9c1b51eSKate Stone bool PlatformAndroid::GetRemoteOSVersion() {
3602272c481SPavel Labath   m_os_version = llvm::VersionTuple(GetSdkVersion());
3612272c481SPavel Labath   return !m_os_version.empty();
3623c47151dSTamas Berghammer }
3633cb132a0STamas Berghammer 
364f7a5851dSNitesh Jain llvm::StringRef
365f7a5851dSNitesh Jain PlatformAndroid::GetLibdlFunctionDeclarations(lldb_private::Process *process) {
366f7a5851dSNitesh Jain   SymbolContextList matching_symbols;
367f7a5851dSNitesh Jain   std::vector<const char *> dl_open_names = { "__dl_dlopen", "dlopen" };
368f7a5851dSNitesh Jain   const char *dl_open_name = nullptr;
369f7a5851dSNitesh Jain   Target &target = process->GetTarget();
370f7a5851dSNitesh Jain   for (auto name: dl_open_names) {
371f7a5851dSNitesh Jain     if (target.GetImages().FindFunctionSymbols(ConstString(name),
372f7a5851dSNitesh Jain                                                eFunctionNameTypeFull,
373f7a5851dSNitesh Jain                                                matching_symbols)) {
374f7a5851dSNitesh Jain        dl_open_name = name;
375f7a5851dSNitesh Jain        break;
376f7a5851dSNitesh Jain     }
377f7a5851dSNitesh Jain   }
378d37c9469SPavel Labath   // Older platform versions have the dl function symbols mangled
379f7a5851dSNitesh Jain   if (dl_open_name == dl_open_names[0])
380fcdde63fSTamas Berghammer     return R"(
381fcdde63fSTamas Berghammer               extern "C" void* dlopen(const char*, int) asm("__dl_dlopen");
382fcdde63fSTamas Berghammer               extern "C" void* dlsym(void*, const char*) asm("__dl_dlsym");
383fcdde63fSTamas Berghammer               extern "C" int   dlclose(void*) asm("__dl_dlclose");
384fcdde63fSTamas Berghammer               extern "C" char* dlerror(void) asm("__dl_dlerror");
3853cb132a0STamas Berghammer              )";
386d37c9469SPavel Labath 
387f7a5851dSNitesh Jain   return PlatformPOSIX::GetLibdlFunctionDeclarations(process);
3883cb132a0STamas Berghammer }
389d6a143fcSOleksiy Vyalov 
39097206d57SZachary Turner AdbClient::SyncService *PlatformAndroid::GetSyncService(Status &error) {
391d6a143fcSOleksiy Vyalov   if (m_adb_sync_svc && m_adb_sync_svc->IsConnected())
392d6a143fcSOleksiy Vyalov     return m_adb_sync_svc.get();
393d6a143fcSOleksiy Vyalov 
394d6a143fcSOleksiy Vyalov   AdbClient adb(m_device_id);
395d6a143fcSOleksiy Vyalov   m_adb_sync_svc = adb.GetSyncService(error);
396d6a143fcSOleksiy Vyalov   return (error.Success()) ? m_adb_sync_svc.get() : nullptr;
397d6a143fcSOleksiy Vyalov }
398