blob: 109fae18d7f32ae2757b1259b19ea57db65f6bb0 [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 "extensions/renderer/object_backed_native_handler.h"
#include <stddef.h>
#include "base/logging.h"
#include "base/memory/linked_ptr.h"
#include "extensions/renderer/console.h"
#include "extensions/renderer/module_system.h"
#include "extensions/renderer/script_context.h"
#include "extensions/renderer/script_context_set.h"
#include "v8/include/v8.h"
namespace extensions {
namespace {
// Key for the base::Bound routed function.
const char* kHandlerFunction = "handler_function";
} // namespace
ObjectBackedNativeHandler::ObjectBackedNativeHandler(ScriptContext* context)
: router_data_(context->isolate()),
context_(context),
object_template_(context->isolate(),
v8::ObjectTemplate::New(context->isolate())) {
}
ObjectBackedNativeHandler::~ObjectBackedNativeHandler() {
}
v8::Local<v8::Object> ObjectBackedNativeHandler::NewInstance() {
return v8::Local<v8::ObjectTemplate>::New(GetIsolate(), object_template_)
->NewInstance();
}
// static
void ObjectBackedNativeHandler::Router(
const v8::FunctionCallbackInfo<v8::Value>& args) {
v8::Isolate* isolate = args.GetIsolate();
v8::HandleScope handle_scope(isolate);
v8::Local<v8::Object> data = args.Data().As<v8::Object>();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
v8::Local<v8::Value> handler_function_value;
// See comment in header file for why we do this.
if (!GetPrivate(context, data, kHandlerFunction, &handler_function_value) ||
handler_function_value->IsUndefined()) {
ScriptContext* script_context =
ScriptContextSet::GetContextByV8Context(context);
console::Error(script_context ? script_context->GetRenderFrame() : nullptr,
"Extension view no longer exists");
return;
}
// This CHECK is *important*. Otherwise, we'll go around happily executing
// something random. See crbug.com/548273.
CHECK(handler_function_value->IsExternal());
static_cast<HandlerFunction*>(
handler_function_value.As<v8::External>()->Value())->Run(args);
}
void ObjectBackedNativeHandler::RouteFunction(
const std::string& name,
const HandlerFunction& handler_function) {
v8::Isolate* isolate = v8::Isolate::GetCurrent();
v8::HandleScope handle_scope(isolate);
v8::Context::Scope context_scope(context_->v8_context());
v8::Local<v8::Object> data = v8::Object::New(isolate);
SetPrivate(data, kHandlerFunction,
v8::External::New(isolate, new HandlerFunction(handler_function)));
v8::Local<v8::FunctionTemplate> function_template =
v8::FunctionTemplate::New(isolate, Router, data);
v8::Local<v8::ObjectTemplate>::New(isolate, object_template_)
->Set(isolate, name.c_str(), function_template);
router_data_.Append(data);
}
v8::Isolate* ObjectBackedNativeHandler::GetIsolate() const {
return context_->isolate();
}
void ObjectBackedNativeHandler::Invalidate() {
v8::Isolate* isolate = GetIsolate();
v8::HandleScope handle_scope(isolate);
v8::Context::Scope context_scope(context_->v8_context());
for (size_t i = 0; i < router_data_.Size(); i++) {
v8::Local<v8::Object> data = router_data_.Get(i);
v8::Local<v8::Value> handler_function_value;
CHECK(GetPrivate(data, kHandlerFunction, &handler_function_value));
delete static_cast<HandlerFunction*>(
handler_function_value.As<v8::External>()->Value());
DeletePrivate(data, kHandlerFunction);
}
router_data_.Clear();
object_template_.Reset();
NativeHandler::Invalidate();
}
void ObjectBackedNativeHandler::SetPrivate(v8::Local<v8::Object> obj,
const char* key,
v8::Local<v8::Value> value) {
SetPrivate(context_->v8_context(), obj, key, value);
}
// static
void ObjectBackedNativeHandler::SetPrivate(v8::Local<v8::Context> context,
v8::Local<v8::Object> obj,
const char* key,
v8::Local<v8::Value> value) {
obj->SetPrivate(context, v8::Private::ForApi(context->GetIsolate(),
v8::String::NewFromUtf8(
context->GetIsolate(), key)),
value)
.FromJust();
}
bool ObjectBackedNativeHandler::GetPrivate(v8::Local<v8::Object> obj,
const char* key,
v8::Local<v8::Value>* result) {
return GetPrivate(context_->v8_context(), obj, key, result);
}
// static
bool ObjectBackedNativeHandler::GetPrivate(v8::Local<v8::Context> context,
v8::Local<v8::Object> obj,
const char* key,
v8::Local<v8::Value>* result) {
return obj->GetPrivate(context,
v8::Private::ForApi(context->GetIsolate(),
v8::String::NewFromUtf8(
context->GetIsolate(), key)))
.ToLocal(result);
}
void ObjectBackedNativeHandler::DeletePrivate(v8::Local<v8::Object> obj,
const char* key) {
DeletePrivate(context_->v8_context(), obj, key);
}
// static
void ObjectBackedNativeHandler::DeletePrivate(v8::Local<v8::Context> context,
v8::Local<v8::Object> obj,
const char* key) {
obj->DeletePrivate(context,
v8::Private::ForApi(
context->GetIsolate(),
v8::String::NewFromUtf8(context->GetIsolate(), key)))
.FromJust();
}
} // namespace extensions