HEX
Server: Apache
System: Linux wp02.tdr-lab.com 3.10.0-1160.42.2.el7.x86_64 #1 SMP Tue Sep 7 14:49:57 UTC 2021 x86_64
User: kusanagi (1001)
PHP: 7.4.23
Disabled: NONE
Upload Files
File: //usr/include/hphp/util/perf-event.h
/*
   +----------------------------------------------------------------------+
   | HipHop for PHP                                                       |
   +----------------------------------------------------------------------+
   | Copyright (c) 2010-present Facebook, Inc. (http://www.facebook.com)  |
   +----------------------------------------------------------------------+
   | This source file is subject to version 3.01 of the PHP license,      |
   | that is bundled with this package in the file LICENSE, and is        |
   | available through the world-wide-web at the following url:           |
   | http://www.php.net/license/3_01.txt                                  |
   | If you did not receive a copy of the PHP license and are unable to   |
   | obtain it through the world-wide-web, please send a note to          |
   | license@php.net so we can mail you a copy immediately.               |
   +----------------------------------------------------------------------+
*/

#ifndef incl_HPHP_UTIL_PERF_EVENT_H_
#define incl_HPHP_UTIL_PERF_EVENT_H_

#include <cstdint>

namespace HPHP {

///////////////////////////////////////////////////////////////////////////////

enum class PerfEvent { Load, Store };

struct perf_event_sample_tail;

/*
 * Raw data from a sampled perf event.
 *
 * A perf_event_sample is always followed immediately in memory by the
 * corresponding perf_event_sample_tail.
 */
struct perf_event_sample {
  uintptr_t ip;     // address of the sampled instruction
  uint32_t pid;     // process in which the event occurred
  uint32_t tid;     // thread in which the event occurred
  uintptr_t addr;   // memory address corresponding to the event, if applicable
  uint64_t nr;      // number of addresses in the callchain
  uintptr_t ips[];  // instruction pointers in the callchain for the event

  const perf_event_sample_tail* tail() const {
    return reinterpret_cast<const perf_event_sample_tail*>(
      reinterpret_cast<const uintptr_t*>(this + 1) + nr
    );
  }
  perf_event_sample_tail* tail() {
    return const_cast<perf_event_sample_tail*>(
      const_cast<const perf_event_sample*>(this)->tail()
    );
  }
};

struct perf_event_sample_tail {
  uint64_t data_src;
};

/*
 * Metadata about the raw `data_src' of a perf_event_sample.
 */
struct perf_event_data_src_info {
  const char* mem_lvl;  // memory hierarchy level reported
  const char* tlb;      // TLB level reported

  /*
   * Ternary bits: 1 is true, -1 is false, 0 is unknown.
   */
  int mem_hit : 2;    // cache hit/miss
  int snoop : 2;      // snoop?
  int snoop_hit : 2;  // snoop hit/miss
  int snoop_hitm : 2; // snoop hit modified?
  int locked : 2;     // whether the op was part of a locked transaction
  int tlb_hit : 2;    // TLB hit/miss
};

/*
 * Interpret a raw `data_src' value, based on the event `kind'.
 */
perf_event_data_src_info perf_event_data_src(PerfEvent kind, uint64_t data_src);

///////////////////////////////////////////////////////////////////////////////

using perf_event_signal_fn_t = void (*)(PerfEvent);
using perf_event_consume_fn_t = void (*)(PerfEvent, const perf_event_sample*);

/*
 * Enable sampling of load and store instructions on this thread.
 *
 * Perf events will be sampled roughly `sample_freq' times per second (with
 * loads and stores sampled separately).  On each sampled event, `signal_fn'
 * will be invoked in the context of the calling thread, via signal handler.
 *
 * Behavior is undefined if the SIGIO handler is reset, or if SIGIO is masked,
 * after calling this function.
 *
 * Returns true if sampling was successfully enabled, else false (both if
 * sampling has already been enabled, or if an error occurs).
 */
bool perf_event_enable(uint64_t sample_freq, perf_event_signal_fn_t signal_fn);

/*
 * Disable perf memory event sampling on this thread.
 *
 * All unconsumed samples will be lost, even if sampling is later reenabled.
 */
void perf_event_disable();

/*
 * Process and clear sampled perf events.
 *
 * This function can be safely called as long as sampling has been enabled for
 * the calling thread.  Each invocation of `signal_fn' indicates that that
 * there are events to be consumed---though consuming them from within
 * `signal_fn' is discouraged.
 *
 * Each sampled event is passed to `consume' exactly once.
 */
void perf_event_consume(perf_event_consume_fn_t consume);

/*
 * Pause or resume sampling for an event.
 *
 * Can be used instead of perf_event_{enable,disable}() for an event that has
 * already been opened, but which should be briefly turned off.
 *
 * This should likely be called in whatever routine is used to consume events,
 * to avoid reentrant sampling.
 */
void perf_event_pause();
void perf_event_resume();

///////////////////////////////////////////////////////////////////////////////

}

#endif