xref: /openbsd-src/gnu/llvm/compiler-rt/lib/xray/xray_utils.cpp (revision d89ec533011f513df1010f142a111086a0785f09)
13cab2bb3Spatrick //===-- xray_utils.cpp ------------------------------------------*- C++ -*-===//
23cab2bb3Spatrick //
33cab2bb3Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
43cab2bb3Spatrick // See https://llvm.org/LICENSE.txt for license information.
53cab2bb3Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
63cab2bb3Spatrick //
73cab2bb3Spatrick //===----------------------------------------------------------------------===//
83cab2bb3Spatrick //
93cab2bb3Spatrick // This file is a part of XRay, a dynamic runtime instrumentation system.
103cab2bb3Spatrick //
113cab2bb3Spatrick //===----------------------------------------------------------------------===//
123cab2bb3Spatrick #include "xray_utils.h"
133cab2bb3Spatrick 
143cab2bb3Spatrick #include "sanitizer_common/sanitizer_allocator_internal.h"
153cab2bb3Spatrick #include "sanitizer_common/sanitizer_common.h"
163cab2bb3Spatrick #include "xray_allocator.h"
173cab2bb3Spatrick #include "xray_defs.h"
183cab2bb3Spatrick #include "xray_flags.h"
193cab2bb3Spatrick #include <cstdio>
203cab2bb3Spatrick #include <errno.h>
213cab2bb3Spatrick #include <fcntl.h>
223cab2bb3Spatrick #include <iterator>
23*d89ec533Spatrick #include <new>
243cab2bb3Spatrick #include <stdlib.h>
253cab2bb3Spatrick #include <sys/types.h>
263cab2bb3Spatrick #include <tuple>
273cab2bb3Spatrick #include <unistd.h>
283cab2bb3Spatrick #include <utility>
293cab2bb3Spatrick 
303cab2bb3Spatrick #if SANITIZER_FUCHSIA
313cab2bb3Spatrick #include "sanitizer_common/sanitizer_symbolizer_fuchsia.h"
323cab2bb3Spatrick 
333cab2bb3Spatrick #include <inttypes.h>
343cab2bb3Spatrick #include <zircon/process.h>
353cab2bb3Spatrick #include <zircon/sanitizer.h>
363cab2bb3Spatrick #include <zircon/status.h>
373cab2bb3Spatrick #include <zircon/syscalls.h>
383cab2bb3Spatrick #endif
393cab2bb3Spatrick 
403cab2bb3Spatrick namespace __xray {
413cab2bb3Spatrick 
423cab2bb3Spatrick #if SANITIZER_FUCHSIA
433cab2bb3Spatrick constexpr const char* ProfileSinkName = "llvm-xray";
443cab2bb3Spatrick 
~LogWriter()453cab2bb3Spatrick LogWriter::~LogWriter() {
463cab2bb3Spatrick   _zx_handle_close(Vmo);
473cab2bb3Spatrick }
483cab2bb3Spatrick 
WriteAll(const char * Begin,const char * End)493cab2bb3Spatrick void LogWriter::WriteAll(const char *Begin, const char *End) XRAY_NEVER_INSTRUMENT {
503cab2bb3Spatrick   if (Begin == End)
513cab2bb3Spatrick     return;
523cab2bb3Spatrick   auto TotalBytes = std::distance(Begin, End);
533cab2bb3Spatrick 
543cab2bb3Spatrick   const size_t PageSize = flags()->xray_page_size_override > 0
553cab2bb3Spatrick                               ? flags()->xray_page_size_override
563cab2bb3Spatrick                               : GetPageSizeCached();
573cab2bb3Spatrick   if (RoundUpTo(Offset, PageSize) != RoundUpTo(Offset + TotalBytes, PageSize)) {
583cab2bb3Spatrick     // Resize the VMO to ensure there's sufficient space for the data.
593cab2bb3Spatrick     zx_status_t Status = _zx_vmo_set_size(Vmo, Offset + TotalBytes);
603cab2bb3Spatrick     if (Status != ZX_OK) {
613cab2bb3Spatrick       Report("Failed to resize VMO: %s\n", _zx_status_get_string(Status));
623cab2bb3Spatrick       return;
633cab2bb3Spatrick     }
643cab2bb3Spatrick   }
653cab2bb3Spatrick 
663cab2bb3Spatrick   // Write the data into VMO.
673cab2bb3Spatrick   zx_status_t Status = _zx_vmo_write(Vmo, Begin, Offset, TotalBytes);
683cab2bb3Spatrick   if (Status != ZX_OK) {
693cab2bb3Spatrick     Report("Failed to write: %s\n", _zx_status_get_string(Status));
703cab2bb3Spatrick     return;
713cab2bb3Spatrick   }
723cab2bb3Spatrick   Offset += TotalBytes;
731f9cb04fSpatrick 
741f9cb04fSpatrick   // Record the data size as a property of the VMO.
751f9cb04fSpatrick   _zx_object_set_property(Vmo, ZX_PROP_VMO_CONTENT_SIZE,
761f9cb04fSpatrick                           &Offset, sizeof(Offset));
773cab2bb3Spatrick }
783cab2bb3Spatrick 
Flush()793cab2bb3Spatrick void LogWriter::Flush() XRAY_NEVER_INSTRUMENT {
803cab2bb3Spatrick   // Nothing to do here since WriteAll writes directly into the VMO.
813cab2bb3Spatrick }
823cab2bb3Spatrick 
Open()833cab2bb3Spatrick LogWriter *LogWriter::Open() XRAY_NEVER_INSTRUMENT {
843cab2bb3Spatrick   // Create VMO to hold the profile data.
853cab2bb3Spatrick   zx_handle_t Vmo;
863cab2bb3Spatrick   zx_status_t Status = _zx_vmo_create(0, ZX_VMO_RESIZABLE, &Vmo);
873cab2bb3Spatrick   if (Status != ZX_OK) {
883cab2bb3Spatrick     Report("XRay: cannot create VMO: %s\n", _zx_status_get_string(Status));
893cab2bb3Spatrick     return nullptr;
903cab2bb3Spatrick   }
913cab2bb3Spatrick 
923cab2bb3Spatrick   // Get the KOID of the current process to use in the VMO name.
933cab2bb3Spatrick   zx_info_handle_basic_t Info;
943cab2bb3Spatrick   Status = _zx_object_get_info(_zx_process_self(), ZX_INFO_HANDLE_BASIC, &Info,
953cab2bb3Spatrick                                sizeof(Info), NULL, NULL);
963cab2bb3Spatrick   if (Status != ZX_OK) {
973cab2bb3Spatrick     Report("XRay: cannot get basic info about current process handle: %s\n",
983cab2bb3Spatrick            _zx_status_get_string(Status));
993cab2bb3Spatrick     return nullptr;
1003cab2bb3Spatrick   }
1013cab2bb3Spatrick 
1023cab2bb3Spatrick   // Give the VMO a name including our process KOID so it's easy to spot.
1033cab2bb3Spatrick   char VmoName[ZX_MAX_NAME_LEN];
1043cab2bb3Spatrick   internal_snprintf(VmoName, sizeof(VmoName), "%s.%zu", ProfileSinkName,
1053cab2bb3Spatrick                     Info.koid);
1063cab2bb3Spatrick   _zx_object_set_property(Vmo, ZX_PROP_NAME, VmoName, strlen(VmoName));
1073cab2bb3Spatrick 
1083cab2bb3Spatrick   // Duplicate the handle since __sanitizer_publish_data consumes it and
1093cab2bb3Spatrick   // LogWriter needs to hold onto it.
1103cab2bb3Spatrick   zx_handle_t Handle;
1113cab2bb3Spatrick   Status =_zx_handle_duplicate(Vmo, ZX_RIGHT_SAME_RIGHTS, &Handle);
1123cab2bb3Spatrick   if (Status != ZX_OK) {
1133cab2bb3Spatrick     Report("XRay: cannot duplicate VMO handle: %s\n",
1143cab2bb3Spatrick            _zx_status_get_string(Status));
1153cab2bb3Spatrick     return nullptr;
1163cab2bb3Spatrick   }
1173cab2bb3Spatrick 
1183cab2bb3Spatrick   // Publish the VMO that receives the logging. Note the VMO's contents can
1193cab2bb3Spatrick   // grow and change after publication. The contents won't be read out until
1203cab2bb3Spatrick   // after the process exits.
1213cab2bb3Spatrick   __sanitizer_publish_data(ProfileSinkName, Handle);
1223cab2bb3Spatrick 
1233cab2bb3Spatrick   // Use the dumpfile symbolizer markup element to write the name of the VMO.
1243cab2bb3Spatrick   Report("XRay: " FORMAT_DUMPFILE "\n", ProfileSinkName, VmoName);
1253cab2bb3Spatrick 
1263cab2bb3Spatrick   LogWriter *LW = reinterpret_cast<LogWriter *>(InternalAlloc(sizeof(LogWriter)));
1273cab2bb3Spatrick   new (LW) LogWriter(Vmo);
1283cab2bb3Spatrick   return LW;
1293cab2bb3Spatrick }
1303cab2bb3Spatrick 
Close(LogWriter * LW)1313cab2bb3Spatrick void LogWriter::Close(LogWriter *LW) {
1323cab2bb3Spatrick   LW->~LogWriter();
1333cab2bb3Spatrick   InternalFree(LW);
1343cab2bb3Spatrick }
1353cab2bb3Spatrick #else // SANITIZER_FUCHSIA
1363cab2bb3Spatrick LogWriter::~LogWriter() {
1373cab2bb3Spatrick   internal_close(Fd);
1383cab2bb3Spatrick }
1393cab2bb3Spatrick 
1403cab2bb3Spatrick void LogWriter::WriteAll(const char *Begin, const char *End) XRAY_NEVER_INSTRUMENT {
1413cab2bb3Spatrick   if (Begin == End)
1423cab2bb3Spatrick     return;
1433cab2bb3Spatrick   auto TotalBytes = std::distance(Begin, End);
1443cab2bb3Spatrick   while (auto Written = write(Fd, Begin, TotalBytes)) {
1453cab2bb3Spatrick     if (Written < 0) {
1463cab2bb3Spatrick       if (errno == EINTR)
1473cab2bb3Spatrick         continue; // Try again.
1483cab2bb3Spatrick       Report("Failed to write; errno = %d\n", errno);
1493cab2bb3Spatrick       return;
1503cab2bb3Spatrick     }
1513cab2bb3Spatrick     TotalBytes -= Written;
1523cab2bb3Spatrick     if (TotalBytes == 0)
1533cab2bb3Spatrick       break;
1543cab2bb3Spatrick     Begin += Written;
1553cab2bb3Spatrick   }
1563cab2bb3Spatrick }
1573cab2bb3Spatrick 
1583cab2bb3Spatrick void LogWriter::Flush() XRAY_NEVER_INSTRUMENT {
1593cab2bb3Spatrick   fsync(Fd);
1603cab2bb3Spatrick }
1613cab2bb3Spatrick 
1623cab2bb3Spatrick LogWriter *LogWriter::Open() XRAY_NEVER_INSTRUMENT {
1633cab2bb3Spatrick   // Open a temporary file once for the log.
1643cab2bb3Spatrick   char TmpFilename[256] = {};
1653cab2bb3Spatrick   char TmpWildcardPattern[] = "XXXXXX";
1663cab2bb3Spatrick   auto **Argv = GetArgv();
1673cab2bb3Spatrick   const char *Progname = !Argv ? "(unknown)" : Argv[0];
1683cab2bb3Spatrick   const char *LastSlash = internal_strrchr(Progname, '/');
1693cab2bb3Spatrick 
1703cab2bb3Spatrick   if (LastSlash != nullptr)
1713cab2bb3Spatrick     Progname = LastSlash + 1;
1723cab2bb3Spatrick 
1733cab2bb3Spatrick   int NeededLength = internal_snprintf(
1743cab2bb3Spatrick       TmpFilename, sizeof(TmpFilename), "%s%s.%s",
1753cab2bb3Spatrick       flags()->xray_logfile_base, Progname, TmpWildcardPattern);
1763cab2bb3Spatrick   if (NeededLength > int(sizeof(TmpFilename))) {
1773cab2bb3Spatrick     Report("XRay log file name too long (%d): %s\n", NeededLength, TmpFilename);
1783cab2bb3Spatrick     return nullptr;
1793cab2bb3Spatrick   }
1803cab2bb3Spatrick   int Fd = mkstemp(TmpFilename);
1813cab2bb3Spatrick   if (Fd == -1) {
1823cab2bb3Spatrick     Report("XRay: Failed opening temporary file '%s'; not logging events.\n",
1833cab2bb3Spatrick            TmpFilename);
1843cab2bb3Spatrick     return nullptr;
1853cab2bb3Spatrick   }
1863cab2bb3Spatrick   if (Verbosity())
1873cab2bb3Spatrick     Report("XRay: Log file in '%s'\n", TmpFilename);
1883cab2bb3Spatrick 
1893cab2bb3Spatrick   LogWriter *LW = allocate<LogWriter>();
1903cab2bb3Spatrick   new (LW) LogWriter(Fd);
1913cab2bb3Spatrick   return LW;
1923cab2bb3Spatrick }
1933cab2bb3Spatrick 
1943cab2bb3Spatrick void LogWriter::Close(LogWriter *LW) {
1953cab2bb3Spatrick   LW->~LogWriter();
1963cab2bb3Spatrick   deallocate(LW);
1973cab2bb3Spatrick }
1983cab2bb3Spatrick #endif // SANITIZER_FUCHSIA
1993cab2bb3Spatrick 
2003cab2bb3Spatrick } // namespace __xray
201