| /* | 
 |  * Copyright (C) 2016 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. | 
 |  */ | 
 |  | 
 | #include "sysdeps.h" | 
 |  | 
 | bool set_tcp_keepalive(borrowed_fd fd, int interval_sec) { | 
 |     int enable = (interval_sec > 0); | 
 |     if (adb_setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, &enable, sizeof(enable))) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     if (!enable) { | 
 |         return true; | 
 |     } | 
 |  | 
 |     // Idle time before sending the first keepalive is TCP_KEEPIDLE on Linux, TCP_KEEPALIVE on Mac. | 
 | #if defined(TCP_KEEPIDLE) | 
 |     if (adb_setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE, &interval_sec, sizeof(interval_sec))) { | 
 |         return false; | 
 |     } | 
 | #elif defined(TCP_KEEPALIVE) | 
 |     if (adb_setsockopt(fd, IPPROTO_TCP, TCP_KEEPALIVE, &interval_sec, sizeof(interval_sec))) { | 
 |         return false; | 
 |     } | 
 | #endif | 
 |  | 
 |     // TCP_KEEPINTVL and TCP_KEEPCNT are available on Linux 2.4+ and OS X 10.8+ (Mountain Lion). | 
 | #if defined(TCP_KEEPINTVL) | 
 |     if (adb_setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL, &interval_sec, sizeof(interval_sec))) { | 
 |         return false; | 
 |     } | 
 | #endif | 
 |  | 
 | #if defined(TCP_KEEPCNT) | 
 |     // On Windows this value is hardcoded to 10. This is a reasonable value, so we do the same here | 
 |     // to match behavior. See SO_KEEPALIVE documentation at | 
 |     // https://msdn.microsoft.com/en-us/library/windows/desktop/ee470551(v=vs.85).aspx. | 
 |     const int keepcnt = 10; | 
 |     if (adb_setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT, &keepcnt, sizeof(keepcnt))) { | 
 |         return false; | 
 |     } | 
 | #endif | 
 |  | 
 |     return true; | 
 | } | 
 |  | 
 | static __inline__ void disable_close_on_exec(borrowed_fd fd) { | 
 |     const auto oldFlags = fcntl(fd.get(), F_GETFD); | 
 |     const auto newFlags = (oldFlags & ~FD_CLOEXEC); | 
 |     if (newFlags != oldFlags) { | 
 |         fcntl(fd.get(), F_SETFD, newFlags); | 
 |     } | 
 | } | 
 |  | 
 | Process adb_launch_process(std::string_view executable, std::vector<std::string> args, | 
 |                            std::initializer_list<int> fds_to_inherit) { | 
 |     const auto pid = fork(); | 
 |     if (pid != 0) { | 
 |         // parent, includes the case when failed to fork() | 
 |         return Process(pid); | 
 |     } | 
 |     // child | 
 |     std::vector<std::string> copies; | 
 |     copies.reserve(args.size() + 1); | 
 |     copies.emplace_back(executable); | 
 |     copies.insert(copies.end(), std::make_move_iterator(args.begin()), | 
 |                   std::make_move_iterator(args.end())); | 
 |  | 
 |     std::vector<char*> rawArgs; | 
 |     rawArgs.reserve(copies.size() + 1); | 
 |     for (auto&& str : copies) { | 
 |         rawArgs.push_back(str.data()); | 
 |     } | 
 |     rawArgs.push_back(nullptr); | 
 |     for (auto fd : fds_to_inherit) { | 
 |         disable_close_on_exec(fd); | 
 |     } | 
 |     exit(execv(copies.front().data(), rawArgs.data())); | 
 | } |