The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2007 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
Yabin Cui | 19bec5b | 2015-09-22 15:52:57 -0700 | [diff] [blame] | 17 | #define TRACE_TAG TRANSPORT |
Dan Albert | db6fe64 | 2015-03-19 15:21:08 -0700 | [diff] [blame] | 18 | |
| 19 | #include "sysdeps.h" |
| 20 | #include "transport.h" |
| 21 | |
Dan Albert | b302d12 | 2015-02-24 15:51:19 -0800 | [diff] [blame] | 22 | #include <errno.h> |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 23 | #include <stdio.h> |
| 24 | #include <stdlib.h> |
| 25 | #include <string.h> |
Dan Albert | b302d12 | 2015-02-24 15:51:19 -0800 | [diff] [blame] | 26 | #include <sys/types.h> |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 27 | |
Pirama Arumuga Nainar | bd4d4e1 | 2016-09-06 13:34:42 -0700 | [diff] [blame] | 28 | #include <condition_variable> |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 29 | #include <functional> |
Pirama Arumuga Nainar | 5231aff | 2018-08-08 10:33:24 -0700 | [diff] [blame] | 30 | #include <memory> |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 31 | #include <mutex> |
Elliott Hughes | 7392598 | 2016-11-15 12:37:32 -0800 | [diff] [blame] | 32 | #include <thread> |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 33 | #include <unordered_map> |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 34 | #include <vector> |
| 35 | |
Casey Dahlin | 3122cdf | 2016-06-23 14:19:37 -0700 | [diff] [blame] | 36 | #include <android-base/parsenetaddress.h> |
Elliott Hughes | f55ead9 | 2015-12-04 22:00:26 -0800 | [diff] [blame] | 37 | #include <android-base/stringprintf.h> |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 38 | #include <android-base/thread_annotations.h> |
Elliott Hughes | 43df109 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 39 | #include <cutils/sockets.h> |
Elliott Hughes | fb59684 | 2015-05-01 17:04:38 -0700 | [diff] [blame] | 40 | |
Dan Albert | b302d12 | 2015-02-24 15:51:19 -0800 | [diff] [blame] | 41 | #if !ADB_HOST |
Elliott Hughes | 8b249d2 | 2016-09-23 15:40:03 -0700 | [diff] [blame] | 42 | #include <android-base/properties.h> |
Dan Albert | b302d12 | 2015-02-24 15:51:19 -0800 | [diff] [blame] | 43 | #endif |
Dan Albert | db6fe64 | 2015-03-19 15:21:08 -0700 | [diff] [blame] | 44 | |
| 45 | #include "adb.h" |
| 46 | #include "adb_io.h" |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 47 | #include "adb_unique_fd.h" |
Elliott Hughes | 43df109 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 48 | #include "adb_utils.h" |
Cody Schuffelen | 331a908 | 2019-01-02 14:17:29 -0800 | [diff] [blame] | 49 | #include "socket_spec.h" |
Josh Gao | 70267e4 | 2016-11-15 18:55:47 -0800 | [diff] [blame] | 50 | #include "sysdeps/chrono.h" |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 51 | |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 52 | #if ADB_HOST |
Josh Gao | 89513a5 | 2016-05-06 02:37:24 -0700 | [diff] [blame] | 53 | |
| 54 | // Android Wear has been using port 5601 in all of its documentation/tooling, |
| 55 | // but we search for emulators on ports [5554, 5555 + ADB_LOCAL_TRANSPORT_MAX]. |
Tim Baverstock | 96ee6b1 | 2019-02-08 16:27:16 +0000 | [diff] [blame] | 56 | // Avoid stomping on their port by restricting the active scanning range. |
| 57 | // Once emulators self-(re-)register, they'll have to avoid 5601 in their own way. |
| 58 | static int adb_local_transport_max_port = DEFAULT_ADB_LOCAL_TRANSPORT_PORT + 16 * 2 - 1; |
Josh Gao | 89513a5 | 2016-05-06 02:37:24 -0700 | [diff] [blame] | 59 | |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 60 | static std::mutex& local_transports_lock = *new std::mutex(); |
Josh Gao | 89513a5 | 2016-05-06 02:37:24 -0700 | [diff] [blame] | 61 | |
Tim Baverstock | 96ee6b1 | 2019-02-08 16:27:16 +0000 | [diff] [blame] | 62 | static void adb_local_transport_max_port_env_override() { |
| 63 | const char* env_max_s = getenv("ADB_LOCAL_TRANSPORT_MAX_PORT"); |
| 64 | if (env_max_s != nullptr) { |
| 65 | size_t env_max; |
| 66 | if (ParseUint(&env_max, env_max_s, nullptr) && env_max < 65536) { |
| 67 | // < DEFAULT_ADB_LOCAL_TRANSPORT_PORT harmlessly mimics ADB_EMU=0 |
| 68 | adb_local_transport_max_port = env_max; |
| 69 | D("transport: ADB_LOCAL_TRANSPORT_MAX_PORT read as %d", adb_local_transport_max_port); |
| 70 | } else { |
| 71 | D("transport: ADB_LOCAL_TRANSPORT_MAX_PORT '%s' invalid or >= 65536, so ignored", |
| 72 | env_max_s); |
| 73 | } |
| 74 | } |
| 75 | } |
| 76 | |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 77 | // We keep a map from emulator port to transport. |
| 78 | // TODO: weak_ptr? |
| 79 | static auto& local_transports GUARDED_BY(local_transports_lock) = |
| 80 | *new std::unordered_map<int, atransport*>(); |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 81 | #endif /* ADB_HOST */ |
| 82 | |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 83 | bool local_connect(int port) { |
Elliott Hughes | 43df109 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 84 | std::string dummy; |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 85 | return local_connect_arbitrary_ports(port - 1, port, &dummy) == 0; |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 86 | } |
| 87 | |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 88 | void connect_device(const std::string& address, std::string* response) { |
| 89 | if (address.empty()) { |
| 90 | *response = "empty address"; |
| 91 | return; |
| 92 | } |
| 93 | |
Josh Gao | 8d7080c | 2018-08-10 16:03:09 -0700 | [diff] [blame] | 94 | D("connection requested to '%s'", address.c_str()); |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 95 | unique_fd fd; |
| 96 | int port; |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 97 | std::string serial, prefix_addr; |
| 98 | |
| 99 | // If address does not match any socket type, it should default to TCP. |
| 100 | if (address.starts_with("vsock:") || address.starts_with("localfilesystem:")) { |
| 101 | prefix_addr = address; |
| 102 | } else { |
| 103 | prefix_addr = "tcp:" + address; |
| 104 | } |
| 105 | |
| 106 | socket_spec_connect(&fd, prefix_addr, &port, &serial, response); |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 107 | if (fd.get() == -1) { |
| 108 | return; |
| 109 | } |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 110 | auto reconnect = [prefix_addr](atransport* t) { |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 111 | std::string response; |
| 112 | unique_fd fd; |
| 113 | int port; |
| 114 | std::string serial; |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 115 | socket_spec_connect(&fd, prefix_addr, &port, &serial, &response); |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 116 | if (fd == -1) { |
| 117 | D("reconnect failed: %s", response.c_str()); |
Josh Gao | d24580d | 2018-08-30 11:37:00 -0700 | [diff] [blame] | 118 | return ReconnectResult::Retry; |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 119 | } |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 120 | // This invokes the part of register_socket_transport() that needs to be |
| 121 | // invoked if the atransport* has already been setup. This eventually |
| 122 | // calls atransport->SetConnection() with a newly created Connection* |
| 123 | // that will in turn send the CNXN packet. |
Josh Gao | d24580d | 2018-08-30 11:37:00 -0700 | [diff] [blame] | 124 | return init_socket_transport(t, std::move(fd), port, 0) >= 0 ? ReconnectResult::Success |
| 125 | : ReconnectResult::Retry; |
Luis Hector Chavez | 0aeda10 | 2018-04-20 10:31:29 -0700 | [diff] [blame] | 126 | }; |
| 127 | |
Josh Gao | 597044d | 2018-08-08 16:20:14 -0700 | [diff] [blame] | 128 | int error; |
Joshua Duong | 64fab75 | 2020-01-21 13:19:42 -0800 | [diff] [blame] | 129 | if (!register_socket_transport(std::move(fd), serial, port, 0, std::move(reconnect), false, |
| 130 | &error)) { |
Josh Gao | 597044d | 2018-08-08 16:20:14 -0700 | [diff] [blame] | 131 | if (error == EALREADY) { |
Luis Hector Chavez | 7771920 | 2018-04-17 14:09:21 -0700 | [diff] [blame] | 132 | *response = android::base::StringPrintf("already connected to %s", serial.c_str()); |
Josh Gao | 597044d | 2018-08-08 16:20:14 -0700 | [diff] [blame] | 133 | } else if (error == EPERM) { |
| 134 | *response = android::base::StringPrintf("failed to authenticate to %s", serial.c_str()); |
Luis Hector Chavez | 7771920 | 2018-04-17 14:09:21 -0700 | [diff] [blame] | 135 | } else { |
| 136 | *response = android::base::StringPrintf("failed to connect to %s", serial.c_str()); |
| 137 | } |
Casey Dahlin | 3122cdf | 2016-06-23 14:19:37 -0700 | [diff] [blame] | 138 | } else { |
| 139 | *response = android::base::StringPrintf("connected to %s", serial.c_str()); |
| 140 | } |
| 141 | } |
| 142 | |
| 143 | |
Elliott Hughes | 43df109 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 144 | int local_connect_arbitrary_ports(int console_port, int adb_port, std::string* error) { |
Josh Gao | d8d51f4 | 2018-07-25 17:27:34 -0700 | [diff] [blame] | 145 | unique_fd fd; |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 146 | |
| 147 | #if ADB_HOST |
Lingfeng Yang | 5edb12b | 2016-10-06 12:22:55 -0700 | [diff] [blame] | 148 | if (find_emulator_transport_by_adb_port(adb_port) != nullptr || |
| 149 | find_emulator_transport_by_console_port(console_port) != nullptr) { |
Yabin Cui | d802bcf | 2015-07-31 14:10:54 -0700 | [diff] [blame] | 150 | return -1; |
| 151 | } |
| 152 | |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 153 | const char *host = getenv("ADBHOST"); |
| 154 | if (host) { |
Josh Gao | d8d51f4 | 2018-07-25 17:27:34 -0700 | [diff] [blame] | 155 | fd.reset(network_connect(host, adb_port, SOCK_STREAM, 0, error)); |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 156 | } |
| 157 | #endif |
| 158 | if (fd < 0) { |
Josh Gao | d8d51f4 | 2018-07-25 17:27:34 -0700 | [diff] [blame] | 159 | fd.reset(network_loopback_client(adb_port, SOCK_STREAM, error)); |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 160 | } |
| 161 | |
| 162 | if (fd >= 0) { |
Josh Gao | d8d51f4 | 2018-07-25 17:27:34 -0700 | [diff] [blame] | 163 | D("client: connected on remote on fd %d", fd.get()); |
| 164 | close_on_exec(fd.get()); |
| 165 | disable_tcp_nagle(fd.get()); |
Lingfeng Yang | 5edb12b | 2016-10-06 12:22:55 -0700 | [diff] [blame] | 166 | std::string serial = getEmulatorSerialString(console_port); |
Joshua Duong | 64fab75 | 2020-01-21 13:19:42 -0800 | [diff] [blame] | 167 | if (register_socket_transport( |
| 168 | std::move(fd), std::move(serial), adb_port, 1, |
| 169 | [](atransport*) { return ReconnectResult::Abort; }, false)) { |
Yabin Cui | d802bcf | 2015-07-31 14:10:54 -0700 | [diff] [blame] | 170 | return 0; |
| 171 | } |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 172 | } |
| 173 | return -1; |
| 174 | } |
| 175 | |
Yabin Cui | 5fe6d0d | 2015-08-11 13:40:42 -0700 | [diff] [blame] | 176 | #if ADB_HOST |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 177 | |
| 178 | static void PollAllLocalPortsForEmulator() { |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 179 | // Try to connect to any number of running emulator instances. |
Tim Baverstock | 96ee6b1 | 2019-02-08 16:27:16 +0000 | [diff] [blame] | 180 | for (int port = DEFAULT_ADB_LOCAL_TRANSPORT_PORT; port <= adb_local_transport_max_port; |
| 181 | port += 2) { |
| 182 | local_connect(port); // Note, uses port and port-1, so '=max_port' is OK. |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 183 | } |
| 184 | } |
| 185 | |
| 186 | // Retry the disconnected local port for 60 times, and sleep 1 second between two retries. |
Elliott Hughes | c3462f4 | 2018-09-05 12:13:11 -0700 | [diff] [blame] | 187 | static constexpr uint32_t LOCAL_PORT_RETRY_COUNT = 60; |
| 188 | static constexpr auto LOCAL_PORT_RETRY_INTERVAL = 1s; |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 189 | |
| 190 | struct RetryPort { |
| 191 | int port; |
| 192 | uint32_t retry_count; |
| 193 | }; |
| 194 | |
| 195 | // Retry emulators just kicked. |
| 196 | static std::vector<RetryPort>& retry_ports = *new std::vector<RetryPort>; |
| 197 | std::mutex &retry_ports_lock = *new std::mutex; |
| 198 | std::condition_variable &retry_ports_cond = *new std::condition_variable; |
| 199 | |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 200 | static void client_socket_thread(std::string_view) { |
Siva Velusamy | 2669cf9 | 2015-08-28 16:37:29 -0700 | [diff] [blame] | 201 | adb_thread_setname("client_socket_thread"); |
Yabin Cui | 815ad88 | 2015-09-02 17:44:28 -0700 | [diff] [blame] | 202 | D("transport: client_socket_thread() starting"); |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 203 | PollAllLocalPortsForEmulator(); |
Yabin Cui | d802bcf | 2015-07-31 14:10:54 -0700 | [diff] [blame] | 204 | while (true) { |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 205 | std::vector<RetryPort> ports; |
| 206 | // Collect retry ports. |
| 207 | { |
| 208 | std::unique_lock<std::mutex> lock(retry_ports_lock); |
| 209 | while (retry_ports.empty()) { |
| 210 | retry_ports_cond.wait(lock); |
| 211 | } |
| 212 | retry_ports.swap(ports); |
Yabin Cui | d802bcf | 2015-07-31 14:10:54 -0700 | [diff] [blame] | 213 | } |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 214 | // Sleep here instead of the end of loop, because if we immediately try to reconnect |
| 215 | // the emulator just kicked, the adbd on the emulator may not have time to remove the |
| 216 | // just kicked transport. |
Elliott Hughes | 7392598 | 2016-11-15 12:37:32 -0800 | [diff] [blame] | 217 | std::this_thread::sleep_for(LOCAL_PORT_RETRY_INTERVAL); |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 218 | |
| 219 | // Try connecting retry ports. |
| 220 | std::vector<RetryPort> next_ports; |
| 221 | for (auto& port : ports) { |
| 222 | VLOG(TRANSPORT) << "retry port " << port.port << ", last retry_count " |
| 223 | << port.retry_count; |
| 224 | if (local_connect(port.port)) { |
| 225 | VLOG(TRANSPORT) << "retry port " << port.port << " successfully"; |
| 226 | continue; |
| 227 | } |
| 228 | if (--port.retry_count > 0) { |
| 229 | next_ports.push_back(port); |
| 230 | } else { |
| 231 | VLOG(TRANSPORT) << "stop retrying port " << port.port; |
| 232 | } |
| 233 | } |
| 234 | |
| 235 | // Copy back left retry ports. |
| 236 | { |
| 237 | std::unique_lock<std::mutex> lock(retry_ports_lock); |
| 238 | retry_ports.insert(retry_ports.end(), next_ports.begin(), next_ports.end()); |
| 239 | } |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 240 | } |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 241 | } |
| 242 | |
Josh Gao | 0560feb | 2019-01-22 19:36:15 -0800 | [diff] [blame] | 243 | #else // !ADB_HOST |
Yabin Cui | 5fe6d0d | 2015-08-11 13:40:42 -0700 | [diff] [blame] | 244 | |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 245 | void server_socket_thread(std::function<unique_fd(std::string_view, std::string*)> listen_func, |
| 246 | std::string_view addr) { |
Siva Velusamy | 2669cf9 | 2015-08-28 16:37:29 -0700 | [diff] [blame] | 247 | adb_thread_setname("server socket"); |
Josh Gao | 80e8192 | 2019-01-30 13:59:51 -0800 | [diff] [blame] | 248 | |
| 249 | unique_fd serverfd; |
| 250 | std::string error; |
| 251 | |
Josh Gao | fa3107e | 2018-07-25 17:21:49 -0700 | [diff] [blame] | 252 | while (serverfd == -1) { |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 253 | errno = 0; |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 254 | serverfd = listen_func(addr, &error); |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 255 | if (errno == EAFNOSUPPORT || errno == EINVAL || errno == EPROTONOSUPPORT) { |
| 256 | D("unrecoverable error: '%s'", error.c_str()); |
| 257 | return; |
| 258 | } else if (serverfd < 0) { |
Josh Gao | fa3107e | 2018-07-25 17:21:49 -0700 | [diff] [blame] | 259 | D("server: cannot bind socket yet: %s", error.c_str()); |
| 260 | std::this_thread::sleep_for(1s); |
| 261 | continue; |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 262 | } |
Josh Gao | fa3107e | 2018-07-25 17:21:49 -0700 | [diff] [blame] | 263 | close_on_exec(serverfd.get()); |
| 264 | } |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 265 | |
Josh Gao | fa3107e | 2018-07-25 17:21:49 -0700 | [diff] [blame] | 266 | while (true) { |
Josh Gao | 80e8192 | 2019-01-30 13:59:51 -0800 | [diff] [blame] | 267 | D("server: trying to get new connection from fd %d", serverfd.get()); |
Josh Gao | fa3107e | 2018-07-25 17:21:49 -0700 | [diff] [blame] | 268 | unique_fd fd(adb_socket_accept(serverfd, nullptr, nullptr)); |
| 269 | if (fd >= 0) { |
| 270 | D("server: new connection on fd %d", fd.get()); |
| 271 | close_on_exec(fd.get()); |
| 272 | disable_tcp_nagle(fd.get()); |
| 273 | std::string serial = android::base::StringPrintf("host-%d", fd.get()); |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 274 | // We don't care about port value in "register_socket_transport" as it is used |
| 275 | // only from ADB_HOST. "server_socket_thread" is never called from ADB_HOST. |
Joshua Duong | 64fab75 | 2020-01-21 13:19:42 -0800 | [diff] [blame] | 276 | register_socket_transport( |
| 277 | std::move(fd), std::move(serial), 0, 1, |
| 278 | [](atransport*) { return ReconnectResult::Abort; }, false); |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 279 | } |
| 280 | } |
Yabin Cui | 815ad88 | 2015-09-02 17:44:28 -0700 | [diff] [blame] | 281 | D("transport: server_socket_thread() exiting"); |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 282 | } |
| 283 | |
Josh Gao | 0560feb | 2019-01-22 19:36:15 -0800 | [diff] [blame] | 284 | #endif |
Vladimir Chtchetkine | 7c9339d | 2011-12-09 15:49:47 -0800 | [diff] [blame] | 285 | |
Josh Gao | 80e8192 | 2019-01-30 13:59:51 -0800 | [diff] [blame] | 286 | #if !ADB_HOST |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 287 | unique_fd adb_listen(std::string_view addr, std::string* error) { |
| 288 | return unique_fd{socket_spec_listen(addr, error, nullptr)}; |
Josh Gao | 80e8192 | 2019-01-30 13:59:51 -0800 | [diff] [blame] | 289 | } |
| 290 | #endif |
| 291 | |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 292 | void local_init(const std::string& addr) { |
Vladimir Chtchetkine | 4bdc776 | 2011-12-13 12:19:29 -0800 | [diff] [blame] | 293 | #if ADB_HOST |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 294 | D("transport: local client init"); |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 295 | std::thread(client_socket_thread, addr).detach(); |
Tim Baverstock | 96ee6b1 | 2019-02-08 16:27:16 +0000 | [diff] [blame] | 296 | adb_local_transport_max_port_env_override(); |
Josh Gao | 0560feb | 2019-01-22 19:36:15 -0800 | [diff] [blame] | 297 | #elif !defined(__ANDROID__) |
| 298 | // Host adbd. |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 299 | D("transport: local server init"); |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 300 | std::thread(server_socket_thread, adb_listen, addr).detach(); |
Vladimir Chtchetkine | 4bdc776 | 2011-12-13 12:19:29 -0800 | [diff] [blame] | 301 | #else |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 302 | D("transport: local server init"); |
Elliott Hughes | 8b249d2 | 2016-09-23 15:40:03 -0700 | [diff] [blame] | 303 | // For the adbd daemon in the system image we need to distinguish |
| 304 | // between the device, and the emulator. |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 305 | if (addr.starts_with("tcp:") && use_qemu_goldfish()) { |
| 306 | std::thread(qemu_socket_thread, addr).detach(); |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 307 | } else { |
Jason Jeremy Iman | 8461387 | 2019-07-19 12:44:39 +0900 | [diff] [blame] | 308 | std::thread(server_socket_thread, adb_listen, addr).detach(); |
Cody Schuffelen | 637aaf5 | 2019-01-04 18:51:11 -0800 | [diff] [blame] | 309 | } |
Yabin Cui | 5fe6d0d | 2015-08-11 13:40:42 -0700 | [diff] [blame] | 310 | #endif // !ADB_HOST |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 311 | } |
| 312 | |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 313 | #if ADB_HOST |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 314 | struct EmulatorConnection : public FdConnection { |
| 315 | EmulatorConnection(unique_fd fd, int local_port) |
| 316 | : FdConnection(std::move(fd)), local_port_(local_port) {} |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 317 | |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 318 | ~EmulatorConnection() { |
| 319 | VLOG(TRANSPORT) << "remote_close, local_port = " << local_port_; |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 320 | std::unique_lock<std::mutex> lock(retry_ports_lock); |
| 321 | RetryPort port; |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 322 | port.port = local_port_; |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 323 | port.retry_count = LOCAL_PORT_RETRY_COUNT; |
| 324 | retry_ports.push_back(port); |
| 325 | retry_ports_cond.notify_one(); |
| 326 | } |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 327 | |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 328 | void Close() override { |
| 329 | std::lock_guard<std::mutex> lock(local_transports_lock); |
| 330 | local_transports.erase(local_port_); |
| 331 | FdConnection::Close(); |
| 332 | } |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 333 | |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 334 | int local_port_; |
| 335 | }; |
| 336 | |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 337 | /* Only call this function if you already hold local_transports_lock. */ |
Yabin Cui | f401ead | 2016-04-29 16:53:52 -0700 | [diff] [blame] | 338 | static atransport* find_emulator_transport_by_adb_port_locked(int adb_port) |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 339 | REQUIRES(local_transports_lock) { |
| 340 | auto it = local_transports.find(adb_port); |
| 341 | if (it == local_transports.end()) { |
| 342 | return nullptr; |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 343 | } |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 344 | return it->second; |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 345 | } |
| 346 | |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 347 | atransport* find_emulator_transport_by_adb_port(int adb_port) { |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 348 | std::lock_guard<std::mutex> lock(local_transports_lock); |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 349 | return find_emulator_transport_by_adb_port_locked(adb_port); |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 350 | } |
| 351 | |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 352 | atransport* find_emulator_transport_by_console_port(int console_port) { |
Lingfeng Yang | 5edb12b | 2016-10-06 12:22:55 -0700 | [diff] [blame] | 353 | return find_transport(getEmulatorSerialString(console_port).c_str()); |
| 354 | } |
Stefan Hilzinger | 1ec0342 | 2010-04-26 10:17:43 +0100 | [diff] [blame] | 355 | #endif |
| 356 | |
Tao Bao | 49042e3 | 2018-07-30 20:45:27 -0700 | [diff] [blame] | 357 | std::string getEmulatorSerialString(int console_port) { |
| 358 | return android::base::StringPrintf("emulator-%d", console_port); |
| 359 | } |
| 360 | |
Josh Gao | fa3107e | 2018-07-25 17:21:49 -0700 | [diff] [blame] | 361 | int init_socket_transport(atransport* t, unique_fd fd, int adb_port, int local) { |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 362 | int fail = 0; |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 363 | |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 364 | t->type = kTransportLocal; |
| 365 | |
| 366 | #if ADB_HOST |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 367 | // Emulator connection. |
Yabin Cui | 5fe6d0d | 2015-08-11 13:40:42 -0700 | [diff] [blame] | 368 | if (local) { |
Josh Gao | f2a988c | 2018-03-07 16:51:08 -0800 | [diff] [blame] | 369 | auto emulator_connection = std::make_unique<EmulatorConnection>(std::move(fd), adb_port); |
Luis Hector Chavez | 3c7881d | 2018-04-25 08:56:41 -0700 | [diff] [blame] | 370 | t->SetConnection( |
Joshua Duong | 64fab75 | 2020-01-21 13:19:42 -0800 | [diff] [blame] | 371 | std::make_unique<BlockingConnectionAdapter>(std::move(emulator_connection))); |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 372 | std::lock_guard<std::mutex> lock(local_transports_lock); |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 373 | atransport* existing_transport = find_emulator_transport_by_adb_port_locked(adb_port); |
Yi Kong | 86e6718 | 2018-07-13 18:15:16 -0700 | [diff] [blame] | 374 | if (existing_transport != nullptr) { |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 375 | D("local transport for port %d already registered (%p)?", adb_port, existing_transport); |
| 376 | fail = -1; |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 377 | } else { |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 378 | local_transports[adb_port] = t; |
Josh Gao | e7daf57 | 2016-09-21 12:37:10 -0700 | [diff] [blame] | 379 | } |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 380 | |
| 381 | return fail; |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 382 | } |
| 383 | #endif |
Josh Gao | 395b86a | 2018-01-28 20:32:46 -0800 | [diff] [blame] | 384 | |
| 385 | // Regular tcp connection. |
Josh Gao | f2a988c | 2018-03-07 16:51:08 -0800 | [diff] [blame] | 386 | auto fd_connection = std::make_unique<FdConnection>(std::move(fd)); |
Luis Hector Chavez | 3c7881d | 2018-04-25 08:56:41 -0700 | [diff] [blame] | 387 | t->SetConnection(std::make_unique<BlockingConnectionAdapter>(std::move(fd_connection))); |
The Android Open Source Project | 9ca14dc | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 388 | return fail; |
| 389 | } |