| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 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 | #include "ThreadCapture.h" | 
|  | 18 |  | 
|  | 19 | #include <elf.h> | 
|  | 20 | #include <errno.h> | 
|  | 21 | #include <fcntl.h> | 
|  | 22 | #include <limits.h> | 
|  | 23 | #include <stdlib.h> | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 24 | #include <sys/ptrace.h> | 
|  | 25 | #include <sys/stat.h> | 
|  | 26 | #include <sys/syscall.h> | 
|  | 27 | #include <sys/types.h> | 
|  | 28 | #include <sys/uio.h> | 
|  | 29 | #include <sys/wait.h> | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 30 | #include <unistd.h> | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 31 |  | 
|  | 32 | #include <map> | 
|  | 33 | #include <memory> | 
|  | 34 | #include <set> | 
|  | 35 | #include <vector> | 
|  | 36 |  | 
|  | 37 | #include <android-base/unique_fd.h> | 
|  | 38 |  | 
|  | 39 | #include "Allocator.h" | 
|  | 40 | #include "log.h" | 
|  | 41 |  | 
| Colin Cross | a9939e9 | 2017-06-21 13:13:00 -0700 | [diff] [blame] | 42 | namespace android { | 
|  | 43 |  | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 44 | // bionic interfaces used: | 
|  | 45 | // atoi | 
|  | 46 | // strlcat | 
|  | 47 | // writev | 
|  | 48 |  | 
|  | 49 | // bionic interfaces reimplemented to avoid allocation: | 
|  | 50 | // getdents64 | 
|  | 51 |  | 
|  | 52 | // Convert a pid > 0 to a string.  sprintf might allocate, so we can't use it. | 
|  | 53 | // Returns a pointer somewhere in buf to a null terminated string, or NULL | 
|  | 54 | // on error. | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 55 | static char* pid_to_str(char* buf, size_t len, pid_t pid) { | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 56 | if (pid <= 0) { | 
|  | 57 | return nullptr; | 
|  | 58 | } | 
|  | 59 |  | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 60 | char* ptr = buf + len - 1; | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 61 | *ptr = 0; | 
|  | 62 | while (pid > 0) { | 
|  | 63 | ptr--; | 
|  | 64 | if (ptr < buf) { | 
|  | 65 | return nullptr; | 
|  | 66 | } | 
|  | 67 | *ptr = '0' + (pid % 10); | 
|  | 68 | pid /= 10; | 
|  | 69 | } | 
|  | 70 |  | 
|  | 71 | return ptr; | 
|  | 72 | } | 
|  | 73 |  | 
|  | 74 | class ThreadCaptureImpl { | 
|  | 75 | public: | 
|  | 76 | ThreadCaptureImpl(pid_t pid, Allocator<ThreadCaptureImpl>& allocator); | 
|  | 77 | ~ThreadCaptureImpl() {} | 
|  | 78 | bool ListThreads(TidList& tids); | 
|  | 79 | bool CaptureThreads(); | 
|  | 80 | bool ReleaseThreads(); | 
|  | 81 | bool ReleaseThread(pid_t tid); | 
|  | 82 | bool CapturedThreadInfo(ThreadInfoList& threads); | 
|  | 83 | void InjectTestFunc(std::function<void(pid_t)>&& f) { inject_test_func_ = f; } | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 84 |  | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 85 | private: | 
|  | 86 | int CaptureThread(pid_t tid); | 
|  | 87 | bool ReleaseThread(pid_t tid, unsigned int signal); | 
|  | 88 | int PtraceAttach(pid_t tid); | 
|  | 89 | void PtraceDetach(pid_t tid, unsigned int signal); | 
|  | 90 | bool PtraceThreadInfo(pid_t tid, ThreadInfo& thread_info); | 
|  | 91 |  | 
| Colin Cross | e4cbe0e | 2016-03-04 16:34:42 -0800 | [diff] [blame] | 92 | allocator::map<pid_t, unsigned int> captured_threads_; | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 93 | Allocator<ThreadCaptureImpl> allocator_; | 
|  | 94 | pid_t pid_; | 
|  | 95 | std::function<void(pid_t)> inject_test_func_; | 
|  | 96 | }; | 
|  | 97 |  | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 98 | ThreadCaptureImpl::ThreadCaptureImpl(pid_t pid, Allocator<ThreadCaptureImpl>& allocator) | 
|  | 99 | : captured_threads_(allocator), allocator_(allocator), pid_(pid) {} | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 100 |  | 
|  | 101 | bool ThreadCaptureImpl::ListThreads(TidList& tids) { | 
|  | 102 | tids.clear(); | 
|  | 103 |  | 
|  | 104 | char pid_buf[11]; | 
|  | 105 | char path[256] = "/proc/"; | 
|  | 106 | char* pid_str = pid_to_str(pid_buf, sizeof(pid_buf), pid_); | 
|  | 107 | if (!pid_str) { | 
|  | 108 | return false; | 
|  | 109 | } | 
|  | 110 | strlcat(path, pid_str, sizeof(path)); | 
|  | 111 | strlcat(path, "/task", sizeof(path)); | 
|  | 112 |  | 
| Elliott Hughes | 2c5d1d7 | 2016-03-28 12:15:36 -0700 | [diff] [blame] | 113 | android::base::unique_fd fd(open(path, O_CLOEXEC | O_DIRECTORY | O_RDONLY)); | 
|  | 114 | if (fd == -1) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 115 | MEM_ALOGE("failed to open %s: %s", path, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 116 | return false; | 
|  | 117 | } | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 118 |  | 
|  | 119 | struct linux_dirent64 { | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 120 | uint64_t d_ino; | 
|  | 121 | int64_t d_off; | 
|  | 122 | uint16_t d_reclen; | 
|  | 123 | char d_type; | 
|  | 124 | char d_name[]; | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 125 | } __attribute((packed)); | 
|  | 126 | char dirent_buf[4096]; | 
|  | 127 | ssize_t nread; | 
|  | 128 | do { | 
| Elliott Hughes | 2c5d1d7 | 2016-03-28 12:15:36 -0700 | [diff] [blame] | 129 | nread = syscall(SYS_getdents64, fd.get(), dirent_buf, sizeof(dirent_buf)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 130 | if (nread < 0) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 131 | MEM_ALOGE("failed to get directory entries from %s: %s", path, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 132 | return false; | 
|  | 133 | } else if (nread > 0) { | 
|  | 134 | ssize_t off = 0; | 
|  | 135 | while (off < nread) { | 
|  | 136 | linux_dirent64* dirent = reinterpret_cast<linux_dirent64*>(dirent_buf + off); | 
|  | 137 | off += dirent->d_reclen; | 
|  | 138 | pid_t tid = atoi(dirent->d_name); | 
|  | 139 | if (tid <= 0) { | 
|  | 140 | continue; | 
|  | 141 | } | 
|  | 142 | tids.push_back(tid); | 
|  | 143 | } | 
|  | 144 | } | 
|  | 145 |  | 
|  | 146 | } while (nread != 0); | 
|  | 147 |  | 
|  | 148 | return true; | 
|  | 149 | } | 
|  | 150 |  | 
|  | 151 | bool ThreadCaptureImpl::CaptureThreads() { | 
|  | 152 | TidList tids{allocator_}; | 
|  | 153 |  | 
|  | 154 | bool found_new_thread; | 
|  | 155 | do { | 
|  | 156 | if (!ListThreads(tids)) { | 
|  | 157 | ReleaseThreads(); | 
|  | 158 | return false; | 
|  | 159 | } | 
|  | 160 |  | 
|  | 161 | found_new_thread = false; | 
|  | 162 |  | 
|  | 163 | for (auto it = tids.begin(); it != tids.end(); it++) { | 
|  | 164 | auto captured = captured_threads_.find(*it); | 
|  | 165 | if (captured == captured_threads_.end()) { | 
|  | 166 | if (CaptureThread(*it) < 0) { | 
|  | 167 | ReleaseThreads(); | 
|  | 168 | return false; | 
|  | 169 | } | 
|  | 170 | found_new_thread = true; | 
|  | 171 | } | 
|  | 172 | } | 
|  | 173 | } while (found_new_thread); | 
|  | 174 |  | 
|  | 175 | return true; | 
|  | 176 | } | 
|  | 177 |  | 
|  | 178 | // Detatches from a thread, delivering signal if nonzero, logs on error | 
|  | 179 | void ThreadCaptureImpl::PtraceDetach(pid_t tid, unsigned int signal) { | 
|  | 180 | void* sig_ptr = reinterpret_cast<void*>(static_cast<uintptr_t>(signal)); | 
|  | 181 | if (ptrace(PTRACE_DETACH, tid, NULL, sig_ptr) < 0 && errno != ESRCH) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 182 | MEM_ALOGE("failed to detach from thread %d of process %d: %s", tid, pid_, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 183 | } | 
|  | 184 | } | 
|  | 185 |  | 
|  | 186 | // Attaches to and pauses thread. | 
|  | 187 | // Returns 1 on attach, 0 on tid not found, -1 and logs on error | 
|  | 188 | int ThreadCaptureImpl::PtraceAttach(pid_t tid) { | 
|  | 189 | int ret = ptrace(PTRACE_SEIZE, tid, NULL, NULL); | 
|  | 190 | if (ret < 0) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 191 | MEM_ALOGE("failed to attach to thread %d of process %d: %s", tid, pid_, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 192 | return -1; | 
|  | 193 | } | 
|  | 194 |  | 
|  | 195 | if (inject_test_func_) { | 
|  | 196 | inject_test_func_(tid); | 
|  | 197 | } | 
|  | 198 |  | 
|  | 199 | if (ptrace(PTRACE_INTERRUPT, tid, 0, 0) < 0) { | 
|  | 200 | if (errno == ESRCH) { | 
|  | 201 | return 0; | 
|  | 202 | } else { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 203 | MEM_ALOGE("failed to interrupt thread %d of process %d: %s", tid, pid_, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 204 | PtraceDetach(tid, 0); | 
|  | 205 | return -1; | 
|  | 206 | } | 
|  | 207 | } | 
|  | 208 | return 1; | 
|  | 209 | } | 
|  | 210 |  | 
|  | 211 | bool ThreadCaptureImpl::PtraceThreadInfo(pid_t tid, ThreadInfo& thread_info) { | 
|  | 212 | thread_info.tid = tid; | 
|  | 213 |  | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 214 | const unsigned int max_num_regs = 128;  // larger than number of registers on any device | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 215 | uintptr_t regs[max_num_regs]; | 
|  | 216 | struct iovec iovec; | 
|  | 217 | iovec.iov_base = ®s; | 
|  | 218 | iovec.iov_len = sizeof(regs); | 
|  | 219 |  | 
|  | 220 | if (ptrace(PTRACE_GETREGSET, tid, reinterpret_cast<void*>(NT_PRSTATUS), &iovec)) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 221 | MEM_ALOGE("ptrace getregset for thread %d of process %d failed: %s", tid, pid_, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 222 | return false; | 
|  | 223 | } | 
|  | 224 |  | 
|  | 225 | unsigned int num_regs = iovec.iov_len / sizeof(uintptr_t); | 
|  | 226 | thread_info.regs.assign(®s[0], ®s[num_regs]); | 
|  | 227 |  | 
|  | 228 | const int sp = | 
|  | 229 | #if defined(__x86_64__) | 
|  | 230 | offsetof(struct pt_regs, rsp) / sizeof(uintptr_t) | 
|  | 231 | #elif defined(__i386__) | 
|  | 232 | offsetof(struct pt_regs, esp) / sizeof(uintptr_t) | 
|  | 233 | #elif defined(__arm__) | 
|  | 234 | offsetof(struct pt_regs, ARM_sp) / sizeof(uintptr_t) | 
|  | 235 | #elif defined(__aarch64__) | 
|  | 236 | offsetof(struct user_pt_regs, sp) / sizeof(uintptr_t) | 
|  | 237 | #elif defined(__mips__) || defined(__mips64__) | 
|  | 238 | offsetof(struct pt_regs, regs[29]) / sizeof(uintptr_t) | 
|  | 239 | #else | 
|  | 240 | #error Unrecognized architecture | 
|  | 241 | #endif | 
|  | 242 | ; | 
|  | 243 |  | 
|  | 244 | // TODO(ccross): use /proc/tid/status or /proc/pid/maps to get start_stack | 
|  | 245 |  | 
|  | 246 | thread_info.stack = std::pair<uintptr_t, uintptr_t>(regs[sp], 0); | 
|  | 247 |  | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 248 | return true; | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 249 | } | 
|  | 250 |  | 
|  | 251 | int ThreadCaptureImpl::CaptureThread(pid_t tid) { | 
|  | 252 | int ret = PtraceAttach(tid); | 
|  | 253 | if (ret <= 0) { | 
|  | 254 | return ret; | 
|  | 255 | } | 
|  | 256 |  | 
|  | 257 | int status = 0; | 
|  | 258 | if (TEMP_FAILURE_RETRY(waitpid(tid, &status, __WALL)) < 0) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 259 | MEM_ALOGE("failed to wait for pause of thread %d of process %d: %s", tid, pid_, strerror(errno)); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 260 | PtraceDetach(tid, 0); | 
|  | 261 | return -1; | 
|  | 262 | } | 
|  | 263 |  | 
|  | 264 | if (!WIFSTOPPED(status)) { | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 265 | MEM_ALOGE("thread %d of process %d was not paused after waitpid, killed?", tid, pid_); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 266 | return 0; | 
|  | 267 | } | 
|  | 268 |  | 
|  | 269 | unsigned int resume_signal = 0; | 
|  | 270 |  | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 271 | unsigned int signal = WSTOPSIG(status); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 272 | if ((status >> 16) == PTRACE_EVENT_STOP) { | 
|  | 273 | switch (signal) { | 
|  | 274 | case SIGSTOP: | 
|  | 275 | case SIGTSTP: | 
|  | 276 | case SIGTTIN: | 
|  | 277 | case SIGTTOU: | 
|  | 278 | // group-stop signals | 
|  | 279 | break; | 
|  | 280 | case SIGTRAP: | 
|  | 281 | // normal ptrace interrupt stop | 
|  | 282 | break; | 
|  | 283 | default: | 
| Christopher Ferris | 47dea71 | 2017-05-03 17:34:29 -0700 | [diff] [blame] | 284 | MEM_ALOGE("unexpected signal %d with PTRACE_EVENT_STOP for thread %d of process %d", signal, | 
|  | 285 | tid, pid_); | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 286 | return -1; | 
|  | 287 | } | 
|  | 288 | } else { | 
|  | 289 | // signal-delivery-stop | 
|  | 290 | resume_signal = signal; | 
|  | 291 | } | 
|  | 292 |  | 
|  | 293 | captured_threads_[tid] = resume_signal; | 
|  | 294 | return 1; | 
|  | 295 | } | 
|  | 296 |  | 
|  | 297 | bool ThreadCaptureImpl::ReleaseThread(pid_t tid) { | 
|  | 298 | auto it = captured_threads_.find(tid); | 
|  | 299 | if (it == captured_threads_.end()) { | 
|  | 300 | return false; | 
|  | 301 | } | 
|  | 302 | return ReleaseThread(it->first, it->second); | 
|  | 303 | } | 
|  | 304 |  | 
|  | 305 | bool ThreadCaptureImpl::ReleaseThread(pid_t tid, unsigned int signal) { | 
|  | 306 | PtraceDetach(tid, signal); | 
|  | 307 | return true; | 
|  | 308 | } | 
|  | 309 |  | 
|  | 310 | bool ThreadCaptureImpl::ReleaseThreads() { | 
|  | 311 | bool ret = true; | 
| Colin Cross | a83881e | 2017-06-22 10:50:05 -0700 | [diff] [blame] | 312 | for (auto it = captured_threads_.begin(); it != captured_threads_.end();) { | 
| Colin Cross | bcb4ed3 | 2016-01-14 15:35:40 -0800 | [diff] [blame] | 313 | if (ReleaseThread(it->first, it->second)) { | 
|  | 314 | it = captured_threads_.erase(it); | 
|  | 315 | } else { | 
|  | 316 | it++; | 
|  | 317 | ret = false; | 
|  | 318 | } | 
|  | 319 | } | 
|  | 320 | return ret; | 
|  | 321 | } | 
|  | 322 |  | 
|  | 323 | bool ThreadCaptureImpl::CapturedThreadInfo(ThreadInfoList& threads) { | 
|  | 324 | threads.clear(); | 
|  | 325 |  | 
|  | 326 | for (auto it = captured_threads_.begin(); it != captured_threads_.end(); it++) { | 
|  | 327 | ThreadInfo t{0, allocator::vector<uintptr_t>(allocator_), std::pair<uintptr_t, uintptr_t>(0, 0)}; | 
|  | 328 | if (!PtraceThreadInfo(it->first, t)) { | 
|  | 329 | return false; | 
|  | 330 | } | 
|  | 331 | threads.push_back(t); | 
|  | 332 | } | 
|  | 333 | return true; | 
|  | 334 | } | 
|  | 335 |  | 
|  | 336 | ThreadCapture::ThreadCapture(pid_t pid, Allocator<ThreadCapture> allocator) { | 
|  | 337 | Allocator<ThreadCaptureImpl> impl_allocator = allocator; | 
|  | 338 | impl_ = impl_allocator.make_unique(pid, impl_allocator); | 
|  | 339 | } | 
|  | 340 |  | 
|  | 341 | ThreadCapture::~ThreadCapture() {} | 
|  | 342 |  | 
|  | 343 | bool ThreadCapture::ListThreads(TidList& tids) { | 
|  | 344 | return impl_->ListThreads(tids); | 
|  | 345 | } | 
|  | 346 |  | 
|  | 347 | bool ThreadCapture::CaptureThreads() { | 
|  | 348 | return impl_->CaptureThreads(); | 
|  | 349 | } | 
|  | 350 |  | 
|  | 351 | bool ThreadCapture::ReleaseThreads() { | 
|  | 352 | return impl_->ReleaseThreads(); | 
|  | 353 | } | 
|  | 354 |  | 
|  | 355 | bool ThreadCapture::ReleaseThread(pid_t tid) { | 
|  | 356 | return impl_->ReleaseThread(tid); | 
|  | 357 | } | 
|  | 358 |  | 
|  | 359 | bool ThreadCapture::CapturedThreadInfo(ThreadInfoList& threads) { | 
|  | 360 | return impl_->CapturedThreadInfo(threads); | 
|  | 361 | } | 
|  | 362 |  | 
|  | 363 | void ThreadCapture::InjectTestFunc(std::function<void(pid_t)>&& f) { | 
|  | 364 | impl_->InjectTestFunc(std::forward<std::function<void(pid_t)>>(f)); | 
|  | 365 | } | 
| Colin Cross | a9939e9 | 2017-06-21 13:13:00 -0700 | [diff] [blame] | 366 |  | 
|  | 367 | }  // namespace android |