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, so we first the oldest chunk of data
209     output.insert(output.end(), data.begin() + actual_data_head, data.end());
210     // And we we read the most recent chunk of data
211     output.insert(output.end(), data.begin(), data.begin() + actual_data_head);
212   } else {
213     // There's been no wrapping, so we just read linearly
214     output.insert(output.end(), data.begin(), data.begin() + data_head);
215   }
216 
217   if (was_enabled) {
218     if (Error err = EnableWithIoctl())
219       return std::move(err);
220   }
221 
222   return output;
223 }
224 
225 Expected<std::vector<uint8_t>> PerfEvent::GetReadOnlyAuxBuffer() {
226   // The following code assumes that the protection level of the AUX page
227   // is PROT_READ. If PROT_WRITE is used, then reading would require that
228   // this piece of code updates some pointers. See more about aux_tail
229   // in https://man7.org/linux/man-pages/man2/perf_event_open.2.html.
230 
231   bool was_enabled = m_enabled;
232   if (Error err = DisableWithIoctl())
233     return std::move(err);
234 
235   perf_event_mmap_page &mmap_metadata = GetMetadataPage();
236 
237   ArrayRef<uint8_t> data = GetAuxBuffer();
238   uint64_t aux_head = mmap_metadata.aux_head;
239   std::vector<uint8_t> output;
240   output.reserve(data.size());
241 
242   /**
243    * When configured as ring buffer, the aux buffer keeps wrapping around
244    * the buffer and its not possible to detect how many times the buffer
245    * wrapped. Initially the buffer is filled with zeros,as shown below
246    * so in order to get complete buffer we first copy firstpartsize, followed
247    * by any left over part from beginning to aux_head
248    *
249    * aux_offset [d,d,d,d,d,d,d,d,0,0,0,0,0,0,0,0,0,0,0] aux_size
250    *                 aux_head->||<- firstpartsize  ->|
251    *
252    * */
253 
254   output.insert(output.end(), data.begin() + aux_head, data.end());
255   output.insert(output.end(), data.begin(), data.begin() + aux_head);
256 
257   if (was_enabled) {
258     if (Error err = EnableWithIoctl())
259       return std::move(err);
260   }
261 
262   return output;
263 }
264 
265 Error PerfEvent::DisableWithIoctl() {
266   if (!m_enabled)
267     return Error::success();
268 
269   if (ioctl(*m_fd, PERF_EVENT_IOC_DISABLE, PERF_IOC_FLAG_GROUP) < 0)
270     return createStringError(inconvertibleErrorCode(),
271                              "Can't disable perf event. %s",
272                              std::strerror(errno));
273 
274   m_enabled = false;
275   return Error::success();
276 }
277 
278 bool PerfEvent::IsEnabled() const { return m_enabled; }
279 
280 Error PerfEvent::EnableWithIoctl() {
281   if (m_enabled)
282     return Error::success();
283 
284   if (ioctl(*m_fd, PERF_EVENT_IOC_ENABLE, PERF_IOC_FLAG_GROUP) < 0)
285     return createStringError(inconvertibleErrorCode(),
286                              "Can't enable perf event. %s",
287                              std::strerror(errno));
288 
289   m_enabled = true;
290   return Error::success();
291 }
292 
293 size_t PerfEvent::GetEffectiveDataBufferSize() const {
294   perf_event_mmap_page &mmap_metadata = GetMetadataPage();
295   if (mmap_metadata.data_head < mmap_metadata.data_size)
296     return mmap_metadata.data_head;
297   else
298     return mmap_metadata.data_size; // The buffer has wrapped.
299 }
300 
301 Expected<PerfEvent>
302 lldb_private::process_linux::CreateContextSwitchTracePerfEvent(
303     lldb::cpu_id_t cpu_id, const PerfEvent *parent_perf_event) {
304   Log *log = GetLog(POSIXLog::Trace);
305 #ifndef PERF_ATTR_SIZE_VER5
306   return createStringError(inconvertibleErrorCode(),
307                            "Intel PT Linux perf event not supported");
308 #else
309   perf_event_attr attr;
310   memset(&attr, 0, sizeof(attr));
311   attr.size = sizeof(attr);
312   attr.sample_type = PERF_SAMPLE_TID | PERF_SAMPLE_TIME;
313   attr.type = PERF_TYPE_SOFTWARE;
314   attr.context_switch = 1;
315   attr.exclude_kernel = 1;
316   attr.sample_id_all = 1;
317   attr.exclude_hv = 1;
318   attr.disabled = parent_perf_event ? !parent_perf_event->IsEnabled() : false;
319 
320   // The given perf configuration will produce context switch records of 32
321   // bytes each. Assuming that every context switch will be emitted twice (one
322   // for context switch ins and another one for context switch outs), and that a
323   // context switch will happen at least every half a millisecond per core, we
324   // need 500 * 32 bytes (~16 KB) for a trace of one second, which is much more
325   // than what a regular intel pt trace can get. Pessimistically we pick as
326   // 32KiB for the size of our context switch trace.
327 
328   uint64_t data_buffer_size = 32768;
329   uint64_t data_buffer_numpages = data_buffer_size / getpagesize();
330 
331   LLDB_LOG(log, "Will create context switch trace buffer of size {0}",
332            data_buffer_size);
333 
334   Optional<long> group_fd;
335   if (parent_perf_event)
336     group_fd = parent_perf_event->GetFd();
337 
338   if (Expected<PerfEvent> perf_event =
339           PerfEvent::Init(attr, /*pid=*/None, cpu_id, group_fd, /*flags=*/0)) {
340     if (Error mmap_err = perf_event->MmapMetadataAndBuffers(
341             data_buffer_numpages, 0, /*data_buffer_write=*/false)) {
342       return std::move(mmap_err);
343     }
344     return perf_event;
345   } else {
346     return perf_event.takeError();
347   }
348 #endif
349 }
350