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
18b83b82f9SYi Kong #include <linux/version.h>
191637545fSWalter Erquinigo #include <sys/ioctl.h>
20e6c84f82SJakob Johnson #include <sys/mman.h>
21e6c84f82SJakob Johnson #include <sys/syscall.h>
22e6c84f82SJakob Johnson #include <unistd.h>
23e6c84f82SJakob Johnson
24e6c84f82SJakob Johnson using namespace lldb_private;
25e6c84f82SJakob Johnson using namespace process_linux;
26e6c84f82SJakob Johnson using namespace llvm;
27e6c84f82SJakob Johnson
289b79187cSJakob Johnson Expected<LinuxPerfZeroTscConversion>
LoadPerfTscConversionParameters()299b79187cSJakob Johnson lldb_private::process_linux::LoadPerfTscConversionParameters() {
30b83b82f9SYi Kong #if LINUX_VERSION_CODE >= KERNEL_VERSION(3,12,0)
31e6c84f82SJakob Johnson lldb::pid_t pid = getpid();
32e6c84f82SJakob Johnson perf_event_attr attr;
33e6c84f82SJakob Johnson memset(&attr, 0, sizeof(attr));
34e6c84f82SJakob Johnson attr.size = sizeof(attr);
35e6c84f82SJakob Johnson attr.type = PERF_TYPE_SOFTWARE;
36e6c84f82SJakob Johnson attr.config = PERF_COUNT_SW_DUMMY;
37e6c84f82SJakob Johnson
38e6c84f82SJakob Johnson Expected<PerfEvent> perf_event = PerfEvent::Init(attr, pid);
39e6c84f82SJakob Johnson if (!perf_event)
40e6c84f82SJakob Johnson return perf_event.takeError();
41fc5ef57cSWalter Erquinigo if (Error mmap_err =
42fc5ef57cSWalter Erquinigo perf_event->MmapMetadataAndBuffers(/*num_data_pages=*/0,
43fc5ef57cSWalter Erquinigo /*num_aux_pages=*/0,
44fc5ef57cSWalter Erquinigo /*data_buffer_write=*/false))
45e6c84f82SJakob Johnson return std::move(mmap_err);
46e6c84f82SJakob Johnson
47e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metada = perf_event->GetMetadataPage();
48e6c84f82SJakob Johnson if (mmap_metada.cap_user_time && mmap_metada.cap_user_time_zero) {
499b79187cSJakob Johnson return LinuxPerfZeroTscConversion{
509f45f23dSWalter Erquinigo mmap_metada.time_mult, mmap_metada.time_shift, {mmap_metada.time_zero}};
51e6c84f82SJakob Johnson } else {
52e6c84f82SJakob Johnson auto err_cap =
53e6c84f82SJakob Johnson !mmap_metada.cap_user_time ? "cap_user_time" : "cap_user_time_zero";
54e6c84f82SJakob Johnson std::string err_msg =
55e6c84f82SJakob Johnson llvm::formatv("Can't get TSC to real time conversion values. "
56e6c84f82SJakob Johnson "perf_event capability '{0}' not supported.",
57e6c84f82SJakob Johnson err_cap);
58e6c84f82SJakob Johnson return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg);
59e6c84f82SJakob Johnson }
60b83b82f9SYi Kong #else
61b83b82f9SYi Kong std::string err_msg = "PERF_COUNT_SW_DUMMY requires Linux 3.12";
62b83b82f9SYi Kong return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg);
63b83b82f9SYi Kong #endif
64e6c84f82SJakob Johnson }
65e6c84f82SJakob Johnson
operator ()(void * ptr)66e6c84f82SJakob Johnson void resource_handle::MmapDeleter::operator()(void *ptr) {
67e6c84f82SJakob Johnson if (m_bytes && ptr != nullptr)
68e6c84f82SJakob Johnson munmap(ptr, m_bytes);
69e6c84f82SJakob Johnson }
70e6c84f82SJakob Johnson
operator ()(long * ptr)71e6c84f82SJakob Johnson void resource_handle::FileDescriptorDeleter::operator()(long *ptr) {
72e6c84f82SJakob Johnson if (ptr == nullptr)
73e6c84f82SJakob Johnson return;
74e6c84f82SJakob Johnson if (*ptr == -1)
75e6c84f82SJakob Johnson return;
76e6c84f82SJakob Johnson close(*ptr);
77e6c84f82SJakob Johnson std::default_delete<long>()(ptr);
78e6c84f82SJakob Johnson }
79e6c84f82SJakob Johnson
Init(perf_event_attr & attr,Optional<lldb::pid_t> pid,Optional<lldb::cpu_id_t> cpu,Optional<long> group_fd,unsigned long flags)80e6c84f82SJakob Johnson llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr,
811f49714dSWalter Erquinigo Optional<lldb::pid_t> pid,
826a5355e8SWalter Erquinigo Optional<lldb::cpu_id_t> cpu,
8303cc58ffSWalter Erquinigo Optional<long> group_fd,
841f49714dSWalter Erquinigo unsigned long flags) {
85e6c84f82SJakob Johnson errno = 0;
86aa88161bSKazu Hirata long fd = syscall(SYS_perf_event_open, &attr, pid.value_or(-1),
87aa88161bSKazu Hirata cpu.value_or(-1), group_fd.value_or(-1), flags);
88e6c84f82SJakob Johnson if (fd == -1) {
89e6c84f82SJakob Johnson std::string err_msg =
90e6c84f82SJakob Johnson llvm::formatv("perf event syscall failed: {0}", std::strerror(errno));
91e6c84f82SJakob Johnson return llvm::createStringError(llvm::inconvertibleErrorCode(), err_msg);
92e6c84f82SJakob Johnson }
9303cc58ffSWalter Erquinigo return PerfEvent(fd, !attr.disabled);
94e6c84f82SJakob Johnson }
95e6c84f82SJakob Johnson
Init(perf_event_attr & attr,Optional<lldb::pid_t> pid,Optional<lldb::cpu_id_t> cpu)96e6c84f82SJakob Johnson llvm::Expected<PerfEvent> PerfEvent::Init(perf_event_attr &attr,
971f49714dSWalter Erquinigo Optional<lldb::pid_t> pid,
986a5355e8SWalter Erquinigo Optional<lldb::cpu_id_t> cpu) {
991f49714dSWalter Erquinigo return Init(attr, pid, cpu, -1, 0);
100e6c84f82SJakob Johnson }
101e6c84f82SJakob Johnson
102e6c84f82SJakob Johnson llvm::Expected<resource_handle::MmapUP>
DoMmap(void * addr,size_t length,int prot,int flags,long int offset,llvm::StringRef buffer_name)103e6c84f82SJakob Johnson PerfEvent::DoMmap(void *addr, size_t length, int prot, int flags,
104e6c84f82SJakob Johnson long int offset, llvm::StringRef buffer_name) {
105e6c84f82SJakob Johnson errno = 0;
106a7582059SWalter Erquinigo auto mmap_result = ::mmap(addr, length, prot, flags, GetFd(), offset);
107e6c84f82SJakob Johnson
108e6c84f82SJakob Johnson if (mmap_result == MAP_FAILED) {
109e6c84f82SJakob Johnson std::string err_msg =
110e6c84f82SJakob Johnson llvm::formatv("perf event mmap allocation failed for {0}: {1}",
111e6c84f82SJakob Johnson buffer_name, std::strerror(errno));
112e6c84f82SJakob Johnson return createStringError(inconvertibleErrorCode(), err_msg);
113e6c84f82SJakob Johnson }
114e6c84f82SJakob Johnson return resource_handle::MmapUP(mmap_result, length);
115e6c84f82SJakob Johnson }
116e6c84f82SJakob Johnson
MmapMetadataAndDataBuffer(size_t num_data_pages,bool data_buffer_write)117fc5ef57cSWalter Erquinigo llvm::Error PerfEvent::MmapMetadataAndDataBuffer(size_t num_data_pages,
118fc5ef57cSWalter Erquinigo bool data_buffer_write) {
119e6c84f82SJakob Johnson size_t mmap_size = (num_data_pages + 1) * getpagesize();
120fc5ef57cSWalter Erquinigo if (Expected<resource_handle::MmapUP> mmap_metadata_data = DoMmap(
121fc5ef57cSWalter Erquinigo nullptr, mmap_size, PROT_READ | (data_buffer_write ? PROT_WRITE : 0),
122fc5ef57cSWalter Erquinigo MAP_SHARED, 0, "metadata and data buffer")) {
123e6c84f82SJakob Johnson m_metadata_data_base = std::move(mmap_metadata_data.get());
124e6c84f82SJakob Johnson return Error::success();
125e6c84f82SJakob Johnson } else
126e6c84f82SJakob Johnson return mmap_metadata_data.takeError();
127e6c84f82SJakob Johnson }
128e6c84f82SJakob Johnson
MmapAuxBuffer(size_t num_aux_pages)129e6c84f82SJakob Johnson llvm::Error PerfEvent::MmapAuxBuffer(size_t num_aux_pages) {
130*6599b6e5SDavid Spickett #ifndef PERF_ATTR_SIZE_VER5
131*6599b6e5SDavid Spickett return createStringError(inconvertibleErrorCode(),
132*6599b6e5SDavid Spickett "Intel PT Linux perf event not supported");
133*6599b6e5SDavid Spickett #else
134e6c84f82SJakob Johnson if (num_aux_pages == 0)
135e6c84f82SJakob Johnson return Error::success();
136e6c84f82SJakob Johnson
137e6c84f82SJakob Johnson perf_event_mmap_page &metadata_page = GetMetadataPage();
138fc5ef57cSWalter Erquinigo
139e6c84f82SJakob Johnson metadata_page.aux_offset =
140e6c84f82SJakob Johnson metadata_page.data_offset + metadata_page.data_size;
141e6c84f82SJakob Johnson metadata_page.aux_size = num_aux_pages * getpagesize();
142e6c84f82SJakob Johnson
143e6c84f82SJakob Johnson if (Expected<resource_handle::MmapUP> mmap_aux =
144e6c84f82SJakob Johnson DoMmap(nullptr, metadata_page.aux_size, PROT_READ, MAP_SHARED,
145e6c84f82SJakob Johnson metadata_page.aux_offset, "aux buffer")) {
146e6c84f82SJakob Johnson m_aux_base = std::move(mmap_aux.get());
147e6c84f82SJakob Johnson return Error::success();
148e6c84f82SJakob Johnson } else
149e6c84f82SJakob Johnson return mmap_aux.takeError();
150*6599b6e5SDavid Spickett #endif
151e6c84f82SJakob Johnson }
152e6c84f82SJakob Johnson
MmapMetadataAndBuffers(size_t num_data_pages,size_t num_aux_pages,bool data_buffer_write)153e6c84f82SJakob Johnson llvm::Error PerfEvent::MmapMetadataAndBuffers(size_t num_data_pages,
154fc5ef57cSWalter Erquinigo size_t num_aux_pages,
155fc5ef57cSWalter Erquinigo bool data_buffer_write) {
156e6c84f82SJakob Johnson if (num_data_pages != 0 && !isPowerOf2_64(num_data_pages))
157e6c84f82SJakob Johnson return llvm::createStringError(
158e6c84f82SJakob Johnson llvm::inconvertibleErrorCode(),
159e6c84f82SJakob Johnson llvm::formatv("Number of data pages must be a power of 2, got: {0}",
160e6c84f82SJakob Johnson num_data_pages));
161e6c84f82SJakob Johnson if (num_aux_pages != 0 && !isPowerOf2_64(num_aux_pages))
162e6c84f82SJakob Johnson return llvm::createStringError(
163e6c84f82SJakob Johnson llvm::inconvertibleErrorCode(),
164e6c84f82SJakob Johnson llvm::formatv("Number of aux pages must be a power of 2, got: {0}",
165e6c84f82SJakob Johnson num_aux_pages));
166fc5ef57cSWalter Erquinigo if (Error err = MmapMetadataAndDataBuffer(num_data_pages, data_buffer_write))
167e6c84f82SJakob Johnson return err;
168e6c84f82SJakob Johnson if (Error err = MmapAuxBuffer(num_aux_pages))
169e6c84f82SJakob Johnson return err;
170e6c84f82SJakob Johnson return Error::success();
171e6c84f82SJakob Johnson }
172e6c84f82SJakob Johnson
GetFd() const173e6c84f82SJakob Johnson long PerfEvent::GetFd() const { return *(m_fd.get()); }
174e6c84f82SJakob Johnson
GetMetadataPage() const175e6c84f82SJakob Johnson perf_event_mmap_page &PerfEvent::GetMetadataPage() const {
176e6c84f82SJakob Johnson return *reinterpret_cast<perf_event_mmap_page *>(m_metadata_data_base.get());
177e6c84f82SJakob Johnson }
178e6c84f82SJakob Johnson
GetDataBuffer() const179e6c84f82SJakob Johnson ArrayRef<uint8_t> PerfEvent::GetDataBuffer() const {
180*6599b6e5SDavid Spickett #ifndef PERF_ATTR_SIZE_VER5
181*6599b6e5SDavid Spickett llvm_unreachable("Intel PT Linux perf event not supported");
182*6599b6e5SDavid Spickett #else
183e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metadata = GetMetadataPage();
184e6c84f82SJakob Johnson return {reinterpret_cast<uint8_t *>(m_metadata_data_base.get()) +
185e6c84f82SJakob Johnson mmap_metadata.data_offset,
186e412529cSJakob Johnson static_cast<size_t>(mmap_metadata.data_size)};
187*6599b6e5SDavid Spickett #endif
188e6c84f82SJakob Johnson }
189e6c84f82SJakob Johnson
GetAuxBuffer() const190e6c84f82SJakob Johnson ArrayRef<uint8_t> PerfEvent::GetAuxBuffer() const {
191*6599b6e5SDavid Spickett #ifndef PERF_ATTR_SIZE_VER5
192*6599b6e5SDavid Spickett llvm_unreachable("Intel PT Linux perf event not supported");
193*6599b6e5SDavid Spickett #else
194e6c84f82SJakob Johnson perf_event_mmap_page &mmap_metadata = GetMetadataPage();
195e6c84f82SJakob Johnson return {reinterpret_cast<uint8_t *>(m_aux_base.get()),
196e412529cSJakob Johnson static_cast<size_t>(mmap_metadata.aux_size)};
197*6599b6e5SDavid Spickett #endif
198e6c84f82SJakob Johnson }
1991637545fSWalter Erquinigo
GetReadOnlyDataBuffer()200561a61fbSWalter Erquinigo Expected<std::vector<uint8_t>> PerfEvent::GetReadOnlyDataBuffer() {
20103cc58ffSWalter Erquinigo // The following code assumes that the protection level of the DATA page
20203cc58ffSWalter Erquinigo // is PROT_READ. If PROT_WRITE is used, then reading would require that
20303cc58ffSWalter Erquinigo // this piece of code updates some pointers. See more about data_tail
20403cc58ffSWalter Erquinigo // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html.
20503cc58ffSWalter Erquinigo
206*6599b6e5SDavid Spickett #ifndef PERF_ATTR_SIZE_VER5
207*6599b6e5SDavid Spickett return createStringError(inconvertibleErrorCode(),
208*6599b6e5SDavid Spickett "Intel PT Linux perf event not supported");
209*6599b6e5SDavid Spickett #else
21003cc58ffSWalter Erquinigo bool was_enabled = m_enabled;
211fc5ef57cSWalter Erquinigo if (Error err = DisableWithIoctl())
212fc5ef57cSWalter Erquinigo return std::move(err);
213fc5ef57cSWalter Erquinigo
214fc5ef57cSWalter Erquinigo /**
215fc5ef57cSWalter Erquinigo * The data buffer and aux buffer have different implementations
216561a61fbSWalter Erquinigo * with respect to their definition of head pointer when using PROD_READ only.
217561a61fbSWalter Erquinigo * In the case of Aux data buffer the head always wraps around the aux buffer
218fc5ef57cSWalter Erquinigo * and we don't need to care about it, whereas the data_head keeps
219fc5ef57cSWalter Erquinigo * increasing and needs to be wrapped by modulus operator
220fc5ef57cSWalter Erquinigo */
221fc5ef57cSWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage();
222fc5ef57cSWalter Erquinigo
223fc5ef57cSWalter Erquinigo ArrayRef<uint8_t> data = GetDataBuffer();
224fc5ef57cSWalter Erquinigo uint64_t data_head = mmap_metadata.data_head;
225fc5ef57cSWalter Erquinigo uint64_t data_size = mmap_metadata.data_size;
226fc5ef57cSWalter Erquinigo std::vector<uint8_t> output;
227561a61fbSWalter Erquinigo output.reserve(data.size());
228fc5ef57cSWalter Erquinigo
229fc5ef57cSWalter Erquinigo if (data_head > data_size) {
230fc5ef57cSWalter Erquinigo uint64_t actual_data_head = data_head % data_size;
231ea37cd52SWalter Erquinigo // The buffer has wrapped, so we first the oldest chunk of data
232ea37cd52SWalter Erquinigo output.insert(output.end(), data.begin() + actual_data_head, data.end());
233ea37cd52SWalter Erquinigo // And we we read the most recent chunk of data
234ea37cd52SWalter Erquinigo output.insert(output.end(), data.begin(), data.begin() + actual_data_head);
235fc5ef57cSWalter Erquinigo } else {
236ea37cd52SWalter Erquinigo // There's been no wrapping, so we just read linearly
237ea37cd52SWalter Erquinigo output.insert(output.end(), data.begin(), data.begin() + data_head);
238fc5ef57cSWalter Erquinigo }
239fc5ef57cSWalter Erquinigo
24003cc58ffSWalter Erquinigo if (was_enabled) {
241fc5ef57cSWalter Erquinigo if (Error err = EnableWithIoctl())
242fc5ef57cSWalter Erquinigo return std::move(err);
243fc5ef57cSWalter Erquinigo }
244fc5ef57cSWalter Erquinigo
2451a3f9969SWalter Erquinigo return output;
246*6599b6e5SDavid Spickett #endif
247fc5ef57cSWalter Erquinigo }
248fc5ef57cSWalter Erquinigo
GetReadOnlyAuxBuffer()249561a61fbSWalter Erquinigo Expected<std::vector<uint8_t>> PerfEvent::GetReadOnlyAuxBuffer() {
25003cc58ffSWalter Erquinigo // The following code assumes that the protection level of the AUX page
25103cc58ffSWalter Erquinigo // is PROT_READ. If PROT_WRITE is used, then reading would require that
25203cc58ffSWalter Erquinigo // this piece of code updates some pointers. See more about aux_tail
25303cc58ffSWalter Erquinigo // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html.
25403cc58ffSWalter Erquinigo
255*6599b6e5SDavid Spickett #ifndef PERF_ATTR_SIZE_VER5
256*6599b6e5SDavid Spickett return createStringError(inconvertibleErrorCode(),
257*6599b6e5SDavid Spickett "Intel PT Linux perf event not supported");
258*6599b6e5SDavid Spickett #else
25903cc58ffSWalter Erquinigo bool was_enabled = m_enabled;
260a7582059SWalter Erquinigo if (Error err = DisableWithIoctl())
261a7582059SWalter Erquinigo return std::move(err);
262a7582059SWalter Erquinigo
263a7582059SWalter Erquinigo perf_event_mmap_page &mmap_metadata = GetMetadataPage();
264a7582059SWalter Erquinigo
265fc5ef57cSWalter Erquinigo ArrayRef<uint8_t> data = GetAuxBuffer();
266fc5ef57cSWalter Erquinigo uint64_t aux_head = mmap_metadata.aux_head;
267fc5ef57cSWalter Erquinigo std::vector<uint8_t> output;
268561a61fbSWalter Erquinigo output.reserve(data.size());
269a7582059SWalter Erquinigo
270a7582059SWalter Erquinigo /**
271a7582059SWalter Erquinigo * When configured as ring buffer, the aux buffer keeps wrapping around
272a7582059SWalter Erquinigo * the buffer and its not possible to detect how many times the buffer
273a7582059SWalter Erquinigo * wrapped. Initially the buffer is filled with zeros,as shown below
274a7582059SWalter Erquinigo * so in order to get complete buffer we first copy firstpartsize, followed
275a7582059SWalter Erquinigo * by any left over part from beginning to aux_head
276a7582059SWalter Erquinigo *
277a7582059SWalter Erquinigo * aux_offset [d,d,d,d,d,d,d,d,0,0,0,0,0,0,0,0,0,0,0] aux_size
278a7582059SWalter Erquinigo * aux_head->||<- firstpartsize ->|
279a7582059SWalter Erquinigo *
280a7582059SWalter Erquinigo * */
281a7582059SWalter Erquinigo
282ea37cd52SWalter Erquinigo output.insert(output.end(), data.begin() + aux_head, data.end());
283ea37cd52SWalter Erquinigo output.insert(output.end(), data.begin(), data.begin() + aux_head);
284a7582059SWalter Erquinigo
28503cc58ffSWalter Erquinigo if (was_enabled) {
286a7582059SWalter Erquinigo if (Error err = EnableWithIoctl())
287a7582059SWalter Erquinigo return std::move(err);
288a7582059SWalter Erquinigo }
289a7582059SWalter Erquinigo
2901a3f9969SWalter Erquinigo return output;
291*6599b6e5SDavid Spickett #endif
292a7582059SWalter Erquinigo }
293a7582059SWalter Erquinigo
DisableWithIoctl()294a7582059SWalter Erquinigo Error PerfEvent::DisableWithIoctl() {
29503cc58ffSWalter Erquinigo if (!m_enabled)
296a7582059SWalter Erquinigo return Error::success();
297a7582059SWalter Erquinigo
298a7582059SWalter Erquinigo if (ioctl(*m_fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) < 0)
2991637545fSWalter Erquinigo return createStringError(inconvertibleErrorCode(),
3001637545fSWalter Erquinigo "Can't disable perf event. %s",
3011637545fSWalter Erquinigo std::strerror(errno));
302a7582059SWalter Erquinigo
30303cc58ffSWalter Erquinigo m_enabled = false;
3041637545fSWalter Erquinigo return Error::success();
3051637545fSWalter Erquinigo }
3061637545fSWalter Erquinigo
IsEnabled() const30703cc58ffSWalter Erquinigo bool PerfEvent::IsEnabled() const { return m_enabled; }
30803cc58ffSWalter Erquinigo
EnableWithIoctl()309a7582059SWalter Erquinigo Error PerfEvent::EnableWithIoctl() {
31003cc58ffSWalter Erquinigo if (m_enabled)
3111637545fSWalter Erquinigo return Error::success();
312a7582059SWalter Erquinigo
313a7582059SWalter Erquinigo if (ioctl(*m_fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) < 0)
314a7582059SWalter Erquinigo return createStringError(inconvertibleErrorCode(),
315a7582059SWalter Erquinigo "Can't enable perf event. %s",
316a7582059SWalter Erquinigo std::strerror(errno));
317a7582059SWalter Erquinigo
31803cc58ffSWalter Erquinigo m_enabled = true;
319a7582059SWalter Erquinigo return Error::success();
320a7582059SWalter Erquinigo }
321a7582059SWalter Erquinigo
GetEffectiveDataBufferSize() const322a7582059SWalter Erquinigo size_t PerfEvent::GetEffectiveDataBufferSize() const {
323*6599b6e5SDavid Spickett #ifndef PERF_ATTR_SIZE_VER5
324*6599b6e5SDavid Spickett llvm_unreachable("Intel PT Linux perf event not supported");
325*6599b6e5SDavid Spickett #else
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.
331*6599b6e5SDavid Spickett #endif
3321637545fSWalter Erquinigo }
33303cc58ffSWalter Erquinigo
33403cc58ffSWalter Erquinigo Expected<PerfEvent>
CreateContextSwitchTracePerfEvent(lldb::cpu_id_t cpu_id,const PerfEvent * parent_perf_event)33503cc58ffSWalter Erquinigo lldb_private::process_linux::CreateContextSwitchTracePerfEvent(
3366a5355e8SWalter Erquinigo lldb::cpu_id_t cpu_id, const PerfEvent *parent_perf_event) {
33703cc58ffSWalter Erquinigo Log *log = GetLog(POSIXLog::Trace);
33803cc58ffSWalter Erquinigo #ifndef PERF_ATTR_SIZE_VER5
33903cc58ffSWalter Erquinigo return createStringError(inconvertibleErrorCode(),
34003cc58ffSWalter Erquinigo "Intel PT Linux perf event not supported");
34103cc58ffSWalter Erquinigo #else
34203cc58ffSWalter Erquinigo perf_event_attr attr;
34303cc58ffSWalter Erquinigo memset(&attr, 0, sizeof(attr));
34403cc58ffSWalter Erquinigo attr.size = sizeof(attr);
34503cc58ffSWalter Erquinigo attr.sample_type = PERF_SAMPLE_TID | PERF_SAMPLE_TIME;
34603cc58ffSWalter Erquinigo attr.type = PERF_TYPE_SOFTWARE;
34703cc58ffSWalter Erquinigo attr.context_switch = 1;
34803cc58ffSWalter Erquinigo attr.exclude_kernel = 1;
34903cc58ffSWalter Erquinigo attr.sample_id_all = 1;
35003cc58ffSWalter Erquinigo attr.exclude_hv = 1;
35103cc58ffSWalter Erquinigo attr.disabled = parent_perf_event ? !parent_perf_event->IsEnabled() : false;
35203cc58ffSWalter Erquinigo
35303cc58ffSWalter Erquinigo // The given perf configuration will produce context switch records of 32
35403cc58ffSWalter Erquinigo // bytes each. Assuming that every context switch will be emitted twice (one
35503cc58ffSWalter Erquinigo // for context switch ins and another one for context switch outs), and that a
35603cc58ffSWalter Erquinigo // context switch will happen at least every half a millisecond per core, we
35703cc58ffSWalter Erquinigo // need 500 * 32 bytes (~16 KB) for a trace of one second, which is much more
35803cc58ffSWalter Erquinigo // than what a regular intel pt trace can get. Pessimistically we pick as
35903cc58ffSWalter Erquinigo // 32KiB for the size of our context switch trace.
36003cc58ffSWalter Erquinigo
36103cc58ffSWalter Erquinigo uint64_t data_buffer_size = 32768;
36203cc58ffSWalter Erquinigo uint64_t data_buffer_numpages = data_buffer_size / getpagesize();
36303cc58ffSWalter Erquinigo
36403cc58ffSWalter Erquinigo LLDB_LOG(log, "Will create context switch trace buffer of size {0}",
36503cc58ffSWalter Erquinigo data_buffer_size);
36603cc58ffSWalter Erquinigo
36703cc58ffSWalter Erquinigo Optional<long> group_fd;
36803cc58ffSWalter Erquinigo if (parent_perf_event)
36903cc58ffSWalter Erquinigo group_fd = parent_perf_event->GetFd();
37003cc58ffSWalter Erquinigo
37103cc58ffSWalter Erquinigo if (Expected<PerfEvent> perf_event =
3726a5355e8SWalter Erquinigo PerfEvent::Init(attr, /*pid=*/None, cpu_id, group_fd, /*flags=*/0)) {
37303cc58ffSWalter Erquinigo if (Error mmap_err = perf_event->MmapMetadataAndBuffers(
37403cc58ffSWalter Erquinigo data_buffer_numpages, 0, /*data_buffer_write=*/false)) {
37503cc58ffSWalter Erquinigo return std::move(mmap_err);
37603cc58ffSWalter Erquinigo }
37703cc58ffSWalter Erquinigo return perf_event;
37803cc58ffSWalter Erquinigo } else {
37903cc58ffSWalter Erquinigo return perf_event.takeError();
38003cc58ffSWalter Erquinigo }
38103cc58ffSWalter Erquinigo #endif
38203cc58ffSWalter Erquinigo }
383