// 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::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(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(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