2016-05-03 19:29:58 -07:00
|
|
|
// Copyright 2004-present Facebook. All Rights Reserved.
|
|
|
|
|
|
|
|
#include "JSCPerfStats.h"
|
|
|
|
|
|
|
|
#ifdef JSC_HAS_PERF_STATS_API
|
|
|
|
|
2017-08-03 13:08:21 -07:00
|
|
|
#include <cstdint>
|
|
|
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/resource.h>
|
|
|
|
|
2016-05-03 19:29:58 -07:00
|
|
|
#include <JavaScriptCore/JSPerfStats.h>
|
2016-11-01 11:38:43 -07:00
|
|
|
#include <jschelpers/JSCHelpers.h>
|
|
|
|
#include <jschelpers/Value.h>
|
2016-05-03 19:29:58 -07:00
|
|
|
|
2016-11-18 06:25:24 -08:00
|
|
|
using namespace facebook::react;
|
|
|
|
|
2017-08-03 13:08:21 -07:00
|
|
|
static uint64_t toMillis(struct timeval tv) {
|
|
|
|
return tv.tv_sec * 1000ULL + tv.tv_usec / 1000ULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static JSValueRef nativeGetProcessPerfStats(
|
|
|
|
JSContextRef ctx,
|
|
|
|
JSObjectRef function,
|
|
|
|
JSObjectRef thisObject,
|
|
|
|
size_t argumentCount,
|
|
|
|
const JSValueRef arguments[],
|
|
|
|
JSValueRef* exception) {
|
|
|
|
struct rusage usage{};
|
|
|
|
if (getrusage(RUSAGE_SELF, &usage) != 0) {
|
|
|
|
return Value::makeUndefined(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto result = Object::create(ctx);
|
|
|
|
uint64_t cpu_time_ms = toMillis(usage.ru_utime) + toMillis(usage.ru_stime);
|
|
|
|
result.setProperty("major_faults", Value::makeNumber(ctx, usage.ru_majflt));
|
|
|
|
result.setProperty("minor_faults", Value::makeNumber(ctx, usage.ru_minflt));
|
|
|
|
result.setProperty("cpu_time_ms", Value::makeNumber(ctx, cpu_time_ms));
|
|
|
|
result.setProperty("input_blocks", Value::makeNumber(ctx, usage.ru_inblock));
|
|
|
|
result.setProperty("output_blocks", Value::makeNumber(ctx, usage.ru_oublock));
|
|
|
|
return static_cast<JSObjectRef>(result);
|
|
|
|
}
|
|
|
|
|
2016-05-03 19:29:58 -07:00
|
|
|
static JSValueRef nativeGetHeapStats(
|
|
|
|
JSContextRef ctx,
|
|
|
|
JSObjectRef function,
|
|
|
|
JSObjectRef thisObject,
|
|
|
|
size_t argumentCount,
|
|
|
|
const JSValueRef arguments[],
|
|
|
|
JSValueRef* exception) {
|
|
|
|
JSHeapStats heapStats = {0};
|
|
|
|
JSGetHeapStats(ctx, &heapStats);
|
|
|
|
|
2017-07-25 04:45:08 -07:00
|
|
|
auto result = Object::create(ctx);
|
|
|
|
result.setProperty("size", Value::makeNumber(ctx, heapStats.size));
|
|
|
|
result.setProperty("extra_size", Value::makeNumber(ctx, heapStats.extraSize));
|
|
|
|
result.setProperty("capacity", Value::makeNumber(ctx, heapStats.capacity));
|
|
|
|
result.setProperty("object_count", Value::makeNumber(ctx, heapStats.objectCount));
|
|
|
|
result.setProperty("object_size", Value::makeNumber(ctx, heapStats.objectSizeAfterLastCollect));
|
|
|
|
result.setProperty("object_capacity", Value::makeNumber(ctx, heapStats.objectCapacityAfterLastCollect));
|
|
|
|
result.setProperty("block_size", Value::makeNumber(ctx, heapStats.blockSize));
|
|
|
|
result.setProperty("malloc_size", Value::makeNumber(ctx, heapStats.mallocSize));
|
|
|
|
return static_cast<JSObjectRef>(result);
|
2016-05-03 19:29:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static JSValueRef nativeGetGCStats(
|
|
|
|
JSContextRef ctx,
|
|
|
|
JSObjectRef function,
|
|
|
|
JSObjectRef thisObject,
|
|
|
|
size_t argumentCount,
|
|
|
|
const JSValueRef arguments[],
|
|
|
|
JSValueRef* exception) {
|
|
|
|
JSGCStats gcStats = {0};
|
|
|
|
JSGetGCStats(ctx, &gcStats);
|
|
|
|
|
2017-07-25 04:45:08 -07:00
|
|
|
auto result = Object::create(ctx);
|
|
|
|
result.setProperty("last_full_gc_length", Value::makeNumber(ctx, gcStats.lastFullGCLength));
|
|
|
|
result.setProperty("last_eden_gc_length", Value::makeNumber(ctx, gcStats.lastEdenGCLength));
|
|
|
|
return static_cast<JSObjectRef>(result);
|
2016-05-03 19:29:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace facebook {
|
|
|
|
namespace react {
|
|
|
|
|
|
|
|
void addJSCPerfStatsHooks(JSGlobalContextRef ctx) {
|
|
|
|
#ifdef JSC_HAS_PERF_STATS_API
|
2017-08-03 13:08:21 -07:00
|
|
|
installGlobalFunction(ctx, "nativeGetProcessPerfStats", nativeGetProcessPerfStats);
|
2016-05-03 19:29:58 -07:00
|
|
|
installGlobalFunction(ctx, "nativeGetHeapStats", nativeGetHeapStats);
|
|
|
|
installGlobalFunction(ctx, "nativeGetGCStats", nativeGetGCStats);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
} }
|