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::cpu_id_t> cpu, 77 Optional<long> 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); 88 } 89 90 llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr, 91 Optional<lldb::pid_t> pid, 92 Optional<lldb::cpu_id_t> cpu) { 93 return Init(attr, pid, cpu, -1, 0); 94 } 95 96 llvm::Expected<resource_handle::MmapUP> 97 PerfEvent::DoMmap(void *addr, size_t length, int prot, int flags, 98 long int offset, llvm::StringRef buffer_name) { 99 errno = 0; 100 auto mmap_result = ::mmap(addr, length, prot, flags, GetFd(), offset); 101 102 if (mmap_result == MAP_FAILED) { 103 std::string err_msg = 104 llvm::formatv("perf event mmap allocation failed for {0}: {1}", 105 buffer_name, std::strerror(errno)); 106 return createStringError(inconvertibleErrorCode(), err_msg); 107 } 108 return resource_handle::MmapUP(mmap_result, length); 109 } 110 111 llvm::Error PerfEvent::MmapMetadataAndDataBuffer(size_t num_data_pages, 112 bool data_buffer_write) { 113 size_t mmap_size = (num_data_pages + 1) * getpagesize(); 114 if (Expected<resource_handle::MmapUP> mmap_metadata_data = DoMmap( 115 nullptr, mmap_size, PROT_READ | (data_buffer_write ? PROT_WRITE : 0), 116 MAP_SHARED, 0, "metadata and data buffer")) { 117 m_metadata_data_base = std::move(mmap_metadata_data.get()); 118 return Error::success(); 119 } else 120 return mmap_metadata_data.takeError(); 121 } 122 123 llvm::Error PerfEvent::MmapAuxBuffer(size_t num_aux_pages) { 124 if (num_aux_pages == 0) 125 return Error::success(); 126 127 perf_event_mmap_page &metadata_page = GetMetadataPage(); 128 129 metadata_page.aux_offset = 130 metadata_page.data_offset + metadata_page.data_size; 131 metadata_page.aux_size = num_aux_pages * getpagesize(); 132 133 if (Expected<resource_handle::MmapUP> mmap_aux = 134 DoMmap(nullptr, metadata_page.aux_size, PROT_READ, MAP_SHARED, 135 metadata_page.aux_offset, "aux buffer")) { 136 m_aux_base = std::move(mmap_aux.get()); 137 return Error::success(); 138 } else 139 return mmap_aux.takeError(); 140 } 141 142 llvm::Error PerfEvent::MmapMetadataAndBuffers(size_t num_data_pages, 143 size_t num_aux_pages, 144 bool data_buffer_write) { 145 if (num_data_pages != 0 && !isPowerOf2_64(num_data_pages)) 146 return llvm::createStringError( 147 llvm::inconvertibleErrorCode(), 148 llvm::formatv("Number of data pages must be a power of 2, got: {0}", 149 num_data_pages)); 150 if (num_aux_pages != 0 && !isPowerOf2_64(num_aux_pages)) 151 return llvm::createStringError( 152 llvm::inconvertibleErrorCode(), 153 llvm::formatv("Number of aux pages must be a power of 2, got: {0}", 154 num_aux_pages)); 155 if (Error err = MmapMetadataAndDataBuffer(num_data_pages, data_buffer_write)) 156 return err; 157 if (Error err = MmapAuxBuffer(num_aux_pages)) 158 return err; 159 return Error::success(); 160 } 161 162 long PerfEvent::GetFd() const { return *(m_fd.get()); } 163 164 perf_event_mmap_page &PerfEvent::GetMetadataPage() const { 165 return *reinterpret_cast<perf_event_mmap_page *>(m_metadata_data_base.get()); 166 } 167 168 ArrayRef<uint8_t> PerfEvent::GetDataBuffer() const { 169 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 170 return {reinterpret_cast<uint8_t *>(m_metadata_data_base.get()) + 171 mmap_metadata.data_offset, 172 static_cast<size_t>(mmap_metadata.data_size)}; 173 } 174 175 ArrayRef<uint8_t> PerfEvent::GetAuxBuffer() const { 176 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 177 return {reinterpret_cast<uint8_t *>(m_aux_base.get()), 178 static_cast<size_t>(mmap_metadata.aux_size)}; 179 } 180 181 Expected<std::vector<uint8_t>> PerfEvent::GetReadOnlyDataBuffer() { 182 // The following code assumes that the protection level of the DATA page 183 // is PROT_READ. If PROT_WRITE is used, then reading would require that 184 // this piece of code updates some pointers. See more about data_tail 185 // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html. 186 187 bool was_enabled = m_enabled; 188 if (Error err = DisableWithIoctl()) 189 return std::move(err); 190 191 /** 192 * The data buffer and aux buffer have different implementations 193 * with respect to their definition of head pointer when using PROD_READ only. 194 * In the case of Aux data buffer the head always wraps around the aux buffer 195 * and we don't need to care about it, whereas the data_head keeps 196 * increasing and needs to be wrapped by modulus operator 197 */ 198 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 199 200 ArrayRef<uint8_t> data = GetDataBuffer(); 201 uint64_t data_head = mmap_metadata.data_head; 202 uint64_t data_size = mmap_metadata.data_size; 203 std::vector<uint8_t> output; 204 output.reserve(data.size()); 205 206 if (data_head > data_size) { 207 uint64_t actual_data_head = data_head % data_size; 208 // The buffer has wrapped 209 for (uint64_t i = actual_data_head; i < data_size; i++) 210 output.push_back(data[i]); 211 212 for (uint64_t i = 0; i < actual_data_head; i++) 213 output.push_back(data[i]); 214 } else { 215 for (uint64_t i = 0; i < data_head; i++) 216 output.push_back(data[i]); 217 } 218 219 if (was_enabled) { 220 if (Error err = EnableWithIoctl()) 221 return std::move(err); 222 } 223 224 return output; 225 } 226 227 Expected<std::vector<uint8_t>> PerfEvent::GetReadOnlyAuxBuffer() { 228 // The following code assumes that the protection level of the AUX page 229 // is PROT_READ. If PROT_WRITE is used, then reading would require that 230 // this piece of code updates some pointers. See more about aux_tail 231 // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html. 232 233 bool was_enabled = m_enabled; 234 if (Error err = DisableWithIoctl()) 235 return std::move(err); 236 237 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 238 239 ArrayRef<uint8_t> data = GetAuxBuffer(); 240 uint64_t aux_head = mmap_metadata.aux_head; 241 uint64_t aux_size = mmap_metadata.aux_size; 242 std::vector<uint8_t> output; 243 output.reserve(data.size()); 244 245 /** 246 * When configured as ring buffer, the aux buffer keeps wrapping around 247 * the buffer and its not possible to detect how many times the buffer 248 * wrapped. Initially the buffer is filled with zeros,as shown below 249 * so in order to get complete buffer we first copy firstpartsize, followed 250 * by any left over part from beginning to aux_head 251 * 252 * aux_offset [d,d,d,d,d,d,d,d,0,0,0,0,0,0,0,0,0,0,0] aux_size 253 * aux_head->||<- firstpartsize ->| 254 * 255 * */ 256 257 for (uint64_t i = aux_head; i < aux_size; i++) 258 output.push_back(data[i]); 259 260 for (uint64_t i = 0; i < aux_head; i++) 261 output.push_back(data[i]); 262 263 if (was_enabled) { 264 if (Error err = EnableWithIoctl()) 265 return std::move(err); 266 } 267 268 return output; 269 } 270 271 Error PerfEvent::DisableWithIoctl() { 272 if (!m_enabled) 273 return Error::success(); 274 275 if (ioctl(*m_fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) < 0) 276 return createStringError(inconvertibleErrorCode(), 277 "Can't disable perf event. %s", 278 std::strerror(errno)); 279 280 m_enabled = false; 281 return Error::success(); 282 } 283 284 bool PerfEvent::IsEnabled() const { return m_enabled; } 285 286 Error PerfEvent::EnableWithIoctl() { 287 if (m_enabled) 288 return Error::success(); 289 290 if (ioctl(*m_fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) < 0) 291 return createStringError(inconvertibleErrorCode(), 292 "Can't enable perf event. %s", 293 std::strerror(errno)); 294 295 m_enabled = true; 296 return Error::success(); 297 } 298 299 size_t PerfEvent::GetEffectiveDataBufferSize() const { 300 perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 301 if (mmap_metadata.data_head < mmap_metadata.data_size) 302 return mmap_metadata.data_head; 303 else 304 return mmap_metadata.data_size; // The buffer has wrapped. 305 } 306 307 Expected<PerfEvent> 308 lldb_private::process_linux::CreateContextSwitchTracePerfEvent( 309 lldb::cpu_id_t cpu_id, const PerfEvent *parent_perf_event) { 310 Log *log = GetLog(POSIXLog::Trace); 311 #ifndef PERF_ATTR_SIZE_VER5 312 return createStringError(inconvertibleErrorCode(), 313 "Intel PT Linux perf event not supported"); 314 #else 315 perf_event_attr attr; 316 memset(&attr, 0, sizeof(attr)); 317 attr.size = sizeof(attr); 318 attr.sample_type = PERF_SAMPLE_TID | PERF_SAMPLE_TIME; 319 attr.type = PERF_TYPE_SOFTWARE; 320 attr.context_switch = 1; 321 attr.exclude_kernel = 1; 322 attr.sample_id_all = 1; 323 attr.exclude_hv = 1; 324 attr.disabled = parent_perf_event ? !parent_perf_event->IsEnabled() : false; 325 326 // The given perf configuration will produce context switch records of 32 327 // bytes each. Assuming that every context switch will be emitted twice (one 328 // for context switch ins and another one for context switch outs), and that a 329 // context switch will happen at least every half a millisecond per core, we 330 // need 500 * 32 bytes (~16 KB) for a trace of one second, which is much more 331 // than what a regular intel pt trace can get. Pessimistically we pick as 332 // 32KiB for the size of our context switch trace. 333 334 uint64_t data_buffer_size = 32768; 335 uint64_t data_buffer_numpages = data_buffer_size / getpagesize(); 336 337 LLDB_LOG(log, "Will create context switch trace buffer of size {0}", 338 data_buffer_size); 339 340 Optional<long> group_fd; 341 if (parent_perf_event) 342 group_fd = parent_perf_event->GetFd(); 343 344 if (Expected<PerfEvent> perf_event = 345 PerfEvent::Init(attr, /*pid=*/None, cpu_id, group_fd, /*flags=*/0)) { 346 if (Error mmap_err = perf_event->MmapMetadataAndBuffers( 347 data_buffer_numpages, 0, /*data_buffer_write=*/false)) { 348 return std::move(mmap_err); 349 } 350 return perf_event; 351 } else { 352 return perf_event.takeError(); 353 } 354 #endif 355 } 356