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