100e305d2STamas Berghammer //===-- PlatformAndroidRemoteGDBServer.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 // Other libraries and framework includes 1100e305d2STamas Berghammer #include "lldb/Core/Error.h" 1205a55de3SOleksiy Vyalov #include "lldb/Core/Log.h" 13*e98628ceSOleksiy Vyalov #include "lldb/Host/common/TCPSocket.h" 1405a55de3SOleksiy Vyalov #include "AdbClient.h" 1500e305d2STamas Berghammer #include "PlatformAndroidRemoteGDBServer.h" 1600e305d2STamas Berghammer #include "Utility/UriParser.h" 1700e305d2STamas Berghammer 1854971856SOleksiy Vyalov #include <sstream> 1954971856SOleksiy Vyalov 2000e305d2STamas Berghammer using namespace lldb; 2100e305d2STamas Berghammer using namespace lldb_private; 22db264a6dSTamas Berghammer using namespace platform_android; 2300e305d2STamas Berghammer 2400e305d2STamas Berghammer static const lldb::pid_t g_remote_platform_pid = 0; // Alias for the process id of lldb-platform 2500e305d2STamas Berghammer 2600e305d2STamas Berghammer static Error 27e7eabbb5SOleksiy Vyalov ForwardPortWithAdb (const uint16_t local_port, const uint16_t remote_port, std::string& device_id) 2800e305d2STamas Berghammer { 29db264a6dSTamas Berghammer Log *log(GetLogIfAllCategoriesSet (LIBLLDB_LOG_PLATFORM)); 3000e305d2STamas Berghammer 3105a55de3SOleksiy Vyalov AdbClient adb; 323ea689b3SChaoren Lin auto error = AdbClient::CreateByDeviceID(device_id, adb); 3300e305d2STamas Berghammer if (error.Fail ()) 3400e305d2STamas Berghammer return error; 3500e305d2STamas Berghammer 36f9da9483SOleksiy Vyalov device_id = adb.GetDeviceID(); 3705a55de3SOleksiy Vyalov if (log) 38f9da9483SOleksiy Vyalov log->Printf("Connected to Android device \"%s\"", device_id.c_str ()); 3905a55de3SOleksiy Vyalov 40e7eabbb5SOleksiy Vyalov return adb.SetPortForwarding(local_port, remote_port); 4100e305d2STamas Berghammer } 4200e305d2STamas Berghammer 4300e305d2STamas Berghammer static Error 44e7eabbb5SOleksiy Vyalov DeleteForwardPortWithAdb (uint16_t local_port, const std::string& device_id) 4500e305d2STamas Berghammer { 4605a55de3SOleksiy Vyalov AdbClient adb (device_id); 47e7eabbb5SOleksiy Vyalov return adb.DeletePortForwarding (local_port); 48e7eabbb5SOleksiy Vyalov } 49e7eabbb5SOleksiy Vyalov 50e7eabbb5SOleksiy Vyalov static Error 51e7eabbb5SOleksiy Vyalov FindUnusedPort (uint16_t& port) 52e7eabbb5SOleksiy Vyalov { 53*e98628ceSOleksiy Vyalov Error error; 54*e98628ceSOleksiy Vyalov std::unique_ptr<TCPSocket> tcp_socket(new TCPSocket(false, error)); 55*e98628ceSOleksiy Vyalov if (error.Fail()) 56*e98628ceSOleksiy Vyalov return error; 57*e98628ceSOleksiy Vyalov 58*e98628ceSOleksiy Vyalov error = tcp_socket->Listen("127.0.0.1:0", 1); 59e7eabbb5SOleksiy Vyalov if (error.Success()) 60*e98628ceSOleksiy Vyalov port = tcp_socket->GetLocalPortNumber(); 61*e98628ceSOleksiy Vyalov 62e7eabbb5SOleksiy Vyalov return error; 6300e305d2STamas Berghammer } 6400e305d2STamas Berghammer 6500e305d2STamas Berghammer PlatformAndroidRemoteGDBServer::PlatformAndroidRemoteGDBServer () 6600e305d2STamas Berghammer { 6700e305d2STamas Berghammer } 6800e305d2STamas Berghammer 6900e305d2STamas Berghammer PlatformAndroidRemoteGDBServer::~PlatformAndroidRemoteGDBServer () 7000e305d2STamas Berghammer { 7100e305d2STamas Berghammer for (const auto& it : m_port_forwards) 723ea689b3SChaoren Lin DeleteForwardPortWithAdb(it.second, m_device_id); 7300e305d2STamas Berghammer } 7400e305d2STamas Berghammer 7500e305d2STamas Berghammer uint16_t 7600e305d2STamas Berghammer PlatformAndroidRemoteGDBServer::LaunchGDBserverAndGetPort (lldb::pid_t &pid) 7700e305d2STamas Berghammer { 78e7eabbb5SOleksiy Vyalov uint16_t remote_port = m_gdb_client.LaunchGDBserverAndGetPort (pid, "127.0.0.1"); 79e7eabbb5SOleksiy Vyalov if (remote_port == 0) 80e7eabbb5SOleksiy Vyalov return remote_port; 8100e305d2STamas Berghammer 82e7eabbb5SOleksiy Vyalov uint16_t local_port = 0; 83e7eabbb5SOleksiy Vyalov auto error = SetPortForwarding (pid, remote_port, local_port); 84e7eabbb5SOleksiy Vyalov return error.Success() ? local_port : 0; 8500e305d2STamas Berghammer } 8600e305d2STamas Berghammer 8700e305d2STamas Berghammer bool 8800e305d2STamas Berghammer PlatformAndroidRemoteGDBServer::KillSpawnedProcess (lldb::pid_t pid) 8900e305d2STamas Berghammer { 901c1d76b3SOleksiy Vyalov DeleteForwardPort (pid); 9100e305d2STamas Berghammer return m_gdb_client.KillSpawnedProcess (pid); 9200e305d2STamas Berghammer } 9300e305d2STamas Berghammer 9400e305d2STamas Berghammer Error 9500e305d2STamas Berghammer PlatformAndroidRemoteGDBServer::ConnectRemote (Args& args) 9600e305d2STamas Berghammer { 973ea689b3SChaoren Lin m_device_id.clear(); 983ea689b3SChaoren Lin 9900e305d2STamas Berghammer if (args.GetArgumentCount() != 1) 10000e305d2STamas Berghammer return Error("\"platform connect\" takes a single argument: <connect-url>"); 10100e305d2STamas Berghammer 102e7eabbb5SOleksiy Vyalov int remote_port; 10300e305d2STamas Berghammer std::string scheme, host, path; 10400e305d2STamas Berghammer const char *url = args.GetArgumentAtIndex (0); 1053ea689b3SChaoren Lin if (!url) 1063ea689b3SChaoren Lin return Error("URL is null."); 107e7eabbb5SOleksiy Vyalov if (!UriParser::Parse (url, scheme, host, remote_port, path)) 1083ea689b3SChaoren Lin return Error("Invalid URL: %s", url); 1093ea689b3SChaoren Lin if (scheme == "adb") 1103ea689b3SChaoren Lin m_device_id = host; 11100e305d2STamas Berghammer 112e7eabbb5SOleksiy Vyalov uint16_t local_port = 0; 113e7eabbb5SOleksiy Vyalov auto error = SetPortForwarding (g_remote_platform_pid, remote_port, local_port); 11400e305d2STamas Berghammer if (error.Fail ()) 11500e305d2STamas Berghammer return error; 11600e305d2STamas Berghammer 117e7eabbb5SOleksiy Vyalov const std::string new_url = MakeUrl( 118e7eabbb5SOleksiy Vyalov scheme.c_str(), host.c_str(), local_port, path.c_str()); 119e7eabbb5SOleksiy Vyalov args.ReplaceArgumentAtIndex (0, new_url.c_str ()); 120e7eabbb5SOleksiy Vyalov 121e7eabbb5SOleksiy Vyalov Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM)); 122e7eabbb5SOleksiy Vyalov if (log) 123e7eabbb5SOleksiy Vyalov log->Printf("Rewritten URL: %s", new_url.c_str()); 12400e305d2STamas Berghammer 1251c1d76b3SOleksiy Vyalov error = PlatformRemoteGDBServer::ConnectRemote(args); 1261c1d76b3SOleksiy Vyalov if (error.Fail ()) 1271c1d76b3SOleksiy Vyalov DeleteForwardPort (g_remote_platform_pid); 1281c1d76b3SOleksiy Vyalov 1291c1d76b3SOleksiy Vyalov return error; 13000e305d2STamas Berghammer } 13100e305d2STamas Berghammer 13200e305d2STamas Berghammer Error 13300e305d2STamas Berghammer PlatformAndroidRemoteGDBServer::DisconnectRemote () 13400e305d2STamas Berghammer { 1351c1d76b3SOleksiy Vyalov DeleteForwardPort (g_remote_platform_pid); 1361c1d76b3SOleksiy Vyalov return PlatformRemoteGDBServer::DisconnectRemote (); 13700e305d2STamas Berghammer } 13800e305d2STamas Berghammer 1391c1d76b3SOleksiy Vyalov void 1401c1d76b3SOleksiy Vyalov PlatformAndroidRemoteGDBServer::DeleteForwardPort (lldb::pid_t pid) 1411c1d76b3SOleksiy Vyalov { 1421c1d76b3SOleksiy Vyalov Log *log(GetLogIfAllCategoriesSet(LIBLLDB_LOG_PLATFORM)); 1431c1d76b3SOleksiy Vyalov 1441c1d76b3SOleksiy Vyalov auto it = m_port_forwards.find(pid); 1451c1d76b3SOleksiy Vyalov if (it == m_port_forwards.end()) 1461c1d76b3SOleksiy Vyalov return; 1471c1d76b3SOleksiy Vyalov 1483ea689b3SChaoren Lin const auto port = it->second; 1493ea689b3SChaoren Lin const auto error = DeleteForwardPortWithAdb(port, m_device_id); 1501c1d76b3SOleksiy Vyalov if (error.Fail()) { 1511c1d76b3SOleksiy Vyalov if (log) 1521c1d76b3SOleksiy Vyalov log->Printf("Failed to delete port forwarding (pid=%" PRIu64 ", port=%d, device=%s): %s", 1533ea689b3SChaoren Lin pid, port, m_device_id.c_str(), error.AsCString()); 1541c1d76b3SOleksiy Vyalov } 1551c1d76b3SOleksiy Vyalov m_port_forwards.erase(it); 15600e305d2STamas Berghammer } 15754971856SOleksiy Vyalov 158e7eabbb5SOleksiy Vyalov Error 159e7eabbb5SOleksiy Vyalov PlatformAndroidRemoteGDBServer::SetPortForwarding(const lldb::pid_t pid, 160e7eabbb5SOleksiy Vyalov const uint16_t remote_port, 161e7eabbb5SOleksiy Vyalov uint16_t &local_port) 162e7eabbb5SOleksiy Vyalov { 163e7eabbb5SOleksiy Vyalov static const int kAttempsNum = 5; 164e7eabbb5SOleksiy Vyalov 165e7eabbb5SOleksiy Vyalov Error error; 166e7eabbb5SOleksiy Vyalov // There is a race possibility that somebody will occupy 167e7eabbb5SOleksiy Vyalov // a port while we're in between FindUnusedPort and ForwardPortWithAdb - 168e7eabbb5SOleksiy Vyalov // adding the loop to mitigate such problem. 169e7eabbb5SOleksiy Vyalov for (auto i = 0; i < kAttempsNum; ++i) 170e7eabbb5SOleksiy Vyalov { 171e7eabbb5SOleksiy Vyalov error = FindUnusedPort(local_port); 172e7eabbb5SOleksiy Vyalov if (error.Fail()) 173e7eabbb5SOleksiy Vyalov return error; 174e7eabbb5SOleksiy Vyalov 175e7eabbb5SOleksiy Vyalov error = ForwardPortWithAdb(local_port, remote_port, m_device_id); 176e7eabbb5SOleksiy Vyalov if (error.Success()) 177e7eabbb5SOleksiy Vyalov { 178e7eabbb5SOleksiy Vyalov m_port_forwards[pid] = local_port; 179e7eabbb5SOleksiy Vyalov break; 180e7eabbb5SOleksiy Vyalov } 181e7eabbb5SOleksiy Vyalov } 182e7eabbb5SOleksiy Vyalov 183e7eabbb5SOleksiy Vyalov return error; 184e7eabbb5SOleksiy Vyalov } 185e7eabbb5SOleksiy Vyalov 18654971856SOleksiy Vyalov std::string 187e7eabbb5SOleksiy Vyalov PlatformAndroidRemoteGDBServer::MakeUrl(const char* scheme, 18854971856SOleksiy Vyalov const char* hostname, 189e7eabbb5SOleksiy Vyalov uint16_t port, 190e7eabbb5SOleksiy Vyalov const char* path) 19154971856SOleksiy Vyalov { 19254971856SOleksiy Vyalov std::ostringstream hostname_str; 19354971856SOleksiy Vyalov if (!strcmp(scheme, "adb")) 19454971856SOleksiy Vyalov hostname_str << "[" << hostname << "]"; 19554971856SOleksiy Vyalov else 19654971856SOleksiy Vyalov hostname_str << hostname; 19754971856SOleksiy Vyalov 198e7eabbb5SOleksiy Vyalov return PlatformRemoteGDBServer::MakeUrl(scheme, 19954971856SOleksiy Vyalov hostname_str.str().c_str(), 200e7eabbb5SOleksiy Vyalov port, 201e7eabbb5SOleksiy Vyalov path); 20254971856SOleksiy Vyalov } 203