100e305d2STamas Berghammer //===-- PlatformAndroid.cpp -------------------------------------*- C++ -*-===//
200e305d2STamas Berghammer //
300e305d2STamas Berghammer //                     The LLVM Compiler Infrastructure
400e305d2STamas Berghammer //
500e305d2STamas Berghammer // This file is distributed under the University of Illinois Open Source
600e305d2STamas Berghammer // License. See LICENSE.TXT for details.
700e305d2STamas Berghammer //
800e305d2STamas Berghammer //===----------------------------------------------------------------------===//
900e305d2STamas Berghammer 
1000e305d2STamas Berghammer // C Includes
1100e305d2STamas Berghammer // C++ Includes
1200e305d2STamas Berghammer // Other libraries and framework includes
13b9c1b51eSKate Stone #include "Utility/UriParser.h"
1400e305d2STamas Berghammer #include "lldb/Core/Log.h"
15ec3f92a5STamas Berghammer #include "lldb/Core/Module.h"
1600e305d2STamas Berghammer #include "lldb/Core/PluginManager.h"
173cb132a0STamas Berghammer #include "lldb/Core/Scalar.h"
18ec3f92a5STamas Berghammer #include "lldb/Core/Section.h"
193cb132a0STamas Berghammer #include "lldb/Core/ValueObject.h"
201c6a1ea9STamas Berghammer #include "lldb/Host/HostInfo.h"
211d0d90b9STamas Berghammer #include "lldb/Host/StringConvert.h"
2200e305d2STamas Berghammer 
2300e305d2STamas Berghammer // Project includes
246f001068SOleksiy Vyalov #include "AdbClient.h"
2500e305d2STamas Berghammer #include "PlatformAndroid.h"
2600e305d2STamas Berghammer #include "PlatformAndroidRemoteGDBServer.h"
2700e305d2STamas Berghammer 
2800e305d2STamas Berghammer using namespace lldb;
2900e305d2STamas Berghammer using namespace lldb_private;
30db264a6dSTamas Berghammer using namespace lldb_private::platform_android;
31*ce255ff2SPavel Labath using namespace std::chrono;
3200e305d2STamas Berghammer 
3300e305d2STamas Berghammer static uint32_t g_initialize_count = 0;
34b9c1b51eSKate Stone static const unsigned int g_android_default_cache_size =
35b9c1b51eSKate Stone     2048; // Fits inside 4k adb packet.
3600e305d2STamas Berghammer 
37b9c1b51eSKate Stone void PlatformAndroid::Initialize() {
383c4f89d7STamas Berghammer   PlatformLinux::Initialize();
393c4f89d7STamas Berghammer 
40b9c1b51eSKate Stone   if (g_initialize_count++ == 0) {
411c6a1ea9STamas Berghammer #if defined(__ANDROID__)
421c6a1ea9STamas Berghammer     PlatformSP default_platform_sp(new PlatformAndroid(true));
431c6a1ea9STamas Berghammer     default_platform_sp->SetSystemArchitecture(HostInfo::GetArchitecture());
441c6a1ea9STamas Berghammer     Platform::SetHostPlatform(default_platform_sp);
451c6a1ea9STamas Berghammer #endif
46b9c1b51eSKate Stone     PluginManager::RegisterPlugin(
47b9c1b51eSKate Stone         PlatformAndroid::GetPluginNameStatic(false),
481c6a1ea9STamas Berghammer         PlatformAndroid::GetPluginDescriptionStatic(false),
4900e305d2STamas Berghammer         PlatformAndroid::CreateInstance);
5000e305d2STamas Berghammer   }
5100e305d2STamas Berghammer }
5200e305d2STamas Berghammer 
53b9c1b51eSKate Stone void PlatformAndroid::Terminate() {
54b9c1b51eSKate Stone   if (g_initialize_count > 0) {
55b9c1b51eSKate Stone     if (--g_initialize_count == 0) {
5600e305d2STamas Berghammer       PluginManager::UnregisterPlugin(PlatformAndroid::CreateInstance);
5700e305d2STamas Berghammer     }
5800e305d2STamas Berghammer   }
593c4f89d7STamas Berghammer 
603c4f89d7STamas Berghammer   PlatformLinux::Terminate();
6100e305d2STamas Berghammer }
6200e305d2STamas Berghammer 
63b9c1b51eSKate Stone PlatformSP PlatformAndroid::CreateInstance(bool force, const ArchSpec *arch) {
64db264a6dSTamas Berghammer   Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM));
65b9c1b51eSKate Stone   if (log) {
6600e305d2STamas Berghammer     const char *arch_name;
6700e305d2STamas Berghammer     if (arch && arch->GetArchitectureName())
6800e305d2STamas Berghammer       arch_name = arch->GetArchitectureName();
6900e305d2STamas Berghammer     else
7000e305d2STamas Berghammer       arch_name = "<null>";
7100e305d2STamas Berghammer 
72b9c1b51eSKate Stone     const char *triple_cstr =
73b9c1b51eSKate Stone         arch ? arch->GetTriple().getTriple().c_str() : "<null>";
7400e305d2STamas Berghammer 
75b9c1b51eSKate Stone     log->Printf("PlatformAndroid::%s(force=%s, arch={%s,%s})", __FUNCTION__,
76b9c1b51eSKate Stone                 force ? "true" : "false", arch_name, triple_cstr);
7700e305d2STamas Berghammer   }
7800e305d2STamas Berghammer 
7900e305d2STamas Berghammer   bool create = force;
80b9c1b51eSKate Stone   if (create == false && arch && arch->IsValid()) {
8100e305d2STamas Berghammer     const llvm::Triple &triple = arch->GetTriple();
82b9c1b51eSKate Stone     switch (triple.getVendor()) {
8300e305d2STamas Berghammer     case llvm::Triple::PC:
8400e305d2STamas Berghammer       create = true;
8500e305d2STamas Berghammer       break;
8600e305d2STamas Berghammer 
8700e305d2STamas Berghammer #if defined(__ANDROID__)
8800e305d2STamas Berghammer     // Only accept "unknown" for the vendor if the host is android and
8900e305d2STamas Berghammer     // it "unknown" wasn't specified (it was just returned because it
9000e305d2STamas Berghammer     // was NOT specified_
9100e305d2STamas Berghammer     case llvm::Triple::VendorType::UnknownVendor:
9200e305d2STamas Berghammer       create = !arch->TripleVendorWasSpecified();
9300e305d2STamas Berghammer       break;
9400e305d2STamas Berghammer #endif
9500e305d2STamas Berghammer     default:
9600e305d2STamas Berghammer       break;
9700e305d2STamas Berghammer     }
9800e305d2STamas Berghammer 
99b9c1b51eSKate Stone     if (create) {
100b9c1b51eSKate Stone       switch (triple.getOS()) {
1011c6a1ea9STamas Berghammer       case llvm::Triple::Android:
10200e305d2STamas Berghammer         break;
10300e305d2STamas Berghammer 
10400e305d2STamas Berghammer #if defined(__ANDROID__)
10500e305d2STamas Berghammer       // Only accept "unknown" for the OS if the host is android and
10600e305d2STamas Berghammer       // it "unknown" wasn't specified (it was just returned because it
10700e305d2STamas Berghammer       // was NOT specified)
10800e305d2STamas Berghammer       case llvm::Triple::OSType::UnknownOS:
10900e305d2STamas Berghammer         create = !arch->TripleOSWasSpecified();
11000e305d2STamas Berghammer         break;
11100e305d2STamas Berghammer #endif
11200e305d2STamas Berghammer       default:
11300e305d2STamas Berghammer         create = false;
11400e305d2STamas Berghammer         break;
11500e305d2STamas Berghammer       }
11600e305d2STamas Berghammer     }
11700e305d2STamas Berghammer   }
11800e305d2STamas Berghammer 
119b9c1b51eSKate Stone   if (create) {
12000e305d2STamas Berghammer     if (log)
121b9c1b51eSKate Stone       log->Printf("PlatformAndroid::%s() creating remote-android platform",
122b9c1b51eSKate Stone                   __FUNCTION__);
1231c6a1ea9STamas Berghammer     return PlatformSP(new PlatformAndroid(false));
12400e305d2STamas Berghammer   }
12500e305d2STamas Berghammer 
12600e305d2STamas Berghammer   if (log)
127b9c1b51eSKate Stone     log->Printf(
128b9c1b51eSKate Stone         "PlatformAndroid::%s() aborting creation of remote-android platform",
129b9c1b51eSKate Stone         __FUNCTION__);
13000e305d2STamas Berghammer 
13100e305d2STamas Berghammer   return PlatformSP();
13200e305d2STamas Berghammer }
13300e305d2STamas Berghammer 
134b9c1b51eSKate Stone PlatformAndroid::PlatformAndroid(bool is_host)
135b9c1b51eSKate Stone     : PlatformLinux(is_host), m_sdk_version(0) {}
13600e305d2STamas Berghammer 
137b9c1b51eSKate Stone PlatformAndroid::~PlatformAndroid() {}
13800e305d2STamas Berghammer 
139b9c1b51eSKate Stone ConstString PlatformAndroid::GetPluginNameStatic(bool is_host) {
140b9c1b51eSKate Stone   if (is_host) {
1411c6a1ea9STamas Berghammer     static ConstString g_host_name(Platform::GetHostPlatformName());
1421c6a1ea9STamas Berghammer     return g_host_name;
143b9c1b51eSKate Stone   } else {
14400e305d2STamas Berghammer     static ConstString g_remote_name("remote-android");
14500e305d2STamas Berghammer     return g_remote_name;
14600e305d2STamas Berghammer   }
1471c6a1ea9STamas Berghammer }
14800e305d2STamas Berghammer 
149b9c1b51eSKate Stone const char *PlatformAndroid::GetPluginDescriptionStatic(bool is_host) {
1501c6a1ea9STamas Berghammer   if (is_host)
1511c6a1ea9STamas Berghammer     return "Local Android user platform plug-in.";
1521c6a1ea9STamas Berghammer   else
15300e305d2STamas Berghammer     return "Remote Android user platform plug-in.";
15400e305d2STamas Berghammer }
15500e305d2STamas Berghammer 
156b9c1b51eSKate Stone ConstString PlatformAndroid::GetPluginName() {
1571c6a1ea9STamas Berghammer   return GetPluginNameStatic(IsHost());
15800e305d2STamas Berghammer }
15900e305d2STamas Berghammer 
160b9c1b51eSKate Stone Error PlatformAndroid::ConnectRemote(Args &args) {
1616f001068SOleksiy Vyalov   m_device_id.clear();
1626f001068SOleksiy Vyalov 
163b9c1b51eSKate Stone   if (IsHost()) {
164b9c1b51eSKate Stone     return Error("can't connect to the host platform '%s', always connected",
165b9c1b51eSKate Stone                  GetPluginName().GetCString());
1661c6a1ea9STamas Berghammer   }
1671c6a1ea9STamas Berghammer 
16800e305d2STamas Berghammer   if (!m_remote_platform_sp)
16900e305d2STamas Berghammer     m_remote_platform_sp = PlatformSP(new PlatformAndroidRemoteGDBServer());
1706f001068SOleksiy Vyalov 
1713ea689b3SChaoren Lin   int port;
172245f7fdcSZachary Turner   llvm::StringRef scheme, host, path;
1733ea689b3SChaoren Lin   const char *url = args.GetArgumentAtIndex(0);
1743ea689b3SChaoren Lin   if (!url)
1753ea689b3SChaoren Lin     return Error("URL is null.");
1763ea689b3SChaoren Lin   if (!UriParser::Parse(url, scheme, host, port, path))
1773ea689b3SChaoren Lin     return Error("Invalid URL: %s", url);
178a29d6475SOleksiy Vyalov   if (host != "localhost")
1793ea689b3SChaoren Lin     m_device_id = host;
1803ea689b3SChaoren Lin 
1816f001068SOleksiy Vyalov   auto error = PlatformLinux::ConnectRemote(args);
182b9c1b51eSKate Stone   if (error.Success()) {
183d6a143fcSOleksiy Vyalov     AdbClient adb;
184d6a143fcSOleksiy Vyalov     error = AdbClient::CreateByDeviceID(m_device_id, adb);
1856f001068SOleksiy Vyalov     if (error.Fail())
1866f001068SOleksiy Vyalov       return error;
1876f001068SOleksiy Vyalov 
188d6a143fcSOleksiy Vyalov     m_device_id = adb.GetDeviceID();
1896f001068SOleksiy Vyalov   }
1906f001068SOleksiy Vyalov   return error;
1916f001068SOleksiy Vyalov }
1926f001068SOleksiy Vyalov 
193b9c1b51eSKate Stone Error PlatformAndroid::GetFile(const FileSpec &source,
194b9c1b51eSKate Stone                                const FileSpec &destination) {
195291f59c5SOleksiy Vyalov   if (IsHost() || !m_remote_platform_sp)
1960b5ebef7SOleksiy Vyalov     return PlatformLinux::GetFile(source, destination);
197291f59c5SOleksiy Vyalov 
198b9c1b51eSKate Stone   FileSpec source_spec(source.GetPath(false), false,
199b9c1b51eSKate Stone                        FileSpec::ePathSyntaxPosix);
200372e9067SChaoren Lin   if (source_spec.IsRelative())
201b9c1b51eSKate Stone     source_spec = GetRemoteWorkingDirectory().CopyByAppendingPathComponent(
202b9c1b51eSKate Stone         source_spec.GetCString(false));
203291f59c5SOleksiy Vyalov 
204d6a143fcSOleksiy Vyalov   Error error;
205d6a143fcSOleksiy Vyalov   auto sync_service = GetSyncService(error);
206d6a143fcSOleksiy Vyalov   if (error.Fail())
207d6a143fcSOleksiy Vyalov     return error;
208d6a143fcSOleksiy Vyalov 
209c6ac2e1eSOleksiy Vyalov   uint32_t mode = 0, size = 0, mtime = 0;
210c6ac2e1eSOleksiy Vyalov   error = sync_service->Stat(source_spec, mode, size, mtime);
211c6ac2e1eSOleksiy Vyalov   if (error.Fail())
212c6ac2e1eSOleksiy Vyalov     return error;
213c6ac2e1eSOleksiy Vyalov 
214c6ac2e1eSOleksiy Vyalov   if (mode != 0)
215d6a143fcSOleksiy Vyalov     return sync_service->PullFile(source_spec, destination);
216c6ac2e1eSOleksiy Vyalov 
217c6ac2e1eSOleksiy Vyalov   auto source_file = source_spec.GetCString(false);
218c6ac2e1eSOleksiy Vyalov 
219c6ac2e1eSOleksiy Vyalov   Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM));
220c6ac2e1eSOleksiy Vyalov   if (log)
221b9c1b51eSKate Stone     log->Printf("Got mode == 0 on '%s': try to get file via 'shell cat'",
222b9c1b51eSKate Stone                 source_file);
223c6ac2e1eSOleksiy Vyalov 
224c6ac2e1eSOleksiy Vyalov   if (strchr(source_file, '\'') != nullptr)
225c6ac2e1eSOleksiy Vyalov     return Error("Doesn't support single-quotes in filenames");
226c6ac2e1eSOleksiy Vyalov 
227c6ac2e1eSOleksiy Vyalov   // mode == 0 can signify that adbd cannot access the file
228c6ac2e1eSOleksiy Vyalov   // due security constraints - try "cat ..." as a fallback.
229c6ac2e1eSOleksiy Vyalov   AdbClient adb(m_device_id);
230c6ac2e1eSOleksiy Vyalov 
231c6ac2e1eSOleksiy Vyalov   char cmd[PATH_MAX];
232c6ac2e1eSOleksiy Vyalov   snprintf(cmd, sizeof(cmd), "cat '%s'", source_file);
233c6ac2e1eSOleksiy Vyalov 
234*ce255ff2SPavel Labath   return adb.ShellToFile(cmd, minutes(1), destination);
2350b5ebef7SOleksiy Vyalov }
2360b5ebef7SOleksiy Vyalov 
237b9c1b51eSKate Stone Error PlatformAndroid::PutFile(const FileSpec &source,
238b9c1b51eSKate Stone                                const FileSpec &destination, uint32_t uid,
239b9c1b51eSKate Stone                                uint32_t gid) {
24017701b59SOleksiy Vyalov   if (IsHost() || !m_remote_platform_sp)
24117701b59SOleksiy Vyalov     return PlatformLinux::PutFile(source, destination, uid, gid);
24217701b59SOleksiy Vyalov 
243b9c1b51eSKate Stone   FileSpec destination_spec(destination.GetPath(false), false,
244b9c1b51eSKate Stone                             FileSpec::ePathSyntaxPosix);
245372e9067SChaoren Lin   if (destination_spec.IsRelative())
246b9c1b51eSKate Stone     destination_spec = GetRemoteWorkingDirectory().CopyByAppendingPathComponent(
247b9c1b51eSKate Stone         destination_spec.GetCString(false));
24817701b59SOleksiy Vyalov 
24962efb1f6SRobert Flack   // TODO: Set correct uid and gid on remote file.
250d6a143fcSOleksiy Vyalov   Error error;
251d6a143fcSOleksiy Vyalov   auto sync_service = GetSyncService(error);
252d6a143fcSOleksiy Vyalov   if (error.Fail())
253d6a143fcSOleksiy Vyalov     return error;
254d6a143fcSOleksiy Vyalov   return sync_service->PushFile(source, destination_spec);
25562efb1f6SRobert Flack }
25662efb1f6SRobert Flack 
257b9c1b51eSKate Stone const char *PlatformAndroid::GetCacheHostname() { return m_device_id.c_str(); }
25809e9079dSOleksiy Vyalov 
259b9c1b51eSKate Stone Error PlatformAndroid::DownloadModuleSlice(const FileSpec &src_file_spec,
26009e9079dSOleksiy Vyalov                                            const uint64_t src_offset,
26109e9079dSOleksiy Vyalov                                            const uint64_t src_size,
262b9c1b51eSKate Stone                                            const FileSpec &dst_file_spec) {
26309e9079dSOleksiy Vyalov   if (src_offset != 0)
26409e9079dSOleksiy Vyalov     return Error("Invalid offset - %" PRIu64, src_offset);
26509e9079dSOleksiy Vyalov 
2660b5ebef7SOleksiy Vyalov   return GetFile(src_file_spec, dst_file_spec);
26709e9079dSOleksiy Vyalov }
2681d0d90b9STamas Berghammer 
269b9c1b51eSKate Stone Error PlatformAndroid::DisconnectRemote() {
2701d0d90b9STamas Berghammer   Error error = PlatformLinux::DisconnectRemote();
271b9c1b51eSKate Stone   if (error.Success()) {
2721d0d90b9STamas Berghammer     m_device_id.clear();
2731d0d90b9STamas Berghammer     m_sdk_version = 0;
2741d0d90b9STamas Berghammer   }
2751d0d90b9STamas Berghammer   return error;
2761d0d90b9STamas Berghammer }
2771d0d90b9STamas Berghammer 
278b9c1b51eSKate Stone uint32_t PlatformAndroid::GetDefaultMemoryCacheLineSize() {
279f2991488SPavel Labath   return g_android_default_cache_size;
280f2991488SPavel Labath }
281f2991488SPavel Labath 
282b9c1b51eSKate Stone uint32_t PlatformAndroid::GetSdkVersion() {
2831d0d90b9STamas Berghammer   if (!IsConnected())
2841d0d90b9STamas Berghammer     return 0;
2851d0d90b9STamas Berghammer 
2861d0d90b9STamas Berghammer   if (m_sdk_version != 0)
2871d0d90b9STamas Berghammer     return m_sdk_version;
2881d0d90b9STamas Berghammer 
2891d0d90b9STamas Berghammer   std::string version_string;
290d6a143fcSOleksiy Vyalov   AdbClient adb(m_device_id);
291b9c1b51eSKate Stone   Error error =
292*ce255ff2SPavel Labath       adb.Shell("getprop ro.build.version.sdk", seconds(5), &version_string);
2933e8947beSTamas Berghammer   version_string = llvm::StringRef(version_string).trim().str();
2943e8947beSTamas Berghammer 
295b9c1b51eSKate Stone   if (error.Fail() || version_string.empty()) {
2961d0d90b9STamas Berghammer     Log *log = GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM);
2971d0d90b9STamas Berghammer     if (log)
2983e8947beSTamas Berghammer       log->Printf("Get SDK version failed. (error: %s, output: %s)",
2993e8947beSTamas Berghammer                   error.AsCString(), version_string.c_str());
3001d0d90b9STamas Berghammer     return 0;
3011d0d90b9STamas Berghammer   }
3021d0d90b9STamas Berghammer 
3031d0d90b9STamas Berghammer   m_sdk_version = StringConvert::ToUInt32(version_string.c_str());
3041d0d90b9STamas Berghammer   return m_sdk_version;
3051d0d90b9STamas Berghammer }
306ec3f92a5STamas Berghammer 
307b9c1b51eSKate Stone Error PlatformAndroid::DownloadSymbolFile(const lldb::ModuleSP &module_sp,
308b9c1b51eSKate Stone                                           const FileSpec &dst_file_spec) {
309ec3f92a5STamas Berghammer   // For oat file we can try to fetch additional debug info from the device
310dba6503aSTamas Berghammer   ConstString extension = module_sp->GetFileSpec().GetFileNameExtension();
311dba6503aSTamas Berghammer   if (extension != ConstString("oat") && extension != ConstString("odex"))
312b9c1b51eSKate Stone     return Error(
313b9c1b51eSKate Stone         "Symbol file downloading only supported for oat and odex files");
314ec3f92a5STamas Berghammer 
315ec3f92a5STamas Berghammer   // If we have no information about the platform file we can't execute oatdump
316ec3f92a5STamas Berghammer   if (!module_sp->GetPlatformFileSpec())
317ec3f92a5STamas Berghammer     return Error("No platform file specified");
318ec3f92a5STamas Berghammer 
319ec3f92a5STamas Berghammer   // Symbolizer isn't available before SDK version 23
320ec3f92a5STamas Berghammer   if (GetSdkVersion() < 23)
321ec3f92a5STamas Berghammer     return Error("Symbol file generation only supported on SDK 23+");
322ec3f92a5STamas Berghammer 
323ec3f92a5STamas Berghammer   // If we already have symtab then we don't have to try and generate one
324b9c1b51eSKate Stone   if (module_sp->GetSectionList()->FindSectionByName(ConstString(".symtab")) !=
325b9c1b51eSKate Stone       nullptr)
326ec3f92a5STamas Berghammer     return Error("Symtab already available in the module");
327ec3f92a5STamas Berghammer 
328d6a143fcSOleksiy Vyalov   AdbClient adb(m_device_id);
3299d8dde8cSTamas Berghammer   std::string tmpdir;
330b9c1b51eSKate Stone   Error error = adb.Shell("mktemp --directory --tmpdir /data/local/tmp",
331*ce255ff2SPavel Labath                           seconds(5), &tmpdir);
3329d8dde8cSTamas Berghammer   if (error.Fail() || tmpdir.empty())
333b9c1b51eSKate Stone     return Error("Failed to generate temporary directory on the device (%s)",
334b9c1b51eSKate Stone                  error.AsCString());
335539cd118STamas Berghammer   tmpdir = llvm::StringRef(tmpdir).trim().str();
336ec3f92a5STamas Berghammer 
337ec3f92a5STamas Berghammer   // Create file remover for the temporary directory created on the device
338b9c1b51eSKate Stone   std::unique_ptr<std::string, std::function<void(std::string *)>>
339b9c1b51eSKate Stone   tmpdir_remover(&tmpdir, [this, &adb](std::string *s) {
340ec3f92a5STamas Berghammer     StreamString command;
341ec3f92a5STamas Berghammer     command.Printf("rm -rf %s", s->c_str());
342*ce255ff2SPavel Labath     Error error = adb.Shell(command.GetData(), seconds(5), nullptr);
343ec3f92a5STamas Berghammer 
344ec3f92a5STamas Berghammer     Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM));
345dd778ca6SOleksiy Vyalov     if (log && error.Fail())
346ec3f92a5STamas Berghammer       log->Printf("Failed to remove temp directory: %s", error.AsCString());
347b9c1b51eSKate Stone   });
348ec3f92a5STamas Berghammer 
349771ef6d4SMalcolm Parsons   FileSpec symfile_platform_filespec(tmpdir, false);
350ec3f92a5STamas Berghammer   symfile_platform_filespec.AppendPathComponent("symbolized.oat");
351ec3f92a5STamas Berghammer 
352ec3f92a5STamas Berghammer   // Execute oatdump on the remote device to generate a file with symtab
3539d8dde8cSTamas Berghammer   StreamString command;
354ec3f92a5STamas Berghammer   command.Printf("oatdump --symbolize=%s --output=%s",
355ec3f92a5STamas Berghammer                  module_sp->GetPlatformFileSpec().GetCString(false),
356ec3f92a5STamas Berghammer                  symfile_platform_filespec.GetCString(false));
357*ce255ff2SPavel Labath   error = adb.Shell(command.GetData(), minutes(1), nullptr);
3589d8dde8cSTamas Berghammer   if (error.Fail())
359ec3f92a5STamas Berghammer     return Error("Oatdump failed: %s", error.AsCString());
360ec3f92a5STamas Berghammer 
361ec3f92a5STamas Berghammer   // Download the symbolfile from the remote device
362ec3f92a5STamas Berghammer   return GetFile(symfile_platform_filespec, dst_file_spec);
363ec3f92a5STamas Berghammer }
3643c47151dSTamas Berghammer 
365b9c1b51eSKate Stone bool PlatformAndroid::GetRemoteOSVersion() {
3663c47151dSTamas Berghammer   m_major_os_version = GetSdkVersion();
3673c47151dSTamas Berghammer   m_minor_os_version = 0;
3683c47151dSTamas Berghammer   m_update_os_version = 0;
3693c47151dSTamas Berghammer   return m_major_os_version != 0;
3703c47151dSTamas Berghammer }
3713cb132a0STamas Berghammer 
372b9c1b51eSKate Stone const char *PlatformAndroid::GetLibdlFunctionDeclarations() const {
373fcdde63fSTamas Berghammer   return R"(
374fcdde63fSTamas Berghammer               extern "C" void* dlopen(const char*, int) asm("__dl_dlopen");
375fcdde63fSTamas Berghammer               extern "C" void* dlsym(void*, const char*) asm("__dl_dlsym");
376fcdde63fSTamas Berghammer               extern "C" int   dlclose(void*) asm("__dl_dlclose");
377fcdde63fSTamas Berghammer               extern "C" char* dlerror(void) asm("__dl_dlerror");
3783cb132a0STamas Berghammer              )";
3793cb132a0STamas Berghammer }
380d6a143fcSOleksiy Vyalov 
381b9c1b51eSKate Stone AdbClient::SyncService *PlatformAndroid::GetSyncService(Error &error) {
382d6a143fcSOleksiy Vyalov   if (m_adb_sync_svc && m_adb_sync_svc->IsConnected())
383d6a143fcSOleksiy Vyalov     return m_adb_sync_svc.get();
384d6a143fcSOleksiy Vyalov 
385d6a143fcSOleksiy Vyalov   AdbClient adb(m_device_id);
386d6a143fcSOleksiy Vyalov   m_adb_sync_svc = adb.GetSyncService(error);
387d6a143fcSOleksiy Vyalov   return (error.Success()) ? m_adb_sync_svc.get() : nullptr;
388d6a143fcSOleksiy Vyalov }
389