| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2016 The Android Open Source Project | 
|  | 3 | * All rights reserved. | 
|  | 4 | * | 
|  | 5 | * Redistribution and use in source and binary forms, with or without | 
|  | 6 | * modification, are permitted provided that the following conditions | 
|  | 7 | * are met: | 
|  | 8 | *  * Redistributions of source code must retain the above copyright | 
|  | 9 | *    notice, this list of conditions and the following disclaimer. | 
|  | 10 | *  * Redistributions in binary form must reproduce the above copyright | 
|  | 11 | *    notice, this list of conditions and the following disclaimer in | 
|  | 12 | *    the documentation and/or other materials provided with the | 
|  | 13 | *    distribution. | 
|  | 14 | * | 
|  | 15 | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | 16 | * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | 17 | * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
|  | 18 | * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
|  | 19 | * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|  | 20 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
|  | 21 | * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS | 
|  | 22 | * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | 
|  | 23 | * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | 24 | * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | 
|  | 25 | * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | 
|  | 26 | * SUCH DAMAGE. | 
|  | 27 | */ | 
|  | 28 |  | 
| Elliott Hughes | cbc80ba | 2018-02-13 14:26:29 -0800 | [diff] [blame] | 29 | #pragma once | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 30 |  | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 31 | #include <pthread.h> | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 32 | #include <signal.h> | 
| Christopher Ferris | 4da2503 | 2018-03-07 13:38:48 -0800 | [diff] [blame] | 33 | #include <stdint.h> | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 34 | #include <unistd.h> | 
|  | 35 |  | 
|  | 36 | #include <atomic> | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 37 | #include <memory> | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 38 | #include <mutex> | 
|  | 39 | #include <string> | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 40 | #include <vector> | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 41 |  | 
| Josh Gao | 4956c37 | 2019-12-19 16:35:51 -0800 | [diff] [blame] | 42 | #include <platform/bionic/macros.h> | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 43 |  | 
|  | 44 | class RecordEntry { | 
|  | 45 | public: | 
|  | 46 | RecordEntry(); | 
|  | 47 | virtual ~RecordEntry() = default; | 
|  | 48 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 49 | virtual bool Write(int fd) const = 0; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 50 |  | 
|  | 51 | protected: | 
|  | 52 | pid_t tid_; | 
|  | 53 |  | 
|  | 54 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 55 | BIONIC_DISALLOW_COPY_AND_ASSIGN(RecordEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 56 | }; | 
|  | 57 |  | 
|  | 58 | class ThreadCompleteEntry : public RecordEntry { | 
|  | 59 | public: | 
|  | 60 | ThreadCompleteEntry() = default; | 
|  | 61 | virtual ~ThreadCompleteEntry() = default; | 
|  | 62 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 63 | bool Write(int fd) const override; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 64 |  | 
|  | 65 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 66 | BIONIC_DISALLOW_COPY_AND_ASSIGN(ThreadCompleteEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 67 | }; | 
|  | 68 |  | 
|  | 69 | class AllocEntry : public RecordEntry { | 
|  | 70 | public: | 
| Chia-hung Duan | f7e8b17 | 2022-11-01 21:37:56 +0000 | [diff] [blame] | 71 | explicit AllocEntry(void* pointer, uint64_t st, uint64_t et); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 72 | virtual ~AllocEntry() = default; | 
|  | 73 |  | 
|  | 74 | protected: | 
|  | 75 | void* pointer_; | 
|  | 76 |  | 
| Chia-hung Duan | f7e8b17 | 2022-11-01 21:37:56 +0000 | [diff] [blame] | 77 | // The start/end time of this operation. | 
|  | 78 | uint64_t start_ns_; | 
|  | 79 | uint64_t end_ns_; | 
|  | 80 |  | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 81 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 82 | BIONIC_DISALLOW_COPY_AND_ASSIGN(AllocEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 83 | }; | 
|  | 84 |  | 
|  | 85 | class MallocEntry : public AllocEntry { | 
|  | 86 | public: | 
| Chia-hung Duan | f7e8b17 | 2022-11-01 21:37:56 +0000 | [diff] [blame] | 87 | MallocEntry(void* pointer, size_t size, uint64_t st, uint64_t et); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 88 | virtual ~MallocEntry() = default; | 
|  | 89 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 90 | bool Write(int fd) const override; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 91 |  | 
|  | 92 | protected: | 
|  | 93 | size_t size_; | 
|  | 94 |  | 
|  | 95 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 96 | BIONIC_DISALLOW_COPY_AND_ASSIGN(MallocEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 97 | }; | 
|  | 98 |  | 
|  | 99 | class FreeEntry : public AllocEntry { | 
|  | 100 | public: | 
| Chia-hung Duan | f7e8b17 | 2022-11-01 21:37:56 +0000 | [diff] [blame] | 101 | explicit FreeEntry(void* pointer, uint64_t st, uint64_t et); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 102 | virtual ~FreeEntry() = default; | 
|  | 103 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 104 | bool Write(int fd) const override; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 105 |  | 
|  | 106 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 107 | BIONIC_DISALLOW_COPY_AND_ASSIGN(FreeEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 108 | }; | 
|  | 109 |  | 
|  | 110 | class CallocEntry : public MallocEntry { | 
|  | 111 | public: | 
| Greg Kaiser | 2335213 | 2023-01-09 17:53:07 +0000 | [diff] [blame] | 112 | CallocEntry(void* pointer, size_t nmemb, size_t size, uint64_t st, uint64_t et); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 113 | virtual ~CallocEntry() = default; | 
|  | 114 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 115 | bool Write(int fd) const override; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 116 |  | 
|  | 117 | protected: | 
|  | 118 | size_t nmemb_; | 
|  | 119 |  | 
|  | 120 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 121 | BIONIC_DISALLOW_COPY_AND_ASSIGN(CallocEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 122 | }; | 
|  | 123 |  | 
|  | 124 | class ReallocEntry : public MallocEntry { | 
|  | 125 | public: | 
| Chia-hung Duan | f7e8b17 | 2022-11-01 21:37:56 +0000 | [diff] [blame] | 126 | ReallocEntry(void* pointer, size_t size, void* old_pointer, uint64_t st, uint64_t et); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 127 | virtual ~ReallocEntry() = default; | 
|  | 128 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 129 | bool Write(int fd) const override; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 130 |  | 
|  | 131 | protected: | 
|  | 132 | void* old_pointer_; | 
|  | 133 |  | 
|  | 134 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 135 | BIONIC_DISALLOW_COPY_AND_ASSIGN(ReallocEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 136 | }; | 
|  | 137 |  | 
| Christopher Ferris | cae21a9 | 2018-02-05 18:14:55 -0800 | [diff] [blame] | 138 | // aligned_alloc, posix_memalign, memalign, pvalloc, valloc all recorded with this class. | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 139 | class MemalignEntry : public MallocEntry { | 
|  | 140 | public: | 
| Chia-hung Duan | f7e8b17 | 2022-11-01 21:37:56 +0000 | [diff] [blame] | 141 | MemalignEntry(void* pointer, size_t size, size_t alignment, uint64_t st, uint64_t et); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 142 | virtual ~MemalignEntry() = default; | 
|  | 143 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 144 | bool Write(int fd) const override; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 145 |  | 
|  | 146 | protected: | 
|  | 147 | size_t alignment_; | 
|  | 148 |  | 
|  | 149 | private: | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 150 | BIONIC_DISALLOW_COPY_AND_ASSIGN(MemalignEntry); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 151 | }; | 
|  | 152 |  | 
| Christopher Ferris | 2b2b25b | 2017-04-05 19:13:03 -0700 | [diff] [blame] | 153 | class Config; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 154 |  | 
|  | 155 | class RecordData { | 
|  | 156 | public: | 
|  | 157 | RecordData(); | 
|  | 158 | virtual ~RecordData(); | 
|  | 159 |  | 
|  | 160 | bool Initialize(const Config& config); | 
|  | 161 |  | 
|  | 162 | void AddEntry(const RecordEntry* entry); | 
|  | 163 | void AddEntryOnly(const RecordEntry* entry); | 
|  | 164 |  | 
| Christopher Ferris | 9b88d0a | 2024-06-13 13:22:02 -0700 | [diff] [blame] | 165 | const std::string& file() { return file_; } | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 166 | pthread_key_t key() { return key_; } | 
|  | 167 |  | 
| Christopher Ferris | 9b88d0a | 2024-06-13 13:22:02 -0700 | [diff] [blame] | 168 | static void WriteEntriesOnExit(); | 
|  | 169 |  | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 170 | private: | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 171 | static void WriteData(int, siginfo_t*, void*); | 
|  | 172 | static RecordData* record_obj_; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 173 |  | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 174 | void WriteEntries(); | 
| Christopher Ferris | 9b88d0a | 2024-06-13 13:22:02 -0700 | [diff] [blame] | 175 | void WriteEntries(const std::string& file); | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 176 |  | 
|  | 177 | std::mutex entries_lock_; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 178 | pthread_key_t key_; | 
| Christopher Ferris | 06993a6 | 2022-09-07 18:19:45 -0700 | [diff] [blame] | 179 | std::vector<std::unique_ptr<const RecordEntry>> entries_; | 
|  | 180 | size_t cur_index_; | 
| Christopher Ferris | 9b88d0a | 2024-06-13 13:22:02 -0700 | [diff] [blame] | 181 | std::string file_; | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 182 |  | 
| Elliott Hughes | 5e62b34 | 2018-10-25 11:00:00 -0700 | [diff] [blame] | 183 | BIONIC_DISALLOW_COPY_AND_ASSIGN(RecordData); | 
| Christopher Ferris | 7bd0178 | 2016-04-20 12:30:58 -0700 | [diff] [blame] | 184 | }; |