93 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			93 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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()));
 | |
| }
 |