Skip to content

Commit

Permalink
Use Hermes Sampling Profiler API for JS Samplings (facebook#48315)
Browse files Browse the repository at this point in the history
Summary:

# Changelog: [Internal]

Use newly added API to Hermes.

In terms of Trace Events, we are going to emit:
- An event for Process metadata, which will be named "Hermes"
- An event for Thread metadata, on which Hermes runs
- Complete Trace Event for the same Thread, which is required track to appear in the Timeline UI. I believe duration field is the key for this
- Trace Event for Profile
- Trace Event for ProfileChunk, which contains information about nodes and the callstack tree representation

> Important: current implementation emits single ProfileChunk event. I didn't notice any issues in terms of JSON bandwidth while recording 1 minute long session in CPU-heavy application. We should re-evaluate this decision in the future and split Profiles into multiple chunks.

Differential Revision: D67353586
  • Loading branch information
hoxyq authored and facebook-github-bot committed Dec 18, 2024
1 parent 4dac99c commit 19190f2
Show file tree
Hide file tree
Showing 2 changed files with 148 additions and 0 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,9 @@
#include "PerformanceTracer.h"

#include <folly/json.h>
#include <hermes/hermes.h>

#include <iostream>
#include <mutex>
#include <unordered_set>

Expand All @@ -22,6 +24,12 @@ const uint64_t PID = 1000;
/** Default/starting track ID for the "Timings" track. */
const uint64_t USER_TIMINGS_DEFAULT_TRACK = 1000;

long long getCurrentTimestamp() {
return std::chrono::duration_cast<std::chrono::microseconds>(
std::chrono::steady_clock::now().time_since_epoch())
.count();
}

} // namespace

PerformanceTracer& PerformanceTracer::getInstance() {
Expand All @@ -34,10 +42,132 @@ bool PerformanceTracer::startTracing() {
if (tracing_) {
return false;
}

tracingStartedTimestamp_ = getCurrentTimestamp();
facebook::hermes::HermesRuntime::enableSamplingProfiler(1000);

tracing_ = true;
return true;
}

void PerformanceTracer::processSingleHermesDataSource(
const facebook::hermes::sampling_profiler::TraceEventCollectionDataSource&
source) {
buffer_.push_back(TraceEvent{
.name = "process_name",
.cat = "__metadata",
.ph = 'M',
.ts = 0,
.pid = source.getProcess().getId(),
.tid = 0,
.args = folly::dynamic::object("name", source.getProcess().getName()),
});

for (const facebook::hermes::sampling_profiler::Thread& thread :
source.getThreads()) {
buffer_.push_back(TraceEvent{
.name = "thread_name",
.cat = "__metadata",
.ph = 'M',
.ts = 0,
.pid = source.getProcess().getId(),
.tid = thread.getId(),
.args = folly::dynamic::object("name", thread.getName()),
});

buffer_.push_back(TraceEvent{
.name = thread.getName(),
.cat = "disabled-by-default-devtools.timeline",
.ph = 'X',
.ts = 0,
.pid = source.getProcess().getId(),
.tid = thread.getId(),
.dur = 0,
});
}

facebook::hermes::sampling_profiler::Profile profile = source.getProfile();
buffer_.push_back(TraceEvent{
.name = "Profile",
.cat = "disabled-by-default-v8.cpu_profiler",
.ph = 'P',
.ts = profile.getTimestamp(),
.pid = profile.getProcessId(),
.tid = profile.getThreadId(),
.args = folly::dynamic::object(
"data",
folly ::dynamic::object("startTime", tracingStartedTimestamp_)),
});

facebook::hermes::sampling_profiler::ProfileChunk profileChunk =
source.getProfileChunk();
folly::dynamic dynamicNodes = folly::dynamic::array();
for (const facebook::hermes::sampling_profiler::ProfileNode& node :
profileChunk.getNodes()) {
facebook::hermes::sampling_profiler::ProfileNodeCallFrame callFrame =
node.getCallFrame();

folly::dynamic dynamicCallFrame = folly::dynamic::object();
dynamicCallFrame["codeType"] = callFrame.getCodeType();
dynamicCallFrame["scriptId"] = callFrame.getScriptId();
dynamicCallFrame["functionName"] = callFrame.getFunctionName();
if (callFrame.hasUrl()) {
dynamicCallFrame["url"] = callFrame.getUrl();
}
if (callFrame.hasLineNumber()) {
dynamicCallFrame["lineNumber"] = callFrame.getLineNumber();
}
if (callFrame.hasColumnNumber()) {
dynamicCallFrame["columnNumber"] = callFrame.getColumnNumber();
}

folly::dynamic dynamicNode = folly::dynamic::object();
dynamicNode["callFrame"] = dynamicCallFrame;
dynamicNode["id"] = node.getId();
if (node.hasParentId()) {
dynamicNode["parent"] = node.getParentId();
}

dynamicNodes.push_back(dynamicNode);
}
folly::dynamic dynamicSamples = folly::dynamic::array();
for (const uint64_t sampleRootNodeId : profileChunk.getSamples()) {
dynamicSamples.push_back(sampleRootNodeId);
}
folly::dynamic dynamicCpuProfile =
folly::dynamic::object("nodes", dynamicNodes)("samples", dynamicSamples);

folly::dynamic dynamicTimeDeltas = folly::dynamic::array();
for (const uint64_t timeDelta : profileChunk.getTimeDeltas()) {
dynamicTimeDeltas.push_back(timeDelta);
}

buffer_.push_back(TraceEvent{
.name = "ProfileChunk",
.cat = "disabled-by-default-v8.cpu_profiler",
.ph = 'P',
.ts = profileChunk.getTimestamp(),
.pid = profileChunk.getProcessId(),
.tid = profileChunk.getThreadId(),
.args = folly::dynamic::object(
"data",
folly ::dynamic::object("cpuProfile", dynamicCpuProfile)(
"timeDeltas", dynamicTimeDeltas)),
});

//
}

void PerformanceTracer::populateTraceEventsForHermes(
const std::vector<
facebook::hermes::sampling_profiler::TraceEventCollectionDataSource>&
dataSources) {
for (const facebook::hermes::sampling_profiler::
TraceEventCollectionDataSource& source : dataSources) {
processSingleHermesDataSource(source);
}
}

bool PerformanceTracer::stopTracingAndCollectEvents(
const std::function<void(const folly::dynamic& eventsChunk)>&
resultCallback) {
Expand All @@ -47,6 +177,14 @@ bool PerformanceTracer::stopTracingAndCollectEvents(
return false;
}

facebook::hermes::HermesRuntime::disableSamplingProfiler();
std::vector<
facebook::hermes::sampling_profiler::TraceEventCollectionDataSource>
javascriptSamplingsDataSources = facebook::hermes::HermesRuntime::
dumpSampledTraceAsTraceEventCollectionDataSource();

populateTraceEventsForHermes(std::move(javascriptSamplingsDataSources));

tracing_ = false;
if (buffer_.empty()) {
customTrackIdMap_.clear();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
#include "CdpTracing.h"

#include <folly/dynamic.h>
#include <hermes/hermes.h>

#include <functional>
#include <optional>
Expand Down Expand Up @@ -117,7 +118,16 @@ class PerformanceTracer {

folly::dynamic serializeTraceEvent(TraceEvent event) const;

void populateTraceEventsForHermes(
const std::vector<
facebook::hermes::sampling_profiler::TraceEventCollectionDataSource>&
dataSources);
void processSingleHermesDataSource(
const facebook::hermes::sampling_profiler::TraceEventCollectionDataSource&
dataSource);

bool tracing_{false};
uint64_t tracingStartedTimestamp_{};
std::unordered_map<std::string, uint64_t> customTrackIdMap_;
std::vector<TraceEvent> buffer_;
std::mutex mutex_;
Expand Down

0 comments on commit 19190f2

Please sign in to comment.