blob: 556f7fc5f24f0b1c924138a6b77709e7aea5bc5d [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "gin/public/v8_platform.h"
#include "base/bind.h"
#include "base/location.h"
#include "base/sys_info.h"
#include "base/threading/worker_pool.h"
#include "base/trace_event/trace_event.h"
#include "gin/per_isolate_data.h"
namespace gin {
namespace {
base::LazyInstance<V8Platform>::Leaky g_v8_platform = LAZY_INSTANCE_INITIALIZER;
} // namespace
// static
V8Platform* V8Platform::Get() { return g_v8_platform.Pointer(); }
V8Platform::V8Platform() {}
V8Platform::~V8Platform() {}
size_t V8Platform::NumberOfAvailableBackgroundThreads() {
// WorkerPool will currently always create additional threads for posted
// background tasks, unless there are threads sitting idle (on posix).
// Indicate that V8 should create no more than the number of cores available,
// reserving one core for the main thread.
const size_t available_cores =
static_cast<size_t>(base::SysInfo::NumberOfProcessors());
if (available_cores > 1) {
return available_cores - 1;
}
return 1;
}
void V8Platform::CallOnBackgroundThread(
v8::Task* task,
v8::Platform::ExpectedRuntime expected_runtime) {
base::WorkerPool::PostTask(
FROM_HERE,
base::Bind(&v8::Task::Run, base::Owned(task)),
expected_runtime == v8::Platform::kLongRunningTask);
}
void V8Platform::CallOnForegroundThread(v8::Isolate* isolate, v8::Task* task) {
PerIsolateData::From(isolate)->task_runner()->PostTask(
FROM_HERE, base::Bind(&v8::Task::Run, base::Owned(task)));
}
void V8Platform::CallDelayedOnForegroundThread(v8::Isolate* isolate,
v8::Task* task,
double delay_in_seconds) {
PerIsolateData::From(isolate)->task_runner()->PostDelayedTask(
FROM_HERE, base::Bind(&v8::Task::Run, base::Owned(task)),
base::TimeDelta::FromSecondsD(delay_in_seconds));
}
void V8Platform::CallIdleOnForegroundThread(v8::Isolate* isolate,
v8::IdleTask* task) {
DCHECK(PerIsolateData::From(isolate)->idle_task_runner());
PerIsolateData::From(isolate)->idle_task_runner()->PostIdleTask(task);
}
bool V8Platform::IdleTasksEnabled(v8::Isolate* isolate) {
return PerIsolateData::From(isolate)->idle_task_runner() != nullptr;
}
double V8Platform::MonotonicallyIncreasingTime() {
return base::TimeTicks::Now().ToInternalValue() /
static_cast<double>(base::Time::kMicrosecondsPerSecond);
}
const uint8_t* V8Platform::GetCategoryGroupEnabled(const char* name) {
return TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED(name);
}
const char* V8Platform::GetCategoryGroupName(
const uint8_t* category_enabled_flag) {
return base::trace_event::TraceLog::GetCategoryGroupName(
category_enabled_flag);
}
uint64_t V8Platform::AddTraceEvent(char phase,
const uint8_t* category_enabled_flag,
const char* name,
const char* scope,
uint64_t id,
uint64_t bind_id,
int32_t num_args,
const char** arg_names,
const uint8_t* arg_types,
const uint64_t* arg_values,
unsigned int flags) {
base::trace_event::TraceEventHandle handle =
TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_BIND_ID(
phase, category_enabled_flag, name, scope, id, bind_id, num_args,
arg_names, arg_types, (const long long unsigned int*)arg_values, NULL,
flags);
uint64_t result;
memcpy(&result, &handle, sizeof(result));
return result;
}
void V8Platform::UpdateTraceEventDuration(const uint8_t* category_enabled_flag,
const char* name,
uint64_t handle) {
base::trace_event::TraceEventHandle traceEventHandle;
memcpy(&traceEventHandle, &handle, sizeof(handle));
TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION(category_enabled_flag, name,
traceEventHandle);
}
} // namespace gin