// Copyright (c) 2012 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. // TODO(dcarney): Remove this when UnsafePersistent is removed. #define V8_ALLOW_ACCESS_TO_RAW_HANDLE_CONSTRUCTOR #include "chrome/renderer/extensions/object_backed_native_handler.h" #include "base/logging.h" #include "base/memory/linked_ptr.h" #include "chrome/renderer/extensions/chrome_v8_context.h" #include "chrome/renderer/extensions/console.h" #include "chrome/renderer/extensions/module_system.h" #include "v8/include/v8.h" namespace extensions { namespace { // Key for the base::Bound routed function. const char* kHandlerFunction = "handler_function"; } // namespace ObjectBackedNativeHandler::ObjectBackedNativeHandler( ChromeV8Context* context) : context_(context), object_template_(v8::ObjectTemplate::New()) { } ObjectBackedNativeHandler::~ObjectBackedNativeHandler() { Invalidate(); } v8::Handle ObjectBackedNativeHandler::NewInstance() { return object_template_.NewHandle(v8::Isolate::GetCurrent())->NewInstance(); } // static void ObjectBackedNativeHandler::Router( const v8::FunctionCallbackInfo& args) { v8::HandleScope handle_scope(args.GetIsolate()); v8::Handle data = args.Data().As(); v8::Handle handler_function_value = data->Get(v8::String::New(kHandlerFunction)); // See comment in header file for why we do this. if (handler_function_value.IsEmpty() || handler_function_value->IsUndefined()) { console::Error(v8::Context::GetCalling(), "Extension view no longer exists"); return; } DCHECK(handler_function_value->IsExternal()); static_cast( handler_function_value.As()->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::Persistent data(isolate, v8::Object::New()); v8::Local local_data = v8::Local::New(isolate, data); local_data->Set(v8::String::New(kHandlerFunction), v8::External::New(new HandlerFunction(handler_function))); v8::Handle function_template = v8::FunctionTemplate::New(Router, local_data); object_template_.NewHandle(isolate)->Set(name.c_str(), function_template); router_data_.push_back(UnsafePersistent(&data)); } void ObjectBackedNativeHandler::Invalidate() { if (!is_valid()) return; v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::HandleScope handle_scope(isolate); v8::Context::Scope context_scope(context_->v8_context()); for (RouterData::iterator it = router_data_.begin(); it != router_data_.end(); ++it) { v8::Handle data = it->newLocal(isolate); v8::Handle handler_function_value = data->Get(v8::String::New(kHandlerFunction)); CHECK(!handler_function_value.IsEmpty()); delete static_cast( handler_function_value.As()->Value()); data->Delete(v8::String::New(kHandlerFunction)); it->dispose(); } object_template_.reset(); context_ = NULL; NativeHandler::Invalidate(); } } // namespace extensions