The Android Open Source Project | dd7bc33 | 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 | aed3c61 | 2015-09-22 15:52:57 -0700 | [diff] [blame] | 17 | #define TRACE_TAG SERVICES |
Dan Albert | 3313426 | 2015-03-19 15:21:08 -0700 | [diff] [blame] | 18 | |
| 19 | #include "sysdeps.h" |
| 20 | |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 21 | #include <errno.h> |
Dan Albert | 7664901 | 2015-02-24 15:51:19 -0800 | [diff] [blame] | 22 | #include <stddef.h> |
| 23 | #include <stdio.h> |
| 24 | #include <stdlib.h> |
| 25 | #include <string.h> |
| 26 | |
Josh Gao | e1dacfc | 2017-04-12 17:00:49 -0700 | [diff] [blame] | 27 | #include <thread> |
Elliott Hughes | 4f71319 | 2015-12-04 22:00:26 -0800 | [diff] [blame] | 28 | #include <android-base/stringprintf.h> |
| 29 | #include <android-base/strings.h> |
Elliott Hughes | 381cfa9 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 30 | #include <cutils/sockets.h> |
Elliott Hughes | 6c34bba | 2015-04-17 20:11:08 -0700 | [diff] [blame] | 31 | |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 32 | #include "adb.h" |
Dan Albert | cc731cc | 2015-02-24 21:26:58 -0800 | [diff] [blame] | 33 | #include "adb_io.h" |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 34 | #include "adb_unique_fd.h" |
Elliott Hughes | 3d5f60d | 2015-07-18 12:21:30 -0700 | [diff] [blame] | 35 | #include "adb_utils.h" |
David Pursell | 70ef7b4 | 2015-09-30 13:35:42 -0700 | [diff] [blame] | 36 | #include "services.h" |
Josh Gao | cfb2141 | 2016-08-24 18:38:44 -0700 | [diff] [blame] | 37 | #include "socket_spec.h" |
Josh Gao | 0985547 | 2016-02-19 10:42:40 -0800 | [diff] [blame] | 38 | #include "sysdeps.h" |
Dan Albert | 7664901 | 2015-02-24 15:51:19 -0800 | [diff] [blame] | 39 | #include "transport.h" |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 40 | |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 41 | namespace { |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 42 | |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 43 | void service_bootstrap_func(std::string service_name, std::function<void(unique_fd)> func, |
| 44 | unique_fd fd) { |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 45 | adb_thread_setname(android::base::StringPrintf("%s svc %d", service_name.c_str(), fd.get())); |
| 46 | func(std::move(fd)); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 47 | } |
| 48 | |
Josh Gao | 997cfac | 2018-07-25 18:15:52 -0700 | [diff] [blame] | 49 | } // namespace |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 50 | |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 51 | unique_fd create_service_thread(const char* service_name, std::function<void(unique_fd)> func) { |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 52 | int s[2]; |
Dan Albert | bac3474 | 2015-02-25 17:51:28 -0800 | [diff] [blame] | 53 | if (adb_socketpair(s)) { |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 54 | printf("cannot create service socket pair\n"); |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 55 | return unique_fd(); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 56 | } |
Yabin Cui | 7a3f8d6 | 2015-09-02 17:44:28 -0700 | [diff] [blame] | 57 | D("socketpair: (%d,%d)", s[0], s[1]); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 58 | |
Jerry Zhang | 2f8c60b | 2017-02-10 17:45:27 -0800 | [diff] [blame] | 59 | #if !ADB_HOST |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 60 | if (strcmp(service_name, "sync") == 0) { |
Jerry Zhang | 2f8c60b | 2017-02-10 17:45:27 -0800 | [diff] [blame] | 61 | // Set file sync service socket to maximum size |
| 62 | int max_buf = LINUX_MAX_SOCKET_SIZE; |
| 63 | adb_setsockopt(s[0], SOL_SOCKET, SO_SNDBUF, &max_buf, sizeof(max_buf)); |
| 64 | adb_setsockopt(s[1], SOL_SOCKET, SO_SNDBUF, &max_buf, sizeof(max_buf)); |
| 65 | } |
| 66 | #endif // !ADB_HOST |
| 67 | |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 68 | std::thread(service_bootstrap_func, service_name, func, unique_fd(s[1])).detach(); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 69 | |
Yabin Cui | 7a3f8d6 | 2015-09-02 17:44:28 -0700 | [diff] [blame] | 70 | D("service thread started, %d:%d",s[0], s[1]); |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 71 | return unique_fd(s[0]); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 72 | } |
| 73 | |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 74 | unique_fd service_to_fd(std::string_view name, atransport* transport) { |
Cody Schuffelen | af0e220 | 2019-01-02 14:17:29 -0800 | [diff] [blame] | 75 | unique_fd ret; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 76 | |
Josh Gao | cfb2141 | 2016-08-24 18:38:44 -0700 | [diff] [blame] | 77 | if (is_socket_spec(name)) { |
| 78 | std::string error; |
Cody Schuffelen | af0e220 | 2019-01-02 14:17:29 -0800 | [diff] [blame] | 79 | if (!socket_spec_connect(&ret, name, nullptr, nullptr, &error)) { |
Josh Gao | cfb2141 | 2016-08-24 18:38:44 -0700 | [diff] [blame] | 80 | LOG(ERROR) << "failed to connect to socket '" << name << "': " << error; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 81 | } |
Josh Gao | 997cfac | 2018-07-25 18:15:52 -0700 | [diff] [blame] | 82 | } else { |
Benoit Goby | 9470c2f | 2013-02-20 15:04:53 -0800 | [diff] [blame] | 83 | #if !ADB_HOST |
Cody Schuffelen | af0e220 | 2019-01-02 14:17:29 -0800 | [diff] [blame] | 84 | ret = daemon_service_to_fd(name, transport); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 85 | #endif |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 86 | } |
Josh Gao | 997cfac | 2018-07-25 18:15:52 -0700 | [diff] [blame] | 87 | |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 88 | if (ret >= 0) { |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 89 | close_on_exec(ret.get()); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 90 | } |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 91 | return ret; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 92 | } |
| 93 | |
| 94 | #if ADB_HOST |
| 95 | struct state_info { |
Elliott Hughes | 3bd73c1 | 2015-05-05 13:10:43 -0700 | [diff] [blame] | 96 | TransportType transport_type; |
Leo Sartre | 1fbc9db | 2015-11-27 18:56:48 +0100 | [diff] [blame] | 97 | std::string serial; |
Josh Gao | b122b17 | 2017-08-16 16:57:01 -0700 | [diff] [blame] | 98 | TransportId transport_id; |
Dan Albert | dcd78a1 | 2015-05-18 16:43:57 -0700 | [diff] [blame] | 99 | ConnectionState state; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 100 | }; |
| 101 | |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 102 | static void wait_for_state(int fd, state_info* sinfo) { |
Yabin Cui | 7a3f8d6 | 2015-09-02 17:44:28 -0700 | [diff] [blame] | 103 | D("wait_for_state %d", sinfo->state); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 104 | |
Elliott Hughes | 8d28e19 | 2015-10-07 14:55:10 -0700 | [diff] [blame] | 105 | while (true) { |
| 106 | bool is_ambiguous = false; |
| 107 | std::string error = "unknown error"; |
Yi Kong | aed415c | 2018-07-13 18:15:16 -0700 | [diff] [blame] | 108 | const char* serial = sinfo->serial.length() ? sinfo->serial.c_str() : nullptr; |
Josh Gao | b122b17 | 2017-08-16 16:57:01 -0700 | [diff] [blame] | 109 | atransport* t = acquire_one_transport(sinfo->transport_type, serial, sinfo->transport_id, |
| 110 | &is_ambiguous, &error); |
Yabin Cui | b5e1141 | 2017-03-10 16:01:01 -0800 | [diff] [blame] | 111 | if (t != nullptr && (sinfo->state == kCsAny || sinfo->state == t->GetConnectionState())) { |
Elliott Hughes | 8d28e19 | 2015-10-07 14:55:10 -0700 | [diff] [blame] | 112 | SendOkay(fd); |
| 113 | break; |
| 114 | } else if (!is_ambiguous) { |
Josh Gao | 0985547 | 2016-02-19 10:42:40 -0800 | [diff] [blame] | 115 | adb_pollfd pfd = {.fd = fd, .events = POLLIN }; |
| 116 | int rc = adb_poll(&pfd, 1, 1000); |
| 117 | if (rc < 0) { |
| 118 | SendFail(fd, error); |
| 119 | break; |
| 120 | } else if (rc > 0 && (pfd.revents & POLLHUP) != 0) { |
| 121 | // The other end of the socket is closed, probably because the other side was |
| 122 | // terminated, bail out. |
| 123 | break; |
| 124 | } |
| 125 | |
Elliott Hughes | 8d28e19 | 2015-10-07 14:55:10 -0700 | [diff] [blame] | 126 | // Try again... |
| 127 | } else { |
| 128 | SendFail(fd, error); |
| 129 | break; |
| 130 | } |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 131 | } |
| 132 | |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 133 | adb_close(fd); |
Yabin Cui | 7a3f8d6 | 2015-09-02 17:44:28 -0700 | [diff] [blame] | 134 | D("wait_for_state is done"); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 135 | } |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 136 | |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 137 | void connect_emulator(const std::string& port_spec, std::string* response) { |
| 138 | std::vector<std::string> pieces = android::base::Split(port_spec, ","); |
| 139 | if (pieces.size() != 2) { |
| 140 | *response = android::base::StringPrintf("unable to parse '%s' as <console port>,<adb port>", |
| 141 | port_spec.c_str()); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 142 | return; |
| 143 | } |
| 144 | |
Yi Kong | aed415c | 2018-07-13 18:15:16 -0700 | [diff] [blame] | 145 | int console_port = strtol(pieces[0].c_str(), nullptr, 0); |
| 146 | int adb_port = strtol(pieces[1].c_str(), nullptr, 0); |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 147 | if (console_port <= 0 || adb_port <= 0) { |
| 148 | *response = android::base::StringPrintf("Invalid port numbers: %s", port_spec.c_str()); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 149 | return; |
| 150 | } |
| 151 | |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 152 | // Check if the emulator is already known. |
| 153 | // Note: There's a small but harmless race condition here: An emulator not |
| 154 | // present just yet could be registered by another invocation right |
| 155 | // after doing this check here. However, local_connect protects |
| 156 | // against double-registration too. From here, a better error message |
| 157 | // can be produced. In the case of the race condition, the very specific |
| 158 | // error message won't be shown, but the data doesn't get corrupted. |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 159 | atransport* known_emulator = find_emulator_transport_by_adb_port(adb_port); |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 160 | if (known_emulator != nullptr) { |
| 161 | *response = android::base::StringPrintf("Emulator already registered on port %d", adb_port); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 162 | return; |
| 163 | } |
| 164 | |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 165 | // Preconditions met, try to connect to the emulator. |
Elliott Hughes | 381cfa9 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 166 | std::string error; |
| 167 | if (!local_connect_arbitrary_ports(console_port, adb_port, &error)) { |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 168 | *response = android::base::StringPrintf("Connected to emulator on ports %d,%d", |
| 169 | console_port, adb_port); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 170 | } else { |
Elliott Hughes | 381cfa9 | 2015-07-23 17:12:58 -0700 | [diff] [blame] | 171 | *response = android::base::StringPrintf("Could not connect to emulator on ports %d,%d: %s", |
| 172 | console_port, adb_port, error.c_str()); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 173 | } |
| 174 | } |
| 175 | |
Josh Gao | 73a5ee4 | 2018-07-25 16:51:59 -0700 | [diff] [blame] | 176 | static void connect_service(unique_fd fd, std::string host) { |
Elliott Hughes | e67f1f8 | 2015-04-30 17:32:03 -0700 | [diff] [blame] | 177 | std::string response; |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 178 | if (!strncmp(host.c_str(), "emu:", 4)) { |
| 179 | connect_emulator(host.c_str() + 4, &response); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 180 | } else { |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 181 | connect_device(host.c_str(), &response); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 182 | } |
| 183 | |
| 184 | // Send response for emulator and device |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 185 | SendProtocolString(fd.get(), response); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 186 | } |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 187 | #endif |
| 188 | |
| 189 | #if ADB_HOST |
Josh Gao | b122b17 | 2017-08-16 16:57:01 -0700 | [diff] [blame] | 190 | asocket* host_service_to_socket(const char* name, const char* serial, TransportId transport_id) { |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 191 | if (!strcmp(name,"track-devices")) { |
Josh Gao | b0c1802 | 2017-08-14 18:57:54 -0700 | [diff] [blame] | 192 | return create_device_tracker(false); |
| 193 | } else if (!strcmp(name, "track-devices-l")) { |
| 194 | return create_device_tracker(true); |
Leo Sartre | 1fbc9db | 2015-11-27 18:56:48 +0100 | [diff] [blame] | 195 | } else if (android::base::StartsWith(name, "wait-for-")) { |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 196 | name += strlen("wait-for-"); |
| 197 | |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 198 | std::shared_ptr<state_info> sinfo = std::make_shared<state_info>(); |
Leo Sartre | 1fbc9db | 2015-11-27 18:56:48 +0100 | [diff] [blame] | 199 | if (sinfo == nullptr) { |
| 200 | fprintf(stderr, "couldn't allocate state_info: %s", strerror(errno)); |
| 201 | return nullptr; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 202 | } |
| 203 | |
Leo Sartre | 1fbc9db | 2015-11-27 18:56:48 +0100 | [diff] [blame] | 204 | if (serial) sinfo->serial = serial; |
Josh Gao | b122b17 | 2017-08-16 16:57:01 -0700 | [diff] [blame] | 205 | sinfo->transport_id = transport_id; |
Leo Sartre | 1fbc9db | 2015-11-27 18:56:48 +0100 | [diff] [blame] | 206 | |
| 207 | if (android::base::StartsWith(name, "local")) { |
| 208 | name += strlen("local"); |
| 209 | sinfo->transport_type = kTransportLocal; |
| 210 | } else if (android::base::StartsWith(name, "usb")) { |
| 211 | name += strlen("usb"); |
| 212 | sinfo->transport_type = kTransportUsb; |
| 213 | } else if (android::base::StartsWith(name, "any")) { |
| 214 | name += strlen("any"); |
| 215 | sinfo->transport_type = kTransportAny; |
| 216 | } else { |
| 217 | return nullptr; |
| 218 | } |
| 219 | |
| 220 | if (!strcmp(name, "-device")) { |
| 221 | sinfo->state = kCsDevice; |
| 222 | } else if (!strcmp(name, "-recovery")) { |
| 223 | sinfo->state = kCsRecovery; |
| 224 | } else if (!strcmp(name, "-sideload")) { |
| 225 | sinfo->state = kCsSideload; |
| 226 | } else if (!strcmp(name, "-bootloader")) { |
| 227 | sinfo->state = kCsBootloader; |
Josh Gao | 86441c3 | 2016-04-13 12:18:58 -0700 | [diff] [blame] | 228 | } else if (!strcmp(name, "-any")) { |
| 229 | sinfo->state = kCsAny; |
Leo Sartre | 1fbc9db | 2015-11-27 18:56:48 +0100 | [diff] [blame] | 230 | } else { |
| 231 | return nullptr; |
| 232 | } |
| 233 | |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 234 | unique_fd fd = create_service_thread("wait", [sinfo](int fd) { |
| 235 | wait_for_state(fd, sinfo.get()); |
| 236 | }); |
| 237 | return create_local_socket(std::move(fd)); |
Benoit Goby | 1c45ee9 | 2013-03-29 18:22:36 -0700 | [diff] [blame] | 238 | } else if (!strncmp(name, "connect:", 8)) { |
Luis Hector Chavez | 095792c | 2018-07-18 19:40:12 -0700 | [diff] [blame] | 239 | std::string host(name + strlen("connect:")); |
Josh Gao | 74ccdf9 | 2019-01-23 15:36:42 -0800 | [diff] [blame^] | 240 | unique_fd fd = create_service_thread( |
| 241 | "connect", std::bind(connect_service, std::placeholders::_1, host)); |
| 242 | return create_local_socket(std::move(fd)); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 243 | } |
Yi Kong | aed415c | 2018-07-13 18:15:16 -0700 | [diff] [blame] | 244 | return nullptr; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 245 | } |
| 246 | #endif /* ADB_HOST */ |