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