1e6c84f82SJakob Johnson //===-- Perf.cpp ----------------------------------------------------------===// 2e6c84f82SJakob Johnson // 3e6c84f82SJakob Johnson // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4e6c84f82SJakob Johnson // See https://llvm.org/LICENSE.txt for license information. 5e6c84f82SJakob Johnson // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6e6c84f82SJakob Johnson // 7e6c84f82SJakob Johnson //===----------------------------------------------------------------------===// 8e6c84f82SJakob Johnson 9e6c84f82SJakob Johnson #include "Perf.h" 10e6c84f82SJakob Johnson 117b73de9eSWalter Erquinigo #include "Plugins/Process/POSIX/ProcessPOSIXLog.h" 125de0a3e9SWalter Erquinigo #include "lldb/Host/linux/Support.h" 135de0a3e9SWalter Erquinigo 14e6c84f82SJakob Johnson #include "llvm/Support/FormatVariadic.h" 15e6c84f82SJakob Johnson #include "llvm/Support/MathExtras.h" 165de0a3e9SWalter Erquinigo #include "llvm/Support/MemoryBuffer.h" 17e6c84f82SJakob Johnson 181637545fSWalter Erquinigo #include <sys/ioctl.h> 19e6c84f82SJakob Johnson #include <sys/mman.h> 20e6c84f82SJakob Johnson #include <sys/syscall.h> 21e6c84f82SJakob Johnson #include <unistd.h> 22e6c84f82SJakob Johnson 23e6c84f82SJakob Johnson using namespace lldb_private; 24e6c84f82SJakob Johnson using namespace process_linux; 25e6c84f82SJakob Johnson using namespace llvm; 26e6c84f82SJakob Johnson 279b79187cSJakob Johnson Expected<LinuxPerfZeroTscConversion> 289b79187cSJakob Johnson lldb_private::process_linux::LoadPerfTscConversionParameters() { 29e6c84f82SJakob Johnson lldb::pid_t pid = getpid(); 30e6c84f82SJakob Johnson perf_event_attr attr; 31e6c84f82SJakob Johnson memset(&attr, 0, sizeof(attr)); 32e6c84f82SJakob Johnson attr.size = sizeof(attr); 33e6c84f82SJakob Johnson attr.type = PERF_TYPE_SOFTWARE; 34e6c84f82SJakob Johnson attr.config = PERF_COUNT_SW_DUMMY; 35e6c84f82SJakob Johnson 36e6c84f82SJakob Johnson Expected<PerfEvent> perf_event = PerfEvent::Init(attr, pid); 37e6c84f82SJakob Johnson if (!perf_event) 38e6c84f82SJakob Johnson return perf_event.takeError(); 39fc5ef57cSWalter Erquinigo if (Error mmap_err = 40fc5ef57cSWalter Erquinigo perf_event->MmapMetadataAndBuffers(/*num_data_pages=*/0, 41fc5ef57cSWalter Erquinigo /*num_aux_pages=*/0, 42fc5ef57cSWalter Erquinigo /*data_buffer_write=*/false)) 43e6c84f82SJakob Johnson return std::move(mmap_err); 44e6c84f82SJakob Johnson 45e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metada = perf_event->GetMetadataPage(); 46e6c84f82SJakob Johnson if (mmap_metada.cap_user_time && mmap_metada.cap_user_time_zero) { 479b79187cSJakob Johnson return LinuxPerfZeroTscConversion{ 48e6c84f82SJakob Johnson mmap_metada.time_mult, mmap_metada.time_shift, mmap_metada.time_zero}; 49e6c84f82SJakob Johnson } else { 50e6c84f82SJakob Johnson auto err_cap = 51e6c84f82SJakob Johnson !mmap_metada.cap_user_time ? "cap_user_time" : "cap_user_time_zero"; 52e6c84f82SJakob Johnson std::string err_msg = 53e6c84f82SJakob Johnson llvm::formatv("Can't get TSC to real time conversion values. " 54e6c84f82SJakob Johnson "perf_event capability '{0}' not supported.", 55e6c84f82SJakob Johnson err_cap); 56e6c84f82SJakob Johnson return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg); 57e6c84f82SJakob Johnson } 58e6c84f82SJakob Johnson } 59e6c84f82SJakob Johnson 60e6c84f82SJakob Johnson void resource_handle::MmapDeleter::operator()(void *ptr) { 61e6c84f82SJakob Johnson if (m_bytes && ptr != nullptr) 62e6c84f82SJakob Johnson munmap(ptr, m_bytes); 63e6c84f82SJakob Johnson } 64e6c84f82SJakob Johnson 65e6c84f82SJakob Johnson void resource_handle::FileDescriptorDeleter::operator()(long *ptr) { 66e6c84f82SJakob Johnson if (ptr == nullptr) 67e6c84f82SJakob Johnson return; 68e6c84f82SJakob Johnson if (*ptr == -1) 69e6c84f82SJakob Johnson return; 70e6c84f82SJakob Johnson close(*ptr); 71e6c84f82SJakob Johnson std::default_delete<long>()(ptr); 72e6c84f82SJakob Johnson } 73e6c84f82SJakob Johnson 74e6c84f82SJakob Johnson llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr, 751f49714dSWalter Erquinigo Optional<lldb::pid_t> pid, 761f49714dSWalter Erquinigo Optional<lldb::core_id_t> cpu, 77*03cc58ffSWalter Erquinigo Optional<long> group_fd, 781f49714dSWalter Erquinigo unsigned long flags) { 79e6c84f82SJakob Johnson errno = 0; 801f49714dSWalter Erquinigo long fd = syscall(SYS_perf_event_open, &attr, pid.getValueOr(-1), 811f49714dSWalter Erquinigo cpu.getValueOr(-1), group_fd.getValueOr(-1), flags); 82e6c84f82SJakob Johnson if (fd == -1) { 83e6c84f82SJakob Johnson std::string err_msg = 84e6c84f82SJakob Johnson llvm::formatv("perf event syscall failed: {0}", std::strerror(errno)); 85e6c84f82SJakob Johnson return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg); 86e6c84f82SJakob Johnson } 87*03cc58ffSWalter Erquinigo return PerfEvent(fd, !attr.disabled); 88e6c84f82SJakob Johnson } 89e6c84f82SJakob Johnson 90e6c84f82SJakob Johnson llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr, 911f49714dSWalter Erquinigo Optional<lldb::pid_t> pid, 921f49714dSWalter Erquinigo Optional<lldb::core_id_t> cpu) { 931f49714dSWalter Erquinigo return Init(attr, pid, cpu, -1, 0); 94e6c84f82SJakob Johnson } 95e6c84f82SJakob Johnson 96e6c84f82SJakob Johnson llvm::Expected<resource_handle::MmapUP> 97e6c84f82SJakob Johnson PerfEvent::DoMmap(void *addr, size_t length, int prot, int flags, 98e6c84f82SJakob Johnson long int offset, llvm::StringRef buffer_name) { 99e6c84f82SJakob Johnson errno = 0; 100a7582059SWalter Erquinigo auto mmap_result = ::mmap(addr, length, prot, flags, GetFd(), offset); 101e6c84f82SJakob Johnson 102e6c84f82SJakob Johnson if (mmap_result == MAP_FAILED) { 103e6c84f82SJakob Johnson std::string err_msg = 104e6c84f82SJakob Johnson llvm::formatv("perf event mmap allocation failed for {0}: {1}", 105e6c84f82SJakob Johnson buffer_name, std::strerror(errno)); 106e6c84f82SJakob Johnson return createStringError(inconvertibleErrorCode(), err_msg); 107e6c84f82SJakob Johnson } 108e6c84f82SJakob Johnson return resource_handle::MmapUP(mmap_result, length); 109e6c84f82SJakob Johnson } 110e6c84f82SJakob Johnson 111fc5ef57cSWalter Erquinigo llvm::Error PerfEvent::MmapMetadataAndDataBuffer(size_t num_data_pages, 112fc5ef57cSWalter Erquinigo bool data_buffer_write) { 113e6c84f82SJakob Johnson size_t mmap_size = (num_data_pages + 1) * getpagesize(); 114fc5ef57cSWalter Erquinigo if (Expected<resource_handle::MmapUP> mmap_metadata_data = DoMmap( 115fc5ef57cSWalter Erquinigo nullptr, mmap_size, PROT_READ | (data_buffer_write ? PROT_WRITE : 0), 116fc5ef57cSWalter Erquinigo MAP_SHARED, 0, "metadata and data buffer")) { 117e6c84f82SJakob Johnson m_metadata_data_base = std::move(mmap_metadata_data.get()); 118e6c84f82SJakob Johnson return Error::success(); 119e6c84f82SJakob Johnson } else 120e6c84f82SJakob Johnson return mmap_metadata_data.takeError(); 121e6c84f82SJakob Johnson } 122e6c84f82SJakob Johnson 123e6c84f82SJakob Johnson llvm::Error PerfEvent::MmapAuxBuffer(size_t num_aux_pages) { 124e6c84f82SJakob Johnson if (num_aux_pages == 0) 125e6c84f82SJakob Johnson return Error::success(); 126e6c84f82SJakob Johnson 127e6c84f82SJakob Johnson perf_event_mmap_page &metadata_page = GetMetadataPage(); 128fc5ef57cSWalter Erquinigo 129e6c84f82SJakob Johnson metadata_page.aux_offset = 130e6c84f82SJakob Johnson metadata_page.data_offset + metadata_page.data_size; 131e6c84f82SJakob Johnson metadata_page.aux_size = num_aux_pages * getpagesize(); 132e6c84f82SJakob Johnson 133e6c84f82SJakob Johnson if (Expected<resource_handle::MmapUP> mmap_aux = 134e6c84f82SJakob Johnson DoMmap(nullptr, metadata_page.aux_size, PROT_READ, MAP_SHARED, 135e6c84f82SJakob Johnson metadata_page.aux_offset, "aux buffer")) { 136e6c84f82SJakob Johnson m_aux_base = std::move(mmap_aux.get()); 137e6c84f82SJakob Johnson return Error::success(); 138e6c84f82SJakob Johnson } else 139e6c84f82SJakob Johnson return mmap_aux.takeError(); 140e6c84f82SJakob Johnson } 141e6c84f82SJakob Johnson 142e6c84f82SJakob Johnson llvm::Error PerfEvent::MmapMetadataAndBuffers(size_t num_data_pages, 143fc5ef57cSWalter Erquinigo size_t num_aux_pages, 144fc5ef57cSWalter Erquinigo bool data_buffer_write) { 145e6c84f82SJakob Johnson if (num_data_pages != 0 && !isPowerOf2_64(num_data_pages)) 146e6c84f82SJakob Johnson return llvm::createStringError( 147e6c84f82SJakob Johnson llvm::inconvertibleErrorCode(), 148e6c84f82SJakob Johnson llvm::formatv("Number of data pages must be a power of 2, got: {0}", 149e6c84f82SJakob Johnson num_data_pages)); 150e6c84f82SJakob Johnson if (num_aux_pages != 0 && !isPowerOf2_64(num_aux_pages)) 151e6c84f82SJakob Johnson return llvm::createStringError( 152e6c84f82SJakob Johnson llvm::inconvertibleErrorCode(), 153e6c84f82SJakob Johnson llvm::formatv("Number of aux pages must be a power of 2, got: {0}", 154e6c84f82SJakob Johnson num_aux_pages)); 155fc5ef57cSWalter Erquinigo if (Error err = MmapMetadataAndDataBuffer(num_data_pages, data_buffer_write)) 156e6c84f82SJakob Johnson return err; 157e6c84f82SJakob Johnson if (Error err = MmapAuxBuffer(num_aux_pages)) 158e6c84f82SJakob Johnson return err; 159e6c84f82SJakob Johnson return Error::success(); 160e6c84f82SJakob Johnson } 161e6c84f82SJakob Johnson 162e6c84f82SJakob Johnson long PerfEvent::GetFd() const { return *(m_fd.get()); } 163e6c84f82SJakob Johnson 164e6c84f82SJakob Johnson perf_event_mmap_page &PerfEvent::GetMetadataPage() const { 165e6c84f82SJakob Johnson return *reinterpret_cast<perf_event_mmap_page *>(m_metadata_data_base.get()); 166e6c84f82SJakob Johnson } 167e6c84f82SJakob Johnson 168e6c84f82SJakob Johnson ArrayRef<uint8_t> PerfEvent::GetDataBuffer() const { 169e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 170e6c84f82SJakob Johnson return {reinterpret_cast<uint8_t *>(m_metadata_data_base.get()) + 171e6c84f82SJakob Johnson mmap_metadata.data_offset, 172e412529cSJakob Johnson static_cast<size_t>(mmap_metadata.data_size)}; 173e6c84f82SJakob Johnson } 174e6c84f82SJakob Johnson 175e6c84f82SJakob Johnson ArrayRef<uint8_t> PerfEvent::GetAuxBuffer() const { 176e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 177e6c84f82SJakob Johnson return {reinterpret_cast<uint8_t *>(m_aux_base.get()), 178e412529cSJakob Johnson static_cast<size_t>(mmap_metadata.aux_size)}; 179e6c84f82SJakob Johnson } 1801637545fSWalter Erquinigo 181a7582059SWalter Erquinigo Expected<std::vector<uint8_t>> 182fc5ef57cSWalter Erquinigo PerfEvent::ReadFlushedOutDataCyclicBuffer(size_t offset, size_t size) { 183*03cc58ffSWalter Erquinigo // The following code assumes that the protection level of the DATA page 184*03cc58ffSWalter Erquinigo // is PROT_READ. If PROT_WRITE is used, then reading would require that 185*03cc58ffSWalter Erquinigo // this piece of code updates some pointers. See more about data_tail 186*03cc58ffSWalter Erquinigo // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html. 187*03cc58ffSWalter Erquinigo 188*03cc58ffSWalter Erquinigo bool was_enabled = m_enabled; 189fc5ef57cSWalter Erquinigo if (Error err = DisableWithIoctl()) 190fc5ef57cSWalter Erquinigo return std::move(err); 191fc5ef57cSWalter Erquinigo 192fc5ef57cSWalter Erquinigo /** 193fc5ef57cSWalter Erquinigo * The data buffer and aux buffer have different implementations 194fc5ef57cSWalter Erquinigo * with respect to their definition of head pointer. In the case 195fc5ef57cSWalter Erquinigo * of Aux data buffer the head always wraps around the aux buffer 196fc5ef57cSWalter Erquinigo * and we don't need to care about it, whereas the data_head keeps 197fc5ef57cSWalter Erquinigo * increasing and needs to be wrapped by modulus operator 198fc5ef57cSWalter Erquinigo */ 199fc5ef57cSWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 200fc5ef57cSWalter Erquinigo 201fc5ef57cSWalter Erquinigo ArrayRef<uint8_t> data = GetDataBuffer(); 202fc5ef57cSWalter Erquinigo uint64_t data_head = mmap_metadata.data_head; 203fc5ef57cSWalter Erquinigo uint64_t data_size = mmap_metadata.data_size; 204fc5ef57cSWalter Erquinigo std::vector<uint8_t> output; 205fc5ef57cSWalter Erquinigo output.reserve(size); 206fc5ef57cSWalter Erquinigo 207fc5ef57cSWalter Erquinigo if (data_head > data_size) { 208fc5ef57cSWalter Erquinigo uint64_t actual_data_head = data_head % data_size; 209fc5ef57cSWalter Erquinigo // The buffer has wrapped 210fc5ef57cSWalter Erquinigo for (uint64_t i = actual_data_head + offset; 211fc5ef57cSWalter Erquinigo i < data_size && output.size() < size; i++) 212fc5ef57cSWalter Erquinigo output.push_back(data[i]); 213fc5ef57cSWalter Erquinigo 214fc5ef57cSWalter Erquinigo // We need to find the starting position for the left part if the offset was 215fc5ef57cSWalter Erquinigo // too big 216fc5ef57cSWalter Erquinigo uint64_t left_part_start = actual_data_head + offset >= data_size 217fc5ef57cSWalter Erquinigo ? actual_data_head + offset - data_size 218fc5ef57cSWalter Erquinigo : 0; 219fc5ef57cSWalter Erquinigo for (uint64_t i = left_part_start; 220fc5ef57cSWalter Erquinigo i < actual_data_head && output.size() < size; i++) 221fc5ef57cSWalter Erquinigo output.push_back(data[i]); 222fc5ef57cSWalter Erquinigo } else { 223fc5ef57cSWalter Erquinigo for (uint64_t i = offset; i < data_head && output.size() < size; i++) 224fc5ef57cSWalter Erquinigo output.push_back(data[i]); 225fc5ef57cSWalter Erquinigo } 226fc5ef57cSWalter Erquinigo 227*03cc58ffSWalter Erquinigo if (was_enabled) { 228fc5ef57cSWalter Erquinigo if (Error err = EnableWithIoctl()) 229fc5ef57cSWalter Erquinigo return std::move(err); 230fc5ef57cSWalter Erquinigo } 231fc5ef57cSWalter Erquinigo 232fc5ef57cSWalter Erquinigo if (output.size() != size) 233fc5ef57cSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 234fc5ef57cSWalter Erquinigo formatv("Requested {0} bytes of perf_event data " 235fc5ef57cSWalter Erquinigo "buffer but only {1} are available", 236fc5ef57cSWalter Erquinigo size, output.size())); 237fc5ef57cSWalter Erquinigo 2381a3f9969SWalter Erquinigo return output; 239fc5ef57cSWalter Erquinigo } 240fc5ef57cSWalter Erquinigo 241fc5ef57cSWalter Erquinigo Expected<std::vector<uint8_t>> 242a7582059SWalter Erquinigo PerfEvent::ReadFlushedOutAuxCyclicBuffer(size_t offset, size_t size) { 243*03cc58ffSWalter Erquinigo // The following code assumes that the protection level of the AUX page 244*03cc58ffSWalter Erquinigo // is PROT_READ. If PROT_WRITE is used, then reading would require that 245*03cc58ffSWalter Erquinigo // this piece of code updates some pointers. See more about aux_tail 246*03cc58ffSWalter Erquinigo // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html. 247*03cc58ffSWalter Erquinigo 248*03cc58ffSWalter Erquinigo bool was_enabled = m_enabled; 249a7582059SWalter Erquinigo if (Error err = DisableWithIoctl()) 250a7582059SWalter Erquinigo return std::move(err); 251a7582059SWalter Erquinigo 252a7582059SWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 253a7582059SWalter Erquinigo 254fc5ef57cSWalter Erquinigo ArrayRef<uint8_t> data = GetAuxBuffer(); 255fc5ef57cSWalter Erquinigo uint64_t aux_head = mmap_metadata.aux_head; 256fc5ef57cSWalter Erquinigo uint64_t aux_size = mmap_metadata.aux_size; 257fc5ef57cSWalter Erquinigo std::vector<uint8_t> output; 258fc5ef57cSWalter Erquinigo output.reserve(size); 259a7582059SWalter Erquinigo 260a7582059SWalter Erquinigo /** 261a7582059SWalter Erquinigo * When configured as ring buffer, the aux buffer keeps wrapping around 262a7582059SWalter Erquinigo * the buffer and its not possible to detect how many times the buffer 263a7582059SWalter Erquinigo * wrapped. Initially the buffer is filled with zeros,as shown below 264a7582059SWalter Erquinigo * so in order to get complete buffer we first copy firstpartsize, followed 265a7582059SWalter Erquinigo * by any left over part from beginning to aux_head 266a7582059SWalter Erquinigo * 267a7582059SWalter Erquinigo * aux_offset [d,d,d,d,d,d,d,d,0,0,0,0,0,0,0,0,0,0,0] aux_size 268a7582059SWalter Erquinigo * aux_head->||<- firstpartsize ->| 269a7582059SWalter Erquinigo * 270a7582059SWalter Erquinigo * */ 271a7582059SWalter Erquinigo 272fc5ef57cSWalter Erquinigo for (uint64_t i = aux_head + offset; i < aux_size && output.size() < size; 273fc5ef57cSWalter Erquinigo i++) 274fc5ef57cSWalter Erquinigo output.push_back(data[i]); 275fc5ef57cSWalter Erquinigo 276fc5ef57cSWalter Erquinigo // We need to find the starting position for the left part if the offset was 277fc5ef57cSWalter Erquinigo // too big 278fc5ef57cSWalter Erquinigo uint64_t left_part_start = 279fc5ef57cSWalter Erquinigo aux_head + offset >= aux_size ? aux_head + offset - aux_size : 0; 280fc5ef57cSWalter Erquinigo for (uint64_t i = left_part_start; i < aux_head && output.size() < size; i++) 281fc5ef57cSWalter Erquinigo output.push_back(data[i]); 282a7582059SWalter Erquinigo 283*03cc58ffSWalter Erquinigo if (was_enabled) { 284a7582059SWalter Erquinigo if (Error err = EnableWithIoctl()) 285a7582059SWalter Erquinigo return std::move(err); 286a7582059SWalter Erquinigo } 287a7582059SWalter Erquinigo 288fc5ef57cSWalter Erquinigo if (output.size() != size) 289fc5ef57cSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 290fc5ef57cSWalter Erquinigo formatv("Requested {0} bytes of perf_event aux " 291fc5ef57cSWalter Erquinigo "buffer but only {1} are available", 292fc5ef57cSWalter Erquinigo size, output.size())); 293fc5ef57cSWalter Erquinigo 2941a3f9969SWalter Erquinigo return output; 295a7582059SWalter Erquinigo } 296a7582059SWalter Erquinigo 297a7582059SWalter Erquinigo Error PerfEvent::DisableWithIoctl() { 298*03cc58ffSWalter Erquinigo if (!m_enabled) 299a7582059SWalter Erquinigo return Error::success(); 300a7582059SWalter Erquinigo 301a7582059SWalter Erquinigo if (ioctl(*m_fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) < 0) 3021637545fSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 3031637545fSWalter Erquinigo "Can't disable perf event. %s", 3041637545fSWalter Erquinigo std::strerror(errno)); 305a7582059SWalter Erquinigo 306*03cc58ffSWalter Erquinigo m_enabled = false; 3071637545fSWalter Erquinigo return Error::success(); 3081637545fSWalter Erquinigo } 3091637545fSWalter Erquinigo 310*03cc58ffSWalter Erquinigo bool PerfEvent::IsEnabled() const { return m_enabled; } 311*03cc58ffSWalter Erquinigo 312a7582059SWalter Erquinigo Error PerfEvent::EnableWithIoctl() { 313*03cc58ffSWalter Erquinigo if (m_enabled) 3141637545fSWalter Erquinigo return Error::success(); 315a7582059SWalter Erquinigo 316a7582059SWalter Erquinigo if (ioctl(*m_fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) < 0) 317a7582059SWalter Erquinigo return createStringError(inconvertibleErrorCode(), 318a7582059SWalter Erquinigo "Can't enable perf event. %s", 319a7582059SWalter Erquinigo std::strerror(errno)); 320a7582059SWalter Erquinigo 321*03cc58ffSWalter Erquinigo m_enabled = true; 322a7582059SWalter Erquinigo return Error::success(); 323a7582059SWalter Erquinigo } 324a7582059SWalter Erquinigo 325a7582059SWalter Erquinigo size_t PerfEvent::GetEffectiveDataBufferSize() const { 326a7582059SWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 3271a3f9969SWalter Erquinigo if (mmap_metadata.data_head < mmap_metadata.data_size) 328a7582059SWalter Erquinigo return mmap_metadata.data_head; 329a7582059SWalter Erquinigo else 330a7582059SWalter Erquinigo return mmap_metadata.data_size; // The buffer has wrapped. 3311637545fSWalter Erquinigo } 332*03cc58ffSWalter Erquinigo 333*03cc58ffSWalter Erquinigo Expected<PerfEvent> 334*03cc58ffSWalter Erquinigo lldb_private::process_linux::CreateContextSwitchTracePerfEvent( 335*03cc58ffSWalter Erquinigo lldb::core_id_t core_id, const PerfEvent *parent_perf_event) { 336*03cc58ffSWalter Erquinigo Log *log = GetLog(POSIXLog::Trace); 337*03cc58ffSWalter Erquinigo #ifndef PERF_ATTR_SIZE_VER5 338*03cc58ffSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 339*03cc58ffSWalter Erquinigo "Intel PT Linux perf event not supported"); 340*03cc58ffSWalter Erquinigo #else 341*03cc58ffSWalter Erquinigo perf_event_attr attr; 342*03cc58ffSWalter Erquinigo memset(&attr, 0, sizeof(attr)); 343*03cc58ffSWalter Erquinigo attr.size = sizeof(attr); 344*03cc58ffSWalter Erquinigo attr.sample_type = PERF_SAMPLE_TID | PERF_SAMPLE_TIME; 345*03cc58ffSWalter Erquinigo attr.type = PERF_TYPE_SOFTWARE; 346*03cc58ffSWalter Erquinigo attr.context_switch = 1; 347*03cc58ffSWalter Erquinigo attr.exclude_kernel = 1; 348*03cc58ffSWalter Erquinigo attr.sample_id_all = 1; 349*03cc58ffSWalter Erquinigo attr.exclude_hv = 1; 350*03cc58ffSWalter Erquinigo attr.disabled = parent_perf_event ? !parent_perf_event->IsEnabled() : false; 351*03cc58ffSWalter Erquinigo 352*03cc58ffSWalter Erquinigo // The given perf configuration will produce context switch records of 32 353*03cc58ffSWalter Erquinigo // bytes each. Assuming that every context switch will be emitted twice (one 354*03cc58ffSWalter Erquinigo // for context switch ins and another one for context switch outs), and that a 355*03cc58ffSWalter Erquinigo // context switch will happen at least every half a millisecond per core, we 356*03cc58ffSWalter Erquinigo // need 500 * 32 bytes (~16 KB) for a trace of one second, which is much more 357*03cc58ffSWalter Erquinigo // than what a regular intel pt trace can get. Pessimistically we pick as 358*03cc58ffSWalter Erquinigo // 32KiB for the size of our context switch trace. 359*03cc58ffSWalter Erquinigo 360*03cc58ffSWalter Erquinigo uint64_t data_buffer_size = 32768; 361*03cc58ffSWalter Erquinigo uint64_t data_buffer_numpages = data_buffer_size / getpagesize(); 362*03cc58ffSWalter Erquinigo 363*03cc58ffSWalter Erquinigo LLDB_LOG(log, "Will create context switch trace buffer of size {0}", 364*03cc58ffSWalter Erquinigo data_buffer_size); 365*03cc58ffSWalter Erquinigo 366*03cc58ffSWalter Erquinigo Optional<long> group_fd; 367*03cc58ffSWalter Erquinigo if (parent_perf_event) 368*03cc58ffSWalter Erquinigo group_fd = parent_perf_event->GetFd(); 369*03cc58ffSWalter Erquinigo 370*03cc58ffSWalter Erquinigo if (Expected<PerfEvent> perf_event = 371*03cc58ffSWalter Erquinigo PerfEvent::Init(attr, /*pid=*/None, core_id, group_fd, /*flags=*/0)) { 372*03cc58ffSWalter Erquinigo if (Error mmap_err = perf_event->MmapMetadataAndBuffers( 373*03cc58ffSWalter Erquinigo data_buffer_numpages, 0, /*data_buffer_write=*/false)) { 374*03cc58ffSWalter Erquinigo return std::move(mmap_err); 375*03cc58ffSWalter Erquinigo } 376*03cc58ffSWalter Erquinigo return perf_event; 377*03cc58ffSWalter Erquinigo } else { 378*03cc58ffSWalter Erquinigo return perf_event.takeError(); 379*03cc58ffSWalter Erquinigo } 380*03cc58ffSWalter Erquinigo #endif 381*03cc58ffSWalter Erquinigo } 382