|  | /* | 
|  | * 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())); | 
|  | } |