1 /* 2 * Copyright (C) 2016 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 17 #ifndef LIBMEMUNREACHABLE_LEAK_PIPE_H_ 18 #define LIBMEMUNREACHABLE_LEAK_PIPE_H_ 19 20 #include <sys/socket.h> 21 22 #include <vector> 23 24 #include "android-base/macros.h" 25 26 #include "ScopedPipe.h" 27 #include "log.h" 28 29 namespace android { 30 31 // LeakPipe implements a pipe that can transfer vectors of simple objects 32 // between processes. The pipe is created in the sending process and 33 // transferred over a socketpair that was created before forking. This ensures 34 // that only the sending process can have the send side of the pipe open, so if 35 // the sending process dies the pipe will close. 36 class LeakPipe { 37 public: 38 LeakPipe() { 39 int ret = socketpair(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0, sv_); 40 if (ret < 0) { 41 MEM_LOG_ALWAYS_FATAL("failed to create socketpair: %s", strerror(errno)); 42 } 43 } 44 45 ~LeakPipe() { Close(); } 46 47 void Close() { 48 close(sv_[0]); 49 close(sv_[1]); 50 sv_[0] = -1; 51 sv_[1] = -1; 52 } 53 54 bool OpenReceiver() { 55 int fd = ReceiveFd(sv_[0]); 56 if (fd < 0) { 57 return false; 58 } 59 60 receiver_.SetFd(fd); 61 return true; 62 } 63 64 bool OpenSender() { 65 ScopedPipe pipe; 66 67 if (!SendFd(sv_[1], pipe.Receiver())) { 68 return false; 69 } 70 pipe.ReleaseReceiver(); 71 72 sender_.SetFd(pipe.ReleaseSender()); 73 return true; 74 } 75 76 class LeakPipeBase { 77 public: 78 LeakPipeBase() : fd_(-1) {} 79 80 ~LeakPipeBase() { Close(); } 81 82 void SetFd(int fd) { fd_ = fd; } 83 84 void Close() { 85 close(fd_); 86 fd_ = -1; 87 } 88 89 protected: 90 int fd_; 91 92 private: 93 DISALLOW_COPY_AND_ASSIGN(LeakPipeBase); 94 }; 95 96 class LeakPipeSender : public LeakPipeBase { 97 public: 98 using LeakPipeBase::LeakPipeBase; 99 100 template <typename T> 101 bool Send(const T& value) { 102 ssize_t ret = TEMP_FAILURE_RETRY(write(fd_, &value, sizeof(T))); 103 if (ret < 0) { 104 MEM_ALOGE("failed to send value: %s", strerror(errno)); 105 return false; 106 } else if (static_cast<size_t>(ret) != sizeof(T)) { 107 MEM_ALOGE("eof while writing value"); 108 return false; 109 } 110 111 return true; 112 } 113 114 template <class T, class Alloc = std::allocator<T>> 115 bool SendVector(const std::vector<T, Alloc>& vector) { 116 size_t size = vector.size() * sizeof(T); 117 if (!Send(size)) { 118 return false; 119 } 120 121 ssize_t ret = TEMP_FAILURE_RETRY(write(fd_, vector.data(), size)); 122 if (ret < 0) { 123 MEM_ALOGE("failed to send vector: %s", strerror(errno)); 124 return false; 125 } else if (static_cast<size_t>(ret) != size) { 126 MEM_ALOGE("eof while writing vector"); 127 return false; 128 } 129 130 return true; 131 } 132 }; 133 134 class LeakPipeReceiver : public LeakPipeBase { 135 public: 136 using LeakPipeBase::LeakPipeBase; 137 138 template <typename T> 139 bool Receive(T* value) { 140 ssize_t ret = TEMP_FAILURE_RETRY(read(fd_, reinterpret_cast<void*>(value), sizeof(T))); 141 if (ret < 0) { 142 MEM_ALOGE("failed to receive value: %s", strerror(errno)); 143 return false; 144 } else if (static_cast<size_t>(ret) != sizeof(T)) { 145 MEM_ALOGE("eof while receiving value"); 146 return false; 147 } 148 149 return true; 150 } 151 152 template <class T, class Alloc = std::allocator<T>> 153 bool ReceiveVector(std::vector<T, Alloc>& vector) { 154 size_t size = 0; 155 if (!Receive(&size)) { 156 return false; 157 } 158 159 vector.resize(size / sizeof(T)); 160 161 char* ptr = reinterpret_cast<char*>(vector.data()); 162 while (size > 0) { 163 ssize_t ret = TEMP_FAILURE_RETRY(read(fd_, ptr, size)); 164 if (ret < 0) { 165 MEM_ALOGE("failed to send vector: %s", strerror(errno)); 166 return false; 167 } else if (ret == 0) { 168 MEM_ALOGE("eof while reading vector"); 169 return false; 170 } 171 size -= ret; 172 ptr += ret; 173 } 174 175 return true; 176 } 177 }; 178 179 LeakPipeReceiver& Receiver() { return receiver_; } 180 181 LeakPipeSender& Sender() { return sender_; } 182 183 private: 184 LeakPipeReceiver receiver_; 185 LeakPipeSender sender_; 186 bool SendFd(int sock, int fd); 187 int ReceiveFd(int sock); 188 DISALLOW_COPY_AND_ASSIGN(LeakPipe); 189 int sv_[2]; 190 }; 191 192 } // namespace android 193 194 #endif // LIBMEMUNREACHABLE_LEAK_PIPE_H_ 195