1 //===-- Perf.cpp ----------------------------------------------------------===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #include "Perf.h" 10 11 #include "Plugins/Process/POSIX/ProcessPOSIXLog.h" 12 #include "lldb/Host/linux/Support.h" 13 14 #include "llvm/Support/FormatVariadic.h" 15 #include "llvm/Support/MathExtras.h" 16 #include "llvm/Support/MemoryBuffer.h" 17 18 #include <sys/ioctl.h> 19 #include <sys/mman.h> 20 #include <sys/syscall.h> 21 #include <unistd.h> 22 23 using namespace lldb_private; 24 using namespace process_linux; 25 using namespace llvm; 26 27 Expected<LinuxPerfZeroTscConversion> 28 lldb_private::process_linux::LoadPerfTscConversionParameters() { 29 lldb::pid_t pid = getpid(); 30 perf_event_attr attr; 31 memset(&attr, 0, sizeof(attr)); 32 attr.size = sizeof(attr); 33 attr.type = PERF_TYPE_SOFTWARE; 34 attr.config = PERF_COUNT_SW_DUMMY; 35 36 Expected<PerfEvent> perf_event = PerfEvent::Init(attr, pid); 37 if (!perf_event) 38 return perf_event.takeError(); 39 if (Error mmap_err = 40 perf_event->MmapMetadataAndBuffers(/*num_data_pages=*/0, 41 /*num_aux_pages=*/0, 42 /*data_buffer_write=*/false)) 43 return std::move(mmap_err); 44 45 perf_event_mmap_page &mmap_metada = perf_event->GetMetadataPage(); 46 if (mmap_metada.cap_user_time && mmap_metada.cap_user_time_zero) { 47 return LinuxPerfZeroTscConversion{ 48 mmap_metada.time_mult, mmap_metada.time_shift, mmap_metada.time_zero}; 49 } else { 50 auto err_cap = 51 !mmap_metada.cap_user_time ? "cap_user_time" : "cap_user_time_zero"; 52 std::string err_msg = 53 llvm::formatv("Can't get TSC to real time conversion values. " 54 "perf_event capability '{0}' not supported.", 55 err_cap); 56 return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg); 57 } 58 } 59 60 void resource_handle::MmapDeleter::operator()(void *ptr) { 61 if (m_bytes && ptr != nullptr) 62 munmap(ptr, m_bytes); 63 } 64 65 void resource_handle::FileDescriptorDeleter::operator()(long *ptr) { 66 if (ptr == nullptr) 67 return; 68 if (*ptr == -1) 69 return; 70 close(*ptr); 71 std::default_delete<long>()(ptr); 72 } 73 74 llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr, 75 Optional<lldb::pid_t> pid, 76 Optional<lldb::core_id_t> cpu, 77 Optional<int> group_fd, 78 unsigned long flags) { 79 errno = 0; 80 long fd = syscall(SYS_perf_event_open, &attr, pid.getValueOr(-1), 81 cpu.getValueOr(-1), group_fd.getValueOr(-1), flags); 82 if (fd == -1) { 83 std::string err_msg = 84 llvm::formatv("perf event syscall failed: {0}", std::strerror(errno)); 85 return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg); 86 } 87 return PerfEvent(fd, attr.disabled ? CollectionState::Disabled 88 : CollectionState::Enabled); 89 } 90 91 llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr, 92 Optional<lldb::pid_t> pid, 93 Optional<lldb::core_id_t> cpu) { 94 return Init(attr, pid, cpu, -1, 0); 95 } 96 97 llvm::Expected<resource_handle::MmapUP> 98 PerfEvent::DoMmap(void *addr, size_t length, int prot, int flags, 99 long int offset, llvm::StringRef buffer_name) { 100 errno = 0; 101 auto mmap_result = ::mmap(addr, length, prot, flags, GetFd(), offset); 102 103 if (mmap_result == MAP_FAILED) { 104 std::string err_msg = 105 llvm::formatv("perf event mmap allocation failed for {0}: {1}", 106 buffer_name, std::strerror(errno)); 107 return createStringError(inconvertibleErrorCode(), err_msg); 108 } 109 return resource_handle::MmapUP(mmap_result, length); 110 } 111 112 llvm::Error PerfEvent::MmapMetadataAndDataBuffer(size_t num_data_pages, 113 bool data_buffer_write) { 114 size_t mmap_size = (num_data_pages + 1) * getpagesize(); 115 if (Expected<resource_handle::MmapUP> mmap_metadata_data = DoMmap( 116 nullptr, mmap_size, PROT_READ | (data_buffer_write ? PROT_WRITE : 0), 117 MAP_SHARED, 0, "metadata and data buffer")) { 118 m_metadata_data_base = std::move(mmap_metadata_data.get()); 119 return Error::success(); 120 } else 121 return mmap_metadata_data.takeError(); 122 } 123 124 llvm::Error PerfEvent::MmapAuxBuffer(size_t num_aux_pages) { 125 if (num_aux_pages == 0) 126 return Error::success(); 127 128 perf_event_mmap_page &metadata_page = GetMetadataPage(); 129 130 metadata_page.aux_offset = 131 metadata_page.data_offset + metadata_page.data_size; 132 metadata_page.aux_size = num_aux_pages * getpagesize(); 133 134 if (Expected<resource_handle::MmapUP> mmap_aux = 135 DoMmap(nullptr, metadata_page.aux_size, PROT_READ, MAP_SHARED, 136 metadata_page.aux_offset, "aux buffer")) { 137 m_aux_base = std::move(mmap_aux.get()); 138 return Error::success(); 139 } else 140 return mmap_aux.takeError(); 141 } 142 143 llvm::Error PerfEvent::MmapMetadataAndBuffers(size_t num_data_pages, 144 size_t num_aux_pages, 145 bool data_buffer_write) { 146 if (num_data_pages != 0 && !isPowerOf2_64(num_data_pages)) 147 return llvm::createStringError( 148 llvm::inconvertibleErrorCode(), 149 llvm::formatv("Number of data pages must be a power of 2, got: {0}", 150 num_data_pages)); 151 if (num_aux_pages != 0 && !isPowerOf2_64(num_aux_pages)) 152 return llvm::createStringError( 153 llvm::inconvertibleErrorCode(), 154 llvm::formatv("Number of aux pages must be a power of 2, got: {0}", 155 num_aux_pages)); 156 if (Error err = MmapMetadataAndDataBuffer(num_data_pages, data_buffer_write)) 157 return err; 158 if (Error err = MmapAuxBuffer(num_aux_pages)) 159 return err; 160 return Error::success(); 161 } 162 163 long PerfEvent::GetFd() const { return *(m_fd.get()); } 164 165 perf_event_mmap_page &PerfEvent::GetMetadataPage() const { 166 return *reinterpret_cast<perf_event_mmap_page *>(m_metadata_data_base.get()); 167 } 168 169 ArrayRef<uint8_t> PerfEvent::GetDataBuffer() const { 170 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 171 return {reinterpret_cast<uint8_t *>(m_metadata_data_base.get()) + 172 mmap_metadata.data_offset, 173 static_cast<size_t>(mmap_metadata.data_size)}; 174 } 175 176 ArrayRef<uint8_t> PerfEvent::GetAuxBuffer() const { 177 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 178 return {reinterpret_cast<uint8_t *>(m_aux_base.get()), 179 static_cast<size_t>(mmap_metadata.aux_size)}; 180 } 181 182 Expected<std::vector<uint8_t>> 183 PerfEvent::ReadFlushedOutDataCyclicBuffer(size_t offset, size_t size) { 184 CollectionState previous_state = m_collection_state; 185 if (Error err = DisableWithIoctl()) 186 return std::move(err); 187 188 /** 189 * The data buffer and aux buffer have different implementations 190 * with respect to their definition of head pointer. In the case 191 * of Aux data buffer the head always wraps around the aux buffer 192 * and we don't need to care about it, whereas the data_head keeps 193 * increasing and needs to be wrapped by modulus operator 194 */ 195 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 196 197 ArrayRef<uint8_t> data = GetDataBuffer(); 198 uint64_t data_head = mmap_metadata.data_head; 199 uint64_t data_size = mmap_metadata.data_size; 200 std::vector<uint8_t> output; 201 output.reserve(size); 202 203 if (data_head > data_size) { 204 uint64_t actual_data_head = data_head % data_size; 205 // The buffer has wrapped 206 for (uint64_t i = actual_data_head + offset; 207 i < data_size && output.size() < size; i++) 208 output.push_back(data[i]); 209 210 // We need to find the starting position for the left part if the offset was 211 // too big 212 uint64_t left_part_start = actual_data_head + offset >= data_size 213 ? actual_data_head + offset - data_size 214 : 0; 215 for (uint64_t i = left_part_start; 216 i < actual_data_head && output.size() < size; i++) 217 output.push_back(data[i]); 218 } else { 219 for (uint64_t i = offset; i < data_head && output.size() < size; i++) 220 output.push_back(data[i]); 221 } 222 223 if (previous_state == CollectionState::Enabled) { 224 if (Error err = EnableWithIoctl()) 225 return std::move(err); 226 } 227 228 if (output.size() != size) 229 return createStringError(inconvertibleErrorCode(), 230 formatv("Requested {0} bytes of perf_event data " 231 "buffer but only {1} are available", 232 size, output.size())); 233 234 return data; 235 } 236 237 Expected<std::vector<uint8_t>> 238 PerfEvent::ReadFlushedOutAuxCyclicBuffer(size_t offset, size_t size) { 239 CollectionState previous_state = m_collection_state; 240 if (Error err = DisableWithIoctl()) 241 return std::move(err); 242 243 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 244 245 ArrayRef<uint8_t> data = GetAuxBuffer(); 246 uint64_t aux_head = mmap_metadata.aux_head; 247 uint64_t aux_size = mmap_metadata.aux_size; 248 std::vector<uint8_t> output; 249 output.reserve(size); 250 251 /** 252 * When configured as ring buffer, the aux buffer keeps wrapping around 253 * the buffer and its not possible to detect how many times the buffer 254 * wrapped. Initially the buffer is filled with zeros,as shown below 255 * so in order to get complete buffer we first copy firstpartsize, followed 256 * by any left over part from beginning to aux_head 257 * 258 * aux_offset [d,d,d,d,d,d,d,d,0,0,0,0,0,0,0,0,0,0,0] aux_size 259 * aux_head->||<- firstpartsize ->| 260 * 261 * */ 262 263 for (uint64_t i = aux_head + offset; i < aux_size && output.size() < size; 264 i++) 265 output.push_back(data[i]); 266 267 // We need to find the starting position for the left part if the offset was 268 // too big 269 uint64_t left_part_start = 270 aux_head + offset >= aux_size ? aux_head + offset - aux_size : 0; 271 for (uint64_t i = left_part_start; i < aux_head && output.size() < size; i++) 272 output.push_back(data[i]); 273 274 if (previous_state == CollectionState::Enabled) { 275 if (Error err = EnableWithIoctl()) 276 return std::move(err); 277 } 278 279 if (output.size() != size) 280 return createStringError(inconvertibleErrorCode(), 281 formatv("Requested {0} bytes of perf_event aux " 282 "buffer but only {1} are available", 283 size, output.size())); 284 285 return data; 286 } 287 288 Error PerfEvent::DisableWithIoctl() { 289 if (m_collection_state == CollectionState::Disabled) 290 return Error::success(); 291 292 if (ioctl(*m_fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) < 0) 293 return createStringError(inconvertibleErrorCode(), 294 "Can't disable perf event. %s", 295 std::strerror(errno)); 296 297 m_collection_state = CollectionState::Disabled; 298 return Error::success(); 299 } 300 301 Error PerfEvent::EnableWithIoctl() { 302 if (m_collection_state == CollectionState::Enabled) 303 return Error::success(); 304 305 if (ioctl(*m_fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) < 0) 306 return createStringError(inconvertibleErrorCode(), 307 "Can't enable perf event. %s", 308 std::strerror(errno)); 309 310 m_collection_state = CollectionState::Enabled; 311 return Error::success(); 312 } 313 314 size_t PerfEvent::GetEffectiveDataBufferSize() const { 315 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 316 if (mmap_metadata.data_head <= mmap_metadata.data_size) 317 return mmap_metadata.data_head; 318 else 319 return mmap_metadata.data_size; // The buffer has wrapped. 320 } 321