summaryrefslogtreecommitdiff
path: root/chromium/extensions/renderer/object_backed_native_handler.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/extensions/renderer/object_backed_native_handler.cc')
-rw-r--r--chromium/extensions/renderer/object_backed_native_handler.cc222
1 files changed, 222 insertions, 0 deletions
diff --git a/chromium/extensions/renderer/object_backed_native_handler.cc b/chromium/extensions/renderer/object_backed_native_handler.cc
new file mode 100644
index 00000000000..1ffc355cc48
--- /dev/null
+++ b/chromium/extensions/renderer/object_backed_native_handler.cc
@@ -0,0 +1,222 @@
+// 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 "content/public/child/worker_thread.h"
+#include "extensions/common/extension_api.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 "extensions/renderer/v8_helpers.h"
+#include "third_party/WebKit/public/web/WebLocalFrame.h"
+#include "v8/include/v8.h"
+
+namespace extensions {
+
+namespace {
+// Key for the base::Bound routed function.
+const char* kHandlerFunction = "handler_function";
+const char* kFeatureName = "feature_name";
+} // 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;
+ v8::Local<v8::Value> feature_name_value;
+ // See comment in header file for why we do this.
+ if (!GetPrivate(context, data, kHandlerFunction, &handler_function_value) ||
+ handler_function_value->IsUndefined() ||
+ !GetPrivate(context, data, kFeatureName, &feature_name_value) ||
+ !feature_name_value->IsString()) {
+ ScriptContext* script_context =
+ ScriptContextSet::GetContextByV8Context(context);
+ console::Error(script_context ? script_context->GetRenderFrame() : nullptr,
+ "Extension view no longer exists");
+ return;
+ }
+
+ // We can't access the ScriptContextSet on a worker thread. Luckily, we also
+ // don't inject many bindings into worker threads.
+ // TODO(devlin): Figure out a way around this.
+ if (content::WorkerThread::GetCurrentId() == 0) {
+ ScriptContext* script_context =
+ ScriptContextSet::GetContextByV8Context(context);
+ v8::Local<v8::String> feature_name_string =
+ feature_name_value->ToString(context).ToLocalChecked();
+ std::string feature_name = *v8::String::Utf8Value(feature_name_string);
+ // TODO(devlin): Eventually, we should fail if either script_context is null
+ // or feature_name is empty.
+ if (script_context &&
+ !feature_name.empty() &&
+ !script_context->GetAvailability(feature_name).is_available()) {
+ 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);
+
+ // Verify that the return value, if any, is accessible by the context.
+ v8::ReturnValue<v8::Value> ret = args.GetReturnValue();
+ v8::Local<v8::Value> ret_value = ret.Get();
+ if (ret_value->IsObject() && !ret_value->IsNull() &&
+ !ContextCanAccessObject(context, v8::Local<v8::Object>::Cast(ret_value),
+ true)) {
+ NOTREACHED() << "Insecure return value";
+ ret.SetUndefined();
+ }
+}
+
+void ObjectBackedNativeHandler::RouteFunction(
+ const std::string& name,
+ const HandlerFunction& handler_function) {
+ RouteFunction(name, "", handler_function);
+}
+
+void ObjectBackedNativeHandler::RouteFunction(
+ const std::string& name,
+ const std::string& feature_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)));
+ DCHECK(feature_name.empty() ||
+ ExtensionAPI::GetSharedInstance()->GetFeatureDependency(feature_name))
+ << feature_name;
+ SetPrivate(data, kFeatureName,
+ v8_helpers::ToV8StringUnsafe(isolate, feature_name));
+ 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();
+}
+
+// static
+bool ObjectBackedNativeHandler::ContextCanAccessObject(
+ const v8::Local<v8::Context>& context,
+ const v8::Local<v8::Object>& object,
+ bool allow_null_context) {
+ if (object->IsNull())
+ return true;
+ if (context == object->CreationContext())
+ return true;
+ ScriptContext* other_script_context =
+ ScriptContextSet::GetContextByObject(object);
+ if (!other_script_context || !other_script_context->web_frame())
+ return allow_null_context;
+
+ return blink::WebFrame::scriptCanAccess(other_script_context->web_frame());
+}
+
+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