194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
/*===- InstrProfilingPlatformFuchsia.c - Profile data Fuchsia platform ----===*\
 | 
						|
|*
 | 
						|
|* Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | 
						|
|* See https://llvm.org/LICENSE.txt for license information.
 | 
						|
|* SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | 
						|
|*
 | 
						|
\*===----------------------------------------------------------------------===*/
 | 
						|
/*
 | 
						|
 * This file implements the profiling runtime for Fuchsia and defines the
 | 
						|
 * shared profile runtime interface. Each module (executable or DSO) statically
 | 
						|
 * links in the whole profile runtime to satisfy the calls from its
 | 
						|
 * instrumented code. Several modules in the same program might be separately
 | 
						|
 * compiled and even use different versions of the instrumentation ABI and data
 | 
						|
 * format. All they share in common is the VMO and the offset, which live in
 | 
						|
 * exported globals so that exactly one definition will be shared across all
 | 
						|
 * modules. Each module has its own independent runtime that registers its own
 | 
						|
 * atexit hook to append its own data into the shared VMO which is published
 | 
						|
 * via the data sink hook provided by Fuchsia's dynamic linker.
 | 
						|
 */
 | 
						|
 | 
						|
#if defined(__Fuchsia__)
 | 
						|
 | 
						|
#include <inttypes.h>
 | 
						|
#include <stdarg.h>
 | 
						|
#include <stdbool.h>
 | 
						|
#include <stdlib.h>
 | 
						|
 | 
						|
#include <zircon/process.h>
 | 
						|
#include <zircon/sanitizer.h>
 | 
						|
#include <zircon/status.h>
 | 
						|
#include <zircon/syscalls.h>
 | 
						|
 | 
						|
#include "InstrProfiling.h"
 | 
						|
#include "InstrProfilingInternal.h"
 | 
						|
#include "InstrProfilingUtil.h"
 | 
						|
 | 
						|
COMPILER_RT_VISIBILITY unsigned lprofProfileDumped() {
 | 
						|
  return 1;
 | 
						|
}
 | 
						|
COMPILER_RT_VISIBILITY void lprofSetProfileDumped(unsigned Value) {}
 | 
						|
 | 
						|
COMPILER_RT_VISIBILITY unsigned lprofRuntimeCounterRelocation(void) {
 | 
						|
  return 1;
 | 
						|
}
 | 
						|
COMPILER_RT_VISIBILITY void lprofSetRuntimeCounterRelocation(unsigned Value) {}
 | 
						|
 | 
						|
static const char ProfileSinkName[] = "llvm-profile";
 | 
						|
 | 
						|
static inline void lprofWrite(const char *fmt, ...) {
 | 
						|
  char s[256];
 | 
						|
 | 
						|
  va_list ap;
 | 
						|
  va_start(ap, fmt);
 | 
						|
  int ret = vsnprintf(s, sizeof(s), fmt, ap);
 | 
						|
  va_end(ap);
 | 
						|
 | 
						|
  __sanitizer_log_write(s, ret + 1);
 | 
						|
}
 | 
						|
 | 
						|
struct lprofVMOWriterCtx {
 | 
						|
  /* VMO that contains the profile data for this module. */
 | 
						|
  zx_handle_t Vmo;
 | 
						|
  /* Current offset within the VMO where data should be written next. */
 | 
						|
  uint64_t Offset;
 | 
						|
};
 | 
						|
 | 
						|
static uint32_t lprofVMOWriter(ProfDataWriter *This, ProfDataIOVec *IOVecs,
 | 
						|
                               uint32_t NumIOVecs) {
 | 
						|
  struct lprofVMOWriterCtx *Ctx = (struct lprofVMOWriterCtx *)This->WriterCtx;
 | 
						|
 | 
						|
  /* Compute the total length of data to be written. */
 | 
						|
  size_t Length = 0;
 | 
						|
  for (uint32_t I = 0; I < NumIOVecs; I++)
 | 
						|
    Length += IOVecs[I].ElmSize * IOVecs[I].NumElm;
 | 
						|
 | 
						|
  /* Resize the VMO to ensure there's sufficient space for the data. */
 | 
						|
  zx_status_t Status = _zx_vmo_set_size(Ctx->Vmo, Ctx->Offset + Length);
 | 
						|
  if (Status != ZX_OK)
 | 
						|
    return -1;
 | 
						|
 | 
						|
  /* Copy the data into VMO. */
 | 
						|
  for (uint32_t I = 0; I < NumIOVecs; I++) {
 | 
						|
    size_t Length = IOVecs[I].ElmSize * IOVecs[I].NumElm;
 | 
						|
    if (IOVecs[I].Data) {
 | 
						|
      Status = _zx_vmo_write(Ctx->Vmo, IOVecs[I].Data, Ctx->Offset, Length);
 | 
						|
      if (Status != ZX_OK)
 | 
						|
        return -1;
 | 
						|
    } else if (IOVecs[I].UseZeroPadding) {
 | 
						|
      /* Resizing the VMO should zero fill. */
 | 
						|
    }
 | 
						|
    Ctx->Offset += Length;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Record the profile size as a property of the VMO. */
 | 
						|
  _zx_object_set_property(Ctx->Vmo, ZX_PROP_VMO_CONTENT_SIZE, &Ctx->Offset,
 | 
						|
                          sizeof(Ctx->Offset));
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void initVMOWriter(ProfDataWriter *This, struct lprofVMOWriterCtx *Ctx) {
 | 
						|
  This->Write = lprofVMOWriter;
 | 
						|
  This->WriterCtx = Ctx;
 | 
						|
}
 | 
						|
 | 
						|
/* This method is invoked by the runtime initialization hook
 | 
						|
 * InstrProfilingRuntime.o if it is linked in. */
 | 
						|
COMPILER_RT_VISIBILITY
 | 
						|
void __llvm_profile_initialize(void) {
 | 
						|
  /* Check if there is llvm/runtime version mismatch. */
 | 
						|
  if (GET_VERSION(__llvm_profile_get_version()) != INSTR_PROF_RAW_VERSION) {
 | 
						|
    lprofWrite("LLVM Profile: runtime and instrumentation version mismatch: "
 | 
						|
               "expected %d, but got %d\n",
 | 
						|
               INSTR_PROF_RAW_VERSION,
 | 
						|
               (int)GET_VERSION(__llvm_profile_get_version()));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  /* This symbol is defined as weak and initialized to -1 by the runtimer, but
 | 
						|
   * compiler will generate a strong definition initialized to 0 when runtime
 | 
						|
   * counter relocation is used. */
 | 
						|
  if (__llvm_profile_counter_bias == -1) {
 | 
						|
    lprofWrite("LLVM Profile: counter relocation at runtime is required\n");
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  const __llvm_profile_data *DataBegin = __llvm_profile_begin_data();
 | 
						|
  const __llvm_profile_data *DataEnd = __llvm_profile_end_data();
 | 
						|
  const uint64_t DataSize = __llvm_profile_get_data_size(DataBegin, DataEnd);
 | 
						|
  const uint64_t CountersOffset =
 | 
						|
      sizeof(__llvm_profile_header) + (DataSize * sizeof(__llvm_profile_data));
 | 
						|
 | 
						|
  zx_status_t Status;
 | 
						|
 | 
						|
  /* Create VMO to hold the profile data. */
 | 
						|
  zx_handle_t Vmo = ZX_HANDLE_INVALID;
 | 
						|
  Status = _zx_vmo_create(0, ZX_VMO_RESIZABLE, &Vmo);
 | 
						|
  if (Status != ZX_OK) {
 | 
						|
    lprofWrite("LLVM Profile: cannot create VMO: %s\n",
 | 
						|
               _zx_status_get_string(Status));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Give the VMO a name that includes the module signature. */
 | 
						|
  char VmoName[ZX_MAX_NAME_LEN];
 | 
						|
  snprintf(VmoName, sizeof(VmoName), "%" PRIu64 ".profraw",
 | 
						|
           lprofGetLoadModuleSignature());
 | 
						|
  _zx_object_set_property(Vmo, ZX_PROP_NAME, VmoName, strlen(VmoName));
 | 
						|
 | 
						|
  /* Write the profile data into the mapped region. */
 | 
						|
  ProfDataWriter VMOWriter;
 | 
						|
  struct lprofVMOWriterCtx Ctx = {.Vmo = Vmo, .Offset = 0};
 | 
						|
  initVMOWriter(&VMOWriter, &Ctx);
 | 
						|
  if (lprofWriteData(&VMOWriter, 0, 0) != 0) {
 | 
						|
    lprofWrite("LLVM Profile: failed to write data\n");
 | 
						|
    _zx_handle_close(Vmo);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  uint64_t Len = 0;
 | 
						|
  Status = _zx_vmo_get_size(Vmo, &Len);
 | 
						|
  if (Status != ZX_OK) {
 | 
						|
    lprofWrite("LLVM Profile: failed to get the VMO size: %s\n",
 | 
						|
               _zx_status_get_string(Status));
 | 
						|
    _zx_handle_close(Vmo);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  uintptr_t Mapping;
 | 
						|
  Status =
 | 
						|
      _zx_vmar_map(_zx_vmar_root_self(), ZX_VM_PERM_READ | ZX_VM_PERM_WRITE, 0,
 | 
						|
                   Vmo, 0, Len, &Mapping);
 | 
						|
  if (Status != ZX_OK) {
 | 
						|
    lprofWrite("LLVM Profile: failed to map the VMO: %s\n",
 | 
						|
               _zx_status_get_string(Status));
 | 
						|
    _zx_handle_close(Vmo);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  /* Publish the VMO which contains profile data to the system. Note that this
 | 
						|
   * also consumes the VMO handle. */
 | 
						|
  __sanitizer_publish_data(ProfileSinkName, Vmo);
 | 
						|
 | 
						|
  /* Use the dumpfile symbolizer markup element to write the name of VMO. */
 | 
						|
  lprofWrite("LLVM Profile: {{{dumpfile:%s:%s}}}\n", ProfileSinkName, VmoName);
 | 
						|
 | 
						|
  /* Update the profile fields based on the current mapping. */
 | 
						|
  __llvm_profile_counter_bias = (intptr_t)Mapping -
 | 
						|
                                (uintptr_t)__llvm_profile_begin_counters() +
 | 
						|
                                CountersOffset;
 | 
						|
}
 | 
						|
 | 
						|
#endif
 |