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(); 39*fc5ef57cSWalter Erquinigo if (Error mmap_err = 40*fc5ef57cSWalter Erquinigo perf_event->MmapMetadataAndBuffers(/*num_data_pages=*/0, 41*fc5ef57cSWalter Erquinigo /*num_aux_pages=*/0, 42*fc5ef57cSWalter 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, 771f49714dSWalter Erquinigo Optional<int> 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 } 87a7582059SWalter Erquinigo return PerfEvent(fd, attr.disabled ? CollectionState::Disabled 88a7582059SWalter Erquinigo : CollectionState::Enabled); 89e6c84f82SJakob Johnson } 90e6c84f82SJakob Johnson 91e6c84f82SJakob Johnson llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr, 921f49714dSWalter Erquinigo Optional<lldb::pid_t> pid, 931f49714dSWalter Erquinigo Optional<lldb::core_id_t> cpu) { 941f49714dSWalter Erquinigo return Init(attr, pid, cpu, -1, 0); 95e6c84f82SJakob Johnson } 96e6c84f82SJakob Johnson 97e6c84f82SJakob Johnson llvm::Expected<resource_handle::MmapUP> 98e6c84f82SJakob Johnson PerfEvent::DoMmap(void *addr, size_t length, int prot, int flags, 99e6c84f82SJakob Johnson long int offset, llvm::StringRef buffer_name) { 100e6c84f82SJakob Johnson errno = 0; 101a7582059SWalter Erquinigo auto mmap_result = ::mmap(addr, length, prot, flags, GetFd(), offset); 102e6c84f82SJakob Johnson 103e6c84f82SJakob Johnson if (mmap_result == MAP_FAILED) { 104e6c84f82SJakob Johnson std::string err_msg = 105e6c84f82SJakob Johnson llvm::formatv("perf event mmap allocation failed for {0}: {1}", 106e6c84f82SJakob Johnson buffer_name, std::strerror(errno)); 107e6c84f82SJakob Johnson return createStringError(inconvertibleErrorCode(), err_msg); 108e6c84f82SJakob Johnson } 109e6c84f82SJakob Johnson return resource_handle::MmapUP(mmap_result, length); 110e6c84f82SJakob Johnson } 111e6c84f82SJakob Johnson 112*fc5ef57cSWalter Erquinigo llvm::Error PerfEvent::MmapMetadataAndDataBuffer(size_t num_data_pages, 113*fc5ef57cSWalter Erquinigo bool data_buffer_write) { 114e6c84f82SJakob Johnson size_t mmap_size = (num_data_pages + 1) * getpagesize(); 115*fc5ef57cSWalter Erquinigo if (Expected<resource_handle::MmapUP> mmap_metadata_data = DoMmap( 116*fc5ef57cSWalter Erquinigo nullptr, mmap_size, PROT_READ | (data_buffer_write ? PROT_WRITE : 0), 117*fc5ef57cSWalter Erquinigo MAP_SHARED, 0, "metadata and data buffer")) { 118e6c84f82SJakob Johnson m_metadata_data_base = std::move(mmap_metadata_data.get()); 119e6c84f82SJakob Johnson return Error::success(); 120e6c84f82SJakob Johnson } else 121e6c84f82SJakob Johnson return mmap_metadata_data.takeError(); 122e6c84f82SJakob Johnson } 123e6c84f82SJakob Johnson 124e6c84f82SJakob Johnson llvm::Error PerfEvent::MmapAuxBuffer(size_t num_aux_pages) { 125e6c84f82SJakob Johnson if (num_aux_pages == 0) 126e6c84f82SJakob Johnson return Error::success(); 127e6c84f82SJakob Johnson 128e6c84f82SJakob Johnson perf_event_mmap_page &metadata_page = GetMetadataPage(); 129*fc5ef57cSWalter Erquinigo 130e6c84f82SJakob Johnson metadata_page.aux_offset = 131e6c84f82SJakob Johnson metadata_page.data_offset + metadata_page.data_size; 132e6c84f82SJakob Johnson metadata_page.aux_size = num_aux_pages * getpagesize(); 133e6c84f82SJakob Johnson 134e6c84f82SJakob Johnson if (Expected<resource_handle::MmapUP> mmap_aux = 135e6c84f82SJakob Johnson DoMmap(nullptr, metadata_page.aux_size, PROT_READ, MAP_SHARED, 136e6c84f82SJakob Johnson metadata_page.aux_offset, "aux buffer")) { 137e6c84f82SJakob Johnson m_aux_base = std::move(mmap_aux.get()); 138e6c84f82SJakob Johnson return Error::success(); 139e6c84f82SJakob Johnson } else 140e6c84f82SJakob Johnson return mmap_aux.takeError(); 141e6c84f82SJakob Johnson } 142e6c84f82SJakob Johnson 143e6c84f82SJakob Johnson llvm::Error PerfEvent::MmapMetadataAndBuffers(size_t num_data_pages, 144*fc5ef57cSWalter Erquinigo size_t num_aux_pages, 145*fc5ef57cSWalter Erquinigo bool data_buffer_write) { 146e6c84f82SJakob Johnson if (num_data_pages != 0 && !isPowerOf2_64(num_data_pages)) 147e6c84f82SJakob Johnson return llvm::createStringError( 148e6c84f82SJakob Johnson llvm::inconvertibleErrorCode(), 149e6c84f82SJakob Johnson llvm::formatv("Number of data pages must be a power of 2, got: {0}", 150e6c84f82SJakob Johnson num_data_pages)); 151e6c84f82SJakob Johnson if (num_aux_pages != 0 && !isPowerOf2_64(num_aux_pages)) 152e6c84f82SJakob Johnson return llvm::createStringError( 153e6c84f82SJakob Johnson llvm::inconvertibleErrorCode(), 154e6c84f82SJakob Johnson llvm::formatv("Number of aux pages must be a power of 2, got: {0}", 155e6c84f82SJakob Johnson num_aux_pages)); 156*fc5ef57cSWalter Erquinigo if (Error err = MmapMetadataAndDataBuffer(num_data_pages, data_buffer_write)) 157e6c84f82SJakob Johnson return err; 158e6c84f82SJakob Johnson if (Error err = MmapAuxBuffer(num_aux_pages)) 159e6c84f82SJakob Johnson return err; 160e6c84f82SJakob Johnson return Error::success(); 161e6c84f82SJakob Johnson } 162e6c84f82SJakob Johnson 163e6c84f82SJakob Johnson long PerfEvent::GetFd() const { return *(m_fd.get()); } 164e6c84f82SJakob Johnson 165e6c84f82SJakob Johnson perf_event_mmap_page &PerfEvent::GetMetadataPage() const { 166e6c84f82SJakob Johnson return *reinterpret_cast<perf_event_mmap_page *>(m_metadata_data_base.get()); 167e6c84f82SJakob Johnson } 168e6c84f82SJakob Johnson 169e6c84f82SJakob Johnson ArrayRef<uint8_t> PerfEvent::GetDataBuffer() const { 170e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 171e6c84f82SJakob Johnson return {reinterpret_cast<uint8_t *>(m_metadata_data_base.get()) + 172e6c84f82SJakob Johnson mmap_metadata.data_offset, 173e412529cSJakob Johnson static_cast<size_t>(mmap_metadata.data_size)}; 174e6c84f82SJakob Johnson } 175e6c84f82SJakob Johnson 176e6c84f82SJakob Johnson ArrayRef<uint8_t> PerfEvent::GetAuxBuffer() const { 177e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 178e6c84f82SJakob Johnson return {reinterpret_cast<uint8_t *>(m_aux_base.get()), 179e412529cSJakob Johnson static_cast<size_t>(mmap_metadata.aux_size)}; 180e6c84f82SJakob Johnson } 1811637545fSWalter Erquinigo 182a7582059SWalter Erquinigo Expected<std::vector<uint8_t>> 183*fc5ef57cSWalter Erquinigo PerfEvent::ReadFlushedOutDataCyclicBuffer(size_t offset, size_t size) { 184*fc5ef57cSWalter Erquinigo CollectionState previous_state = m_collection_state; 185*fc5ef57cSWalter Erquinigo if (Error err = DisableWithIoctl()) 186*fc5ef57cSWalter Erquinigo return std::move(err); 187*fc5ef57cSWalter Erquinigo 188*fc5ef57cSWalter Erquinigo /** 189*fc5ef57cSWalter Erquinigo * The data buffer and aux buffer have different implementations 190*fc5ef57cSWalter Erquinigo * with respect to their definition of head pointer. In the case 191*fc5ef57cSWalter Erquinigo * of Aux data buffer the head always wraps around the aux buffer 192*fc5ef57cSWalter Erquinigo * and we don't need to care about it, whereas the data_head keeps 193*fc5ef57cSWalter Erquinigo * increasing and needs to be wrapped by modulus operator 194*fc5ef57cSWalter Erquinigo */ 195*fc5ef57cSWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 196*fc5ef57cSWalter Erquinigo 197*fc5ef57cSWalter Erquinigo ArrayRef<uint8_t> data = GetDataBuffer(); 198*fc5ef57cSWalter Erquinigo uint64_t data_head = mmap_metadata.data_head; 199*fc5ef57cSWalter Erquinigo uint64_t data_size = mmap_metadata.data_size; 200*fc5ef57cSWalter Erquinigo std::vector<uint8_t> output; 201*fc5ef57cSWalter Erquinigo output.reserve(size); 202*fc5ef57cSWalter Erquinigo 203*fc5ef57cSWalter Erquinigo if (data_head > data_size) { 204*fc5ef57cSWalter Erquinigo uint64_t actual_data_head = data_head % data_size; 205*fc5ef57cSWalter Erquinigo // The buffer has wrapped 206*fc5ef57cSWalter Erquinigo for (uint64_t i = actual_data_head + offset; 207*fc5ef57cSWalter Erquinigo i < data_size && output.size() < size; i++) 208*fc5ef57cSWalter Erquinigo output.push_back(data[i]); 209*fc5ef57cSWalter Erquinigo 210*fc5ef57cSWalter Erquinigo // We need to find the starting position for the left part if the offset was 211*fc5ef57cSWalter Erquinigo // too big 212*fc5ef57cSWalter Erquinigo uint64_t left_part_start = actual_data_head + offset >= data_size 213*fc5ef57cSWalter Erquinigo ? actual_data_head + offset - data_size 214*fc5ef57cSWalter Erquinigo : 0; 215*fc5ef57cSWalter Erquinigo for (uint64_t i = left_part_start; 216*fc5ef57cSWalter Erquinigo i < actual_data_head && output.size() < size; i++) 217*fc5ef57cSWalter Erquinigo output.push_back(data[i]); 218*fc5ef57cSWalter Erquinigo } else { 219*fc5ef57cSWalter Erquinigo for (uint64_t i = offset; i < data_head && output.size() < size; i++) 220*fc5ef57cSWalter Erquinigo output.push_back(data[i]); 221*fc5ef57cSWalter Erquinigo } 222*fc5ef57cSWalter Erquinigo 223*fc5ef57cSWalter Erquinigo if (previous_state == CollectionState::Enabled) { 224*fc5ef57cSWalter Erquinigo if (Error err = EnableWithIoctl()) 225*fc5ef57cSWalter Erquinigo return std::move(err); 226*fc5ef57cSWalter Erquinigo } 227*fc5ef57cSWalter Erquinigo 228*fc5ef57cSWalter Erquinigo if (output.size() != size) 229*fc5ef57cSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 230*fc5ef57cSWalter Erquinigo formatv("Requested {0} bytes of perf_event data " 231*fc5ef57cSWalter Erquinigo "buffer but only {1} are available", 232*fc5ef57cSWalter Erquinigo size, output.size())); 233*fc5ef57cSWalter Erquinigo 234*fc5ef57cSWalter Erquinigo return data; 235*fc5ef57cSWalter Erquinigo } 236*fc5ef57cSWalter Erquinigo 237*fc5ef57cSWalter Erquinigo Expected<std::vector<uint8_t>> 238a7582059SWalter Erquinigo PerfEvent::ReadFlushedOutAuxCyclicBuffer(size_t offset, size_t size) { 239a7582059SWalter Erquinigo CollectionState previous_state = m_collection_state; 240a7582059SWalter Erquinigo if (Error err = DisableWithIoctl()) 241a7582059SWalter Erquinigo return std::move(err); 242a7582059SWalter Erquinigo 243a7582059SWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 244a7582059SWalter Erquinigo 245*fc5ef57cSWalter Erquinigo ArrayRef<uint8_t> data = GetAuxBuffer(); 246*fc5ef57cSWalter Erquinigo uint64_t aux_head = mmap_metadata.aux_head; 247*fc5ef57cSWalter Erquinigo uint64_t aux_size = mmap_metadata.aux_size; 248*fc5ef57cSWalter Erquinigo std::vector<uint8_t> output; 249*fc5ef57cSWalter Erquinigo output.reserve(size); 250a7582059SWalter Erquinigo 251a7582059SWalter Erquinigo /** 252a7582059SWalter Erquinigo * When configured as ring buffer, the aux buffer keeps wrapping around 253a7582059SWalter Erquinigo * the buffer and its not possible to detect how many times the buffer 254a7582059SWalter Erquinigo * wrapped. Initially the buffer is filled with zeros,as shown below 255a7582059SWalter Erquinigo * so in order to get complete buffer we first copy firstpartsize, followed 256a7582059SWalter Erquinigo * by any left over part from beginning to aux_head 257a7582059SWalter Erquinigo * 258a7582059SWalter Erquinigo * aux_offset [d,d,d,d,d,d,d,d,0,0,0,0,0,0,0,0,0,0,0] aux_size 259a7582059SWalter Erquinigo * aux_head->||<- firstpartsize ->| 260a7582059SWalter Erquinigo * 261a7582059SWalter Erquinigo * */ 262a7582059SWalter Erquinigo 263*fc5ef57cSWalter Erquinigo for (uint64_t i = aux_head + offset; i < aux_size && output.size() < size; 264*fc5ef57cSWalter Erquinigo i++) 265*fc5ef57cSWalter Erquinigo output.push_back(data[i]); 266*fc5ef57cSWalter Erquinigo 267*fc5ef57cSWalter Erquinigo // We need to find the starting position for the left part if the offset was 268*fc5ef57cSWalter Erquinigo // too big 269*fc5ef57cSWalter Erquinigo uint64_t left_part_start = 270*fc5ef57cSWalter Erquinigo aux_head + offset >= aux_size ? aux_head + offset - aux_size : 0; 271*fc5ef57cSWalter Erquinigo for (uint64_t i = left_part_start; i < aux_head && output.size() < size; i++) 272*fc5ef57cSWalter Erquinigo output.push_back(data[i]); 273a7582059SWalter Erquinigo 274a7582059SWalter Erquinigo if (previous_state == CollectionState::Enabled) { 275a7582059SWalter Erquinigo if (Error err = EnableWithIoctl()) 276a7582059SWalter Erquinigo return std::move(err); 277a7582059SWalter Erquinigo } 278a7582059SWalter Erquinigo 279*fc5ef57cSWalter Erquinigo if (output.size() != size) 280*fc5ef57cSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 281*fc5ef57cSWalter Erquinigo formatv("Requested {0} bytes of perf_event aux " 282*fc5ef57cSWalter Erquinigo "buffer but only {1} are available", 283*fc5ef57cSWalter Erquinigo size, output.size())); 284*fc5ef57cSWalter Erquinigo 285a7582059SWalter Erquinigo return data; 286a7582059SWalter Erquinigo } 287a7582059SWalter Erquinigo 288a7582059SWalter Erquinigo Error PerfEvent::DisableWithIoctl() { 289a7582059SWalter Erquinigo if (m_collection_state == CollectionState::Disabled) 290a7582059SWalter Erquinigo return Error::success(); 291a7582059SWalter Erquinigo 292a7582059SWalter Erquinigo if (ioctl(*m_fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) < 0) 2931637545fSWalter Erquinigo return createStringError(inconvertibleErrorCode(), 2941637545fSWalter Erquinigo "Can't disable perf event. %s", 2951637545fSWalter Erquinigo std::strerror(errno)); 296a7582059SWalter Erquinigo 297a7582059SWalter Erquinigo m_collection_state = CollectionState::Disabled; 2981637545fSWalter Erquinigo return Error::success(); 2991637545fSWalter Erquinigo } 3001637545fSWalter Erquinigo 301a7582059SWalter Erquinigo Error PerfEvent::EnableWithIoctl() { 302a7582059SWalter Erquinigo if (m_collection_state == CollectionState::Enabled) 3031637545fSWalter Erquinigo return Error::success(); 304a7582059SWalter Erquinigo 305a7582059SWalter Erquinigo if (ioctl(*m_fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) < 0) 306a7582059SWalter Erquinigo return createStringError(inconvertibleErrorCode(), 307a7582059SWalter Erquinigo "Can't enable perf event. %s", 308a7582059SWalter Erquinigo std::strerror(errno)); 309a7582059SWalter Erquinigo 310a7582059SWalter Erquinigo m_collection_state = CollectionState::Enabled; 311a7582059SWalter Erquinigo return Error::success(); 312a7582059SWalter Erquinigo } 313a7582059SWalter Erquinigo 314a7582059SWalter Erquinigo size_t PerfEvent::GetEffectiveDataBufferSize() const { 315a7582059SWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage(); 316*fc5ef57cSWalter Erquinigo if (mmap_metadata.data_head <= mmap_metadata.data_size) 317a7582059SWalter Erquinigo return mmap_metadata.data_head; 318a7582059SWalter Erquinigo else 319a7582059SWalter Erquinigo return mmap_metadata.data_size; // The buffer has wrapped. 3201637545fSWalter Erquinigo } 321