| /* | 
 |  * Copyright (C) 2007 The Android Open Source Project | 
 |  * | 
 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 |  * you may not use this file except in compliance with the License. | 
 |  * You may obtain a copy of the License at | 
 |  * | 
 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 |  * | 
 |  * Unless required by applicable law or agreed to in writing, software | 
 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 |  * See the License for the specific language governing permissions and | 
 |  * limitations under the License. | 
 |  */ | 
 |  | 
 | #define TRACE_TAG SERVICES | 
 |  | 
 | #include "sysdeps.h" | 
 |  | 
 | #include <errno.h> | 
 | #include <stddef.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 |  | 
 | #include <thread> | 
 |  | 
 | #include <android-base/stringprintf.h> | 
 | #include <android-base/strings.h> | 
 | #include <cutils/sockets.h> | 
 |  | 
 | #include "adb.h" | 
 | #include "adb_io.h" | 
 | #include "adb_unique_fd.h" | 
 | #include "adb_utils.h" | 
 | #include "services.h" | 
 | #include "socket_spec.h" | 
 | #include "sysdeps.h" | 
 | #include "transport.h" | 
 |  | 
 | namespace { | 
 |  | 
 | void service_bootstrap_func(std::string service_name, std::function<void(unique_fd)> func, | 
 |                             unique_fd fd) { | 
 |     adb_thread_setname(android::base::StringPrintf("%s svc %d", service_name.c_str(), fd.get())); | 
 |     func(std::move(fd)); | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | unique_fd create_service_thread(const char* service_name, std::function<void(unique_fd)> func) { | 
 |     int s[2]; | 
 |     if (adb_socketpair(s)) { | 
 |         printf("cannot create service socket pair\n"); | 
 |         return unique_fd(); | 
 |     } | 
 |     D("socketpair: (%d,%d)", s[0], s[1]); | 
 |  | 
 | #if !ADB_HOST | 
 |     if (strcmp(service_name, "sync") == 0) { | 
 |         // Set file sync service socket to maximum size | 
 |         int max_buf = LINUX_MAX_SOCKET_SIZE; | 
 |         adb_setsockopt(s[0], SOL_SOCKET, SO_SNDBUF, &max_buf, sizeof(max_buf)); | 
 |         adb_setsockopt(s[1], SOL_SOCKET, SO_SNDBUF, &max_buf, sizeof(max_buf)); | 
 |     } | 
 | #endif  // !ADB_HOST | 
 |  | 
 |     std::thread(service_bootstrap_func, service_name, func, unique_fd(s[1])).detach(); | 
 |  | 
 |     D("service thread started, %d:%d", s[0], s[1]); | 
 |     return unique_fd(s[0]); | 
 | } | 
 |  | 
 | unique_fd service_to_fd(std::string_view name, atransport* transport) { | 
 |     unique_fd ret; | 
 |  | 
 |     if (is_socket_spec(name)) { | 
 |         std::string error; | 
 |         if (!socket_spec_connect(&ret, name, nullptr, nullptr, &error)) { | 
 |             LOG(ERROR) << "failed to connect to socket '" << name << "': " << error; | 
 |         } | 
 |     } else { | 
 | #if !ADB_HOST | 
 |         ret = daemon_service_to_fd(name, transport); | 
 | #endif | 
 |     } | 
 |  | 
 |     if (ret >= 0) { | 
 |         close_on_exec(ret.get()); | 
 |     } | 
 |     return ret; | 
 | } | 
 |  | 
 | #if ADB_HOST | 
 | struct state_info { | 
 |     TransportType transport_type; | 
 |     std::string serial; | 
 |     TransportId transport_id; | 
 |     ConnectionState state; | 
 | }; | 
 |  | 
 | static void wait_for_state(unique_fd fd, state_info* sinfo) { | 
 |     D("wait_for_state %d", sinfo->state); | 
 |  | 
 |     while (true) { | 
 |         bool is_ambiguous = false; | 
 |         std::string error = "unknown error"; | 
 |         const char* serial = sinfo->serial.length() ? sinfo->serial.c_str() : nullptr; | 
 |         atransport* t = acquire_one_transport(sinfo->transport_type, serial, sinfo->transport_id, | 
 |                                               &is_ambiguous, &error); | 
 |         if (sinfo->state == kCsOffline) { | 
 |             // wait-for-disconnect uses kCsOffline, we don't actually want to wait for 'offline'. | 
 |             if (t == nullptr) { | 
 |                 SendOkay(fd); | 
 |                 break; | 
 |             } | 
 |         } else if (t != nullptr && | 
 |                    (sinfo->state == kCsAny || sinfo->state == t->GetConnectionState())) { | 
 |             SendOkay(fd); | 
 |             break; | 
 |         } | 
 |  | 
 |         if (!is_ambiguous) { | 
 |             adb_pollfd pfd = {.fd = fd.get(), .events = POLLIN}; | 
 |             int rc = adb_poll(&pfd, 1, 100); | 
 |             if (rc < 0) { | 
 |                 SendFail(fd, error); | 
 |                 break; | 
 |             } else if (rc > 0 && (pfd.revents & POLLHUP) != 0) { | 
 |                 // The other end of the socket is closed, probably because the other side was | 
 |                 // terminated, bail out. | 
 |                 break; | 
 |             } | 
 |  | 
 |             // Try again... | 
 |         } else { | 
 |             SendFail(fd, error); | 
 |             break; | 
 |         } | 
 |     } | 
 |  | 
 |     D("wait_for_state is done"); | 
 | } | 
 |  | 
 | void connect_emulator(const std::string& port_spec, std::string* response) { | 
 |     std::vector<std::string> pieces = android::base::Split(port_spec, ","); | 
 |     if (pieces.size() != 2) { | 
 |         *response = android::base::StringPrintf("unable to parse '%s' as <console port>,<adb port>", | 
 |                                                 port_spec.c_str()); | 
 |         return; | 
 |     } | 
 |  | 
 |     int console_port = strtol(pieces[0].c_str(), nullptr, 0); | 
 |     int adb_port = strtol(pieces[1].c_str(), nullptr, 0); | 
 |     if (console_port <= 0 || adb_port <= 0) { | 
 |         *response = android::base::StringPrintf("Invalid port numbers: %s", port_spec.c_str()); | 
 |         return; | 
 |     } | 
 |  | 
 |     // Check if the emulator is already known. | 
 |     // Note: There's a small but harmless race condition here: An emulator not | 
 |     // present just yet could be registered by another invocation right | 
 |     // after doing this check here. However, local_connect protects | 
 |     // against double-registration too. From here, a better error message | 
 |     // can be produced. In the case of the race condition, the very specific | 
 |     // error message won't be shown, but the data doesn't get corrupted. | 
 |     atransport* known_emulator = find_emulator_transport_by_adb_port(adb_port); | 
 |     if (known_emulator != nullptr) { | 
 |         *response = android::base::StringPrintf("Emulator already registered on port %d", adb_port); | 
 |         return; | 
 |     } | 
 |  | 
 |     // Preconditions met, try to connect to the emulator. | 
 |     std::string error; | 
 |     if (!local_connect_arbitrary_ports(console_port, adb_port, &error)) { | 
 |         *response = android::base::StringPrintf("Connected to emulator on ports %d,%d", | 
 |                                                 console_port, adb_port); | 
 |     } else { | 
 |         *response = android::base::StringPrintf("Could not connect to emulator on ports %d,%d: %s", | 
 |                                                 console_port, adb_port, error.c_str()); | 
 |     } | 
 | } | 
 |  | 
 | static void connect_service(unique_fd fd, std::string host) { | 
 |     std::string response; | 
 |     if (!strncmp(host.c_str(), "emu:", 4)) { | 
 |         connect_emulator(host.c_str() + 4, &response); | 
 |     } else { | 
 |         connect_device(host, &response); | 
 |     } | 
 |  | 
 |     // Send response for emulator and device | 
 |     SendProtocolString(fd.get(), response); | 
 | } | 
 | #endif | 
 |  | 
 | #if ADB_HOST | 
 | asocket* host_service_to_socket(std::string_view name, std::string_view serial, | 
 |                                 TransportId transport_id) { | 
 |     if (name == "track-devices") { | 
 |         return create_device_tracker(false); | 
 |     } else if (name == "track-devices-l") { | 
 |         return create_device_tracker(true); | 
 |     } else if (android::base::ConsumePrefix(&name, "wait-for-")) { | 
 |         std::shared_ptr<state_info> sinfo = std::make_shared<state_info>(); | 
 |         if (sinfo == nullptr) { | 
 |             fprintf(stderr, "couldn't allocate state_info: %s", strerror(errno)); | 
 |             return nullptr; | 
 |         } | 
 |  | 
 |         sinfo->serial = serial; | 
 |         sinfo->transport_id = transport_id; | 
 |  | 
 |         if (android::base::ConsumePrefix(&name, "local")) { | 
 |             sinfo->transport_type = kTransportLocal; | 
 |         } else if (android::base::ConsumePrefix(&name, "usb")) { | 
 |             sinfo->transport_type = kTransportUsb; | 
 |         } else if (android::base::ConsumePrefix(&name, "any")) { | 
 |             sinfo->transport_type = kTransportAny; | 
 |         } else { | 
 |             return nullptr; | 
 |         } | 
 |  | 
 |         if (name == "-device") { | 
 |             sinfo->state = kCsDevice; | 
 |         } else if (name == "-recovery") { | 
 |             sinfo->state = kCsRecovery; | 
 |         } else if (name == "-rescue") { | 
 |             sinfo->state = kCsRescue; | 
 |         } else if (name == "-sideload") { | 
 |             sinfo->state = kCsSideload; | 
 |         } else if (name == "-bootloader") { | 
 |             sinfo->state = kCsBootloader; | 
 |         } else if (name == "-any") { | 
 |             sinfo->state = kCsAny; | 
 |         } else if (name == "-disconnect") { | 
 |             sinfo->state = kCsOffline; | 
 |         } else { | 
 |             return nullptr; | 
 |         } | 
 |  | 
 |         unique_fd fd = create_service_thread( | 
 |                 "wait", [sinfo](unique_fd fd) { wait_for_state(std::move(fd), sinfo.get()); }); | 
 |         return create_local_socket(std::move(fd)); | 
 |     } else if (android::base::ConsumePrefix(&name, "connect:")) { | 
 |         std::string host(name); | 
 |         unique_fd fd = create_service_thread( | 
 |                 "connect", std::bind(connect_service, std::placeholders::_1, host)); | 
 |         return create_local_socket(std::move(fd)); | 
 |     } | 
 |     return nullptr; | 
 | } | 
 | #endif /* ADB_HOST */ |