// Copyright 2012 Google LLC // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above // copyright notice, this list of conditions and the following disclaimer // in the documentation and/or other materials provided with the // distribution. // * Neither the name of Google LLC nor the names of its // contributors may be used to endorse or promote products derived from // this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // linux_ptrace_dumper.cc: Implement google_breakpad::LinuxPtraceDumper. // See linux_ptrace_dumper.h for detals. // This class was originally splitted from google_breakpad::LinuxDumper. // This code deals with the mechanics of getting information about a crashed // process. Since this code may run in a compromised address space, the same // rules apply as detailed at the top of minidump_writer.h: no libc calls and // use the alternative allocator. #ifdef HAVE_CONFIG_H #include <config.h> // Must come first #endif #include "client/linux/minidump_writer/linux_ptrace_dumper.h" #include <asm/ptrace.h> #include <assert.h> #include <errno.h> #include <fcntl.h> #include <limits.h> #include <stddef.h> #include <stdlib.h> #include <string.h> #include <sys/ptrace.h> #include <sys/uio.h> #include <sys/wait.h> #if defined(__i386) #include <cpuid.h> #endif #include "client/linux/minidump_writer/directory_reader.h" #include "client/linux/minidump_writer/line_reader.h" #include "common/linux/eintr_wrapper.h" #include "common/linux/linux_libc_support.h" #include "third_party/lss/linux_syscall_support.h" #if defined(__arm__) /* * https://elixir.bootlin.com/linux/v6.8-rc2/source/arch/arm/include/asm/user.h#L81 * User specific VFP registers. If only VFPv2 is present, registers 16 to 31 * are ignored by the ptrace system call and the signal handler. */ typedef struct { unsigned long long fpregs[32]; unsigned long fpscr; // Kernel just appends fpscr to the copy of fpregs, so we need to force // compiler to build the same layout. } __attribute__((packed, aligned(4))) user_vfp_t; #endif // defined(__arm__) // Suspends a thread by attaching to it. static bool SuspendThread(pid_t pid) { … } // Resumes a thread by detaching from it. static bool ResumeThread(pid_t pid) { … } namespace google_breakpad { LinuxPtraceDumper::LinuxPtraceDumper(pid_t pid) : … { … } bool LinuxPtraceDumper::BuildProcPath(char* path, pid_t pid, const char* node) const { … } bool LinuxPtraceDumper::CopyFromProcess(void* dest, pid_t child, const void* src, size_t length) { … } // This read VFP registers via either PTRACE_GETREGSET or PTRACE_GETREGS #if defined(__arm__) static bool ReadVFPRegistersArm32(pid_t tid, struct iovec* io) { #ifdef PTRACE_GETREGSET if (sys_ptrace(PTRACE_GETREGSET, tid, reinterpret_cast<void*>(NT_ARM_VFP), io) == 0 && io->iov_len == sizeof(user_vfp_t)) { return true; } #endif // PTRACE_GETREGSET #ifdef PTRACE_GETVFPREGS if (sys_ptrace(PTRACE_GETVFPREGS, tid, nullptr, io->iov_base) == 0) { return true; } #endif // PTRACE_GETVFPREGS return false; } #endif // defined(__arm__) bool LinuxPtraceDumper::ReadRegisterSet(ThreadInfo* info, pid_t tid) { … } bool LinuxPtraceDumper::ReadRegisters(ThreadInfo* info, pid_t tid) { … } // Read thread info from /proc/$pid/status. // Fill out the |tgid|, |ppid| and |pid| members of |info|. If unavailable, // these members are set to -1. Returns true iff all three members are // available. bool LinuxPtraceDumper::GetThreadInfoByIndex(size_t index, ThreadInfo* info) { … } bool LinuxPtraceDumper::IsPostMortem() const { … } bool LinuxPtraceDumper::ThreadsSuspend() { … } bool LinuxPtraceDumper::ThreadsResume() { … } // Parse /proc/$pid/task to list all the threads of the process identified by // pid. bool LinuxPtraceDumper::EnumerateThreads() { … } } // namespace google_breakpad