]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/compiler-rt/lib/xray/xray_utils.cc
Merge llvm, clang, compiler-rt, libc++, libunwind, lld, lldb and openmp
[FreeBSD/FreeBSD.git] / contrib / compiler-rt / lib / xray / xray_utils.cc
1 //===-- xray_utils.cc -------------------------------------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file is a part of XRay, a dynamic runtime instrumentation system.
11 //
12 //===----------------------------------------------------------------------===//
13 #include "xray_utils.h"
14
15 #include "sanitizer_common/sanitizer_allocator_internal.h"
16 #include "sanitizer_common/sanitizer_common.h"
17 #include "xray_allocator.h"
18 #include "xray_defs.h"
19 #include "xray_flags.h"
20 #include <cstdio>
21 #include <errno.h>
22 #include <fcntl.h>
23 #include <iterator>
24 #include <stdlib.h>
25 #include <sys/types.h>
26 #include <tuple>
27 #include <unistd.h>
28 #include <utility>
29
30 #if SANITIZER_FUCHSIA
31 #include "sanitizer_common/sanitizer_symbolizer_fuchsia.h"
32
33 #include <inttypes.h>
34 #include <zircon/process.h>
35 #include <zircon/sanitizer.h>
36 #include <zircon/status.h>
37 #include <zircon/syscalls.h>
38 #endif
39
40 namespace __xray {
41
42 #if SANITIZER_FUCHSIA
43 constexpr const char* ProfileSinkName = "llvm-xray";
44
45 LogWriter::~LogWriter() {
46   _zx_handle_close(Vmo);
47 }
48
49 void LogWriter::WriteAll(const char *Begin, const char *End) XRAY_NEVER_INSTRUMENT {
50   if (Begin == End)
51     return;
52   auto TotalBytes = std::distance(Begin, End);
53
54   const size_t PageSize = flags()->xray_page_size_override > 0
55                               ? flags()->xray_page_size_override
56                               : GetPageSizeCached();
57   if (RoundUpTo(Offset, PageSize) != RoundUpTo(Offset + TotalBytes, PageSize)) {
58     // Resize the VMO to ensure there's sufficient space for the data.
59     zx_status_t Status = _zx_vmo_set_size(Vmo, Offset + TotalBytes);
60     if (Status != ZX_OK) {
61       Report("Failed to resize VMO: %s\n", _zx_status_get_string(Status));
62       return;
63     }
64   }
65
66   // Write the data into VMO.
67   zx_status_t Status = _zx_vmo_write(Vmo, Begin, Offset, TotalBytes);
68   if (Status != ZX_OK) {
69     Report("Failed to write: %s\n", _zx_status_get_string(Status));
70     return;
71   }
72   Offset += TotalBytes;
73 }
74
75 void LogWriter::Flush() XRAY_NEVER_INSTRUMENT {
76   // Nothing to do here since WriteAll writes directly into the VMO.
77 }
78
79 LogWriter *LogWriter::Open() XRAY_NEVER_INSTRUMENT {
80   // Create VMO to hold the profile data.
81   zx_handle_t Vmo;
82   zx_status_t Status = _zx_vmo_create(0, 0, &Vmo);
83   if (Status != ZX_OK) {
84     Report("XRay: cannot create VMO: %s\n", _zx_status_get_string(Status));
85     return nullptr;
86   }
87
88   // Get the KOID of the current process to use in the VMO name.
89   zx_info_handle_basic_t Info;
90   Status = _zx_object_get_info(_zx_process_self(), ZX_INFO_HANDLE_BASIC, &Info,
91                                sizeof(Info), NULL, NULL);
92   if (Status != ZX_OK) {
93     Report("XRay: cannot get basic info about current process handle: %s\n",
94            _zx_status_get_string(Status));
95     return nullptr;
96   }
97
98   // Give the VMO a name including our process KOID so it's easy to spot.
99   char VmoName[ZX_MAX_NAME_LEN];
100   internal_snprintf(VmoName, sizeof(VmoName), "%s.%zu", ProfileSinkName,
101                     Info.koid);
102   _zx_object_set_property(Vmo, ZX_PROP_NAME, VmoName, strlen(VmoName));
103
104   // Duplicate the handle since __sanitizer_publish_data consumes it and
105   // LogWriter needs to hold onto it.
106   zx_handle_t Handle;
107   Status =_zx_handle_duplicate(Vmo, ZX_RIGHT_SAME_RIGHTS, &Handle);
108   if (Status != ZX_OK) {
109     Report("XRay: cannot duplicate VMO handle: %s\n",
110            _zx_status_get_string(Status));
111     return nullptr;
112   }
113
114   // Publish the VMO that receives the logging. Note the VMO's contents can
115   // grow and change after publication. The contents won't be read out until
116   // after the process exits.
117   __sanitizer_publish_data(ProfileSinkName, Handle);
118
119   // Use the dumpfile symbolizer markup element to write the name of the VMO.
120   Report("XRay: " FORMAT_DUMPFILE "\n", ProfileSinkName, VmoName);
121
122   LogWriter *LW = reinterpret_cast<LogWriter *>(InternalAlloc(sizeof(LogWriter)));
123   new (LW) LogWriter(Vmo);
124   return LW;
125 }
126
127 void LogWriter::Close(LogWriter *LW) {
128   LW->~LogWriter();
129   InternalFree(LW);
130 }
131 #else // SANITIZER_FUCHSIA
132 LogWriter::~LogWriter() {
133   internal_close(Fd);
134 }
135
136 void LogWriter::WriteAll(const char *Begin, const char *End) XRAY_NEVER_INSTRUMENT {
137   if (Begin == End)
138     return;
139   auto TotalBytes = std::distance(Begin, End);
140   while (auto Written = write(Fd, Begin, TotalBytes)) {
141     if (Written < 0) {
142       if (errno == EINTR)
143         continue; // Try again.
144       Report("Failed to write; errno = %d\n", errno);
145       return;
146     }
147     TotalBytes -= Written;
148     if (TotalBytes == 0)
149       break;
150     Begin += Written;
151   }
152 }
153
154 void LogWriter::Flush() XRAY_NEVER_INSTRUMENT {
155   fsync(Fd);
156 }
157
158 LogWriter *LogWriter::Open() XRAY_NEVER_INSTRUMENT {
159   // Open a temporary file once for the log.
160   char TmpFilename[256] = {};
161   char TmpWildcardPattern[] = "XXXXXX";
162   auto **Argv = GetArgv();
163   const char *Progname = !Argv ? "(unknown)" : Argv[0];
164   const char *LastSlash = internal_strrchr(Progname, '/');
165
166   if (LastSlash != nullptr)
167     Progname = LastSlash + 1;
168
169   int NeededLength = internal_snprintf(
170       TmpFilename, sizeof(TmpFilename), "%s%s.%s",
171       flags()->xray_logfile_base, Progname, TmpWildcardPattern);
172   if (NeededLength > int(sizeof(TmpFilename))) {
173     Report("XRay log file name too long (%d): %s\n", NeededLength, TmpFilename);
174     return nullptr;
175   }
176   int Fd = mkstemp(TmpFilename);
177   if (Fd == -1) {
178     Report("XRay: Failed opening temporary file '%s'; not logging events.\n",
179            TmpFilename);
180     return nullptr;
181   }
182   if (Verbosity())
183     Report("XRay: Log file in '%s'\n", TmpFilename);
184
185   LogWriter *LW = allocate<LogWriter>();
186   new (LW) LogWriter(Fd);
187   return LW;
188 }
189
190 void LogWriter::Close(LogWriter *LW) {
191   LW->~LogWriter();
192   deallocate(LW);
193 }
194 #endif // SANITIZER_FUCHSIA
195
196 } // namespace __xray