| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/renderer/extensions/messaging_bindings.h" | 5 #include "chrome/renderer/extensions/messaging_bindings.h" |
| 6 | 6 |
| 7 #include <map> | 7 #include <map> |
| 8 #include <string> | 8 #include <string> |
| 9 | 9 |
| 10 #include "base/basictypes.h" | 10 #include "base/basictypes.h" |
| (...skipping 29 matching lines...) Expand all Loading... |
| 40 // var port = runtime.connect(); | 40 // var port = runtime.connect(); |
| 41 // port.postMessage('Can you hear me now?'); | 41 // port.postMessage('Can you hear me now?'); |
| 42 // port.onmessage.addListener(function(msg, port) { | 42 // port.onmessage.addListener(function(msg, port) { |
| 43 // alert('response=' + msg); | 43 // alert('response=' + msg); |
| 44 // port.postMessage('I got your reponse'); | 44 // port.postMessage('I got your reponse'); |
| 45 // }); | 45 // }); |
| 46 | 46 |
| 47 using content::RenderThread; | 47 using content::RenderThread; |
| 48 using content::V8ValueConverter; | 48 using content::V8ValueConverter; |
| 49 | 49 |
| 50 namespace extensions { |
| 51 |
| 50 namespace { | 52 namespace { |
| 51 | 53 |
| 52 struct ExtensionData { | 54 struct ExtensionData { |
| 53 struct PortData { | 55 struct PortData { |
| 54 int ref_count; // how many contexts have a handle to this port | 56 int ref_count; // how many contexts have a handle to this port |
| 55 PortData() : ref_count(0) {} | 57 PortData() : ref_count(0) {} |
| 56 }; | 58 }; |
| 57 std::map<int, PortData> ports; // port ID -> data | 59 std::map<int, PortData> ports; // port ID -> data |
| 58 }; | 60 }; |
| 59 | 61 |
| 60 static base::LazyInstance<ExtensionData> g_extension_data = | 62 base::LazyInstance<ExtensionData> g_extension_data = |
| 61 LAZY_INSTANCE_INITIALIZER; | 63 LAZY_INSTANCE_INITIALIZER; |
| 62 | 64 |
| 63 static bool HasPortData(int port_id) { | 65 bool HasPortData(int port_id) { |
| 64 return g_extension_data.Get().ports.find(port_id) != | 66 return g_extension_data.Get().ports.find(port_id) != |
| 65 g_extension_data.Get().ports.end(); | 67 g_extension_data.Get().ports.end(); |
| 66 } | 68 } |
| 67 | 69 |
| 68 static ExtensionData::PortData& GetPortData(int port_id) { | 70 ExtensionData::PortData& GetPortData(int port_id) { |
| 69 return g_extension_data.Get().ports[port_id]; | 71 return g_extension_data.Get().ports[port_id]; |
| 70 } | 72 } |
| 71 | 73 |
| 72 static void ClearPortData(int port_id) { | 74 void ClearPortData(int port_id) { |
| 73 g_extension_data.Get().ports.erase(port_id); | 75 g_extension_data.Get().ports.erase(port_id); |
| 74 } | 76 } |
| 75 | 77 |
| 76 const char kPortClosedError[] = "Attempting to use a disconnected port object"; | 78 const char kPortClosedError[] = "Attempting to use a disconnected port object"; |
| 77 const char kReceivingEndDoesntExistError[] = | 79 const char kReceivingEndDoesntExistError[] = |
| 78 "Could not establish connection. Receiving end does not exist."; | 80 "Could not establish connection. Receiving end does not exist."; |
| 79 | 81 |
| 80 class ExtensionImpl : public extensions::ChromeV8Extension { | 82 class ExtensionImpl : public ChromeV8Extension { |
| 81 public: | 83 public: |
| 82 explicit ExtensionImpl(extensions::Dispatcher* dispatcher, | 84 ExtensionImpl(Dispatcher* dispatcher, ChromeV8Context* context) |
| 83 extensions::ChromeV8Context* context) | 85 : ChromeV8Extension(dispatcher, context) { |
| 84 : extensions::ChromeV8Extension(dispatcher, context) { | |
| 85 RouteFunction("CloseChannel", | 86 RouteFunction("CloseChannel", |
| 86 base::Bind(&ExtensionImpl::CloseChannel, base::Unretained(this))); | 87 base::Bind(&ExtensionImpl::CloseChannel, base::Unretained(this))); |
| 87 RouteFunction("PortAddRef", | 88 RouteFunction("PortAddRef", |
| 88 base::Bind(&ExtensionImpl::PortAddRef, base::Unretained(this))); | 89 base::Bind(&ExtensionImpl::PortAddRef, base::Unretained(this))); |
| 89 RouteFunction("PortRelease", | 90 RouteFunction("PortRelease", |
| 90 base::Bind(&ExtensionImpl::PortRelease, base::Unretained(this))); | 91 base::Bind(&ExtensionImpl::PortRelease, base::Unretained(this))); |
| 91 RouteFunction("PostMessage", | 92 RouteFunction("PostMessage", |
| 92 base::Bind(&ExtensionImpl::PostMessage, base::Unretained(this))); | 93 base::Bind(&ExtensionImpl::PostMessage, base::Unretained(this))); |
| 93 // TODO(fsamuel, kalman): Move BindToGC out of messaging natives. | 94 // TODO(fsamuel, kalman): Move BindToGC out of messaging natives. |
| 94 RouteFunction("BindToGC", | 95 RouteFunction("BindToGC", |
| 95 base::Bind(&ExtensionImpl::BindToGC, base::Unretained(this))); | 96 base::Bind(&ExtensionImpl::BindToGC, base::Unretained(this))); |
| 96 } | 97 } |
| 97 | 98 |
| 98 virtual ~ExtensionImpl() {} | 99 virtual ~ExtensionImpl() {} |
| 99 | 100 |
| 101 void ClearPortDataAndNotifyDispatcher(int port_id) { |
| 102 ClearPortData(port_id); |
| 103 dispatcher()->ClearPortData(port_id); |
| 104 } |
| 105 |
| 100 // Sends a message along the given channel. | 106 // Sends a message along the given channel. |
| 101 void PostMessage(const v8::FunctionCallbackInfo<v8::Value>& args) { | 107 void PostMessage(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 102 content::RenderView* renderview = GetRenderView(); | 108 content::RenderView* renderview = GetRenderView(); |
| 103 if (!renderview) | 109 if (!renderview) |
| 104 return; | 110 return; |
| 105 | 111 |
| 106 // Arguments are (int32 port_id, string message). | 112 // Arguments are (int32 port_id, string message). |
| 107 CHECK(args.Length() == 2 && | 113 CHECK(args.Length() == 2 && |
| 108 args[0]->IsInt32() && | 114 args[0]->IsInt32() && |
| 109 args[1]->IsString()); | 115 args[1]->IsString()); |
| 110 | 116 |
| 111 int port_id = args[0]->Int32Value(); | 117 int port_id = args[0]->Int32Value(); |
| 112 if (!HasPortData(port_id)) { | 118 if (!HasPortData(port_id)) { |
| 113 args.GetIsolate()->ThrowException(v8::Exception::Error( | 119 args.GetIsolate()->ThrowException(v8::Exception::Error( |
| 114 v8::String::NewFromUtf8(args.GetIsolate(), kPortClosedError))); | 120 v8::String::NewFromUtf8(args.GetIsolate(), kPortClosedError))); |
| 115 return; | 121 return; |
| 116 } | 122 } |
| 117 | 123 |
| 118 renderview->Send(new ExtensionHostMsg_PostMessage( | 124 renderview->Send(new ExtensionHostMsg_PostMessage( |
| 119 renderview->GetRoutingID(), port_id, | 125 renderview->GetRoutingID(), port_id, |
| 120 extensions::Message( | 126 Message(*v8::String::Utf8Value(args[1]), |
| 121 *v8::String::Utf8Value(args[1]), | 127 blink::WebUserGestureIndicator::isProcessingUserGesture()))); |
| 122 blink::WebUserGestureIndicator::isProcessingUserGesture()))); | |
| 123 } | 128 } |
| 124 | 129 |
| 125 // Forcefully disconnects a port. | 130 // Forcefully disconnects a port. |
| 126 void CloseChannel(const v8::FunctionCallbackInfo<v8::Value>& args) { | 131 void CloseChannel(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 127 // Arguments are (int32 port_id, boolean notify_browser). | 132 // Arguments are (int32 port_id, boolean notify_browser). |
| 128 CHECK_EQ(2, args.Length()); | 133 CHECK_EQ(2, args.Length()); |
| 129 CHECK(args[0]->IsInt32()); | 134 CHECK(args[0]->IsInt32()); |
| 130 CHECK(args[1]->IsBoolean()); | 135 CHECK(args[1]->IsBoolean()); |
| 131 | 136 |
| 132 int port_id = args[0]->Int32Value(); | 137 int port_id = args[0]->Int32Value(); |
| 133 if (!HasPortData(port_id)) | 138 if (!HasPortData(port_id)) |
| 134 return; | 139 return; |
| 135 | 140 |
| 136 // Send via the RenderThread because the RenderView might be closing. | 141 // Send via the RenderThread because the RenderView might be closing. |
| 137 bool notify_browser = args[1]->BooleanValue(); | 142 bool notify_browser = args[1]->BooleanValue(); |
| 138 if (notify_browser) { | 143 if (notify_browser) { |
| 139 content::RenderThread::Get()->Send( | 144 content::RenderThread::Get()->Send( |
| 140 new ExtensionHostMsg_CloseChannel(port_id, std::string())); | 145 new ExtensionHostMsg_CloseChannel(port_id, std::string())); |
| 141 } | 146 } |
| 142 | 147 |
| 143 ClearPortData(port_id); | 148 ClearPortDataAndNotifyDispatcher(port_id); |
| 144 } | 149 } |
| 145 | 150 |
| 146 // A new port has been created for a context. This occurs both when script | 151 // A new port has been created for a context. This occurs both when script |
| 147 // opens a connection, and when a connection is opened to this script. | 152 // opens a connection, and when a connection is opened to this script. |
| 148 void PortAddRef(const v8::FunctionCallbackInfo<v8::Value>& args) { | 153 void PortAddRef(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 149 // Arguments are (int32 port_id). | 154 // Arguments are (int32 port_id). |
| 150 CHECK_EQ(1, args.Length()); | 155 CHECK_EQ(1, args.Length()); |
| 151 CHECK(args[0]->IsInt32()); | 156 CHECK(args[0]->IsInt32()); |
| 152 | 157 |
| 153 int port_id = args[0]->Int32Value(); | 158 int port_id = args[0]->Int32Value(); |
| 154 ++GetPortData(port_id).ref_count; | 159 ++GetPortData(port_id).ref_count; |
| 155 } | 160 } |
| 156 | 161 |
| 157 // The frame a port lived in has been destroyed. When there are no more | 162 // The frame a port lived in has been destroyed. When there are no more |
| 158 // frames with a reference to a given port, we will disconnect it and notify | 163 // frames with a reference to a given port, we will disconnect it and notify |
| 159 // the other end of the channel. | 164 // the other end of the channel. |
| 160 void PortRelease(const v8::FunctionCallbackInfo<v8::Value>& args) { | 165 void PortRelease(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 161 // Arguments are (int32 port_id). | 166 // Arguments are (int32 port_id). |
| 162 CHECK_EQ(1, args.Length()); | 167 CHECK_EQ(1, args.Length()); |
| 163 CHECK(args[0]->IsInt32()); | 168 CHECK(args[0]->IsInt32()); |
| 164 | 169 |
| 165 int port_id = args[0]->Int32Value(); | 170 int port_id = args[0]->Int32Value(); |
| 166 if (HasPortData(port_id) && --GetPortData(port_id).ref_count == 0) { | 171 if (HasPortData(port_id) && --GetPortData(port_id).ref_count == 0) { |
| 167 // Send via the RenderThread because the RenderView might be closing. | 172 // Send via the RenderThread because the RenderView might be closing. |
| 168 content::RenderThread::Get()->Send( | 173 content::RenderThread::Get()->Send( |
| 169 new ExtensionHostMsg_CloseChannel(port_id, std::string())); | 174 new ExtensionHostMsg_CloseChannel(port_id, std::string())); |
| 170 ClearPortData(port_id); | 175 ClearPortDataAndNotifyDispatcher(port_id); |
| 171 } | 176 } |
| 172 } | 177 } |
| 173 | 178 |
| 174 // Holds a |callback| to run sometime after |object| is GC'ed. |callback| will | 179 // Holds a |callback| to run sometime after |object| is GC'ed. |callback| will |
| 175 // not be executed re-entrantly to avoid running JS in an unexpected state. | 180 // not be executed re-entrantly to avoid running JS in an unexpected state. |
| 176 class GCCallback { | 181 class GCCallback { |
| 177 public: | 182 public: |
| 178 static void Bind(v8::Handle<v8::Object> object, | 183 static void Bind(v8::Handle<v8::Object> object, |
| 179 v8::Handle<v8::Function> callback, | 184 v8::Handle<v8::Function> callback, |
| 180 v8::Isolate* isolate) { | 185 v8::Isolate* isolate) { |
| (...skipping 22 matching lines...) Expand all Loading... |
| 203 v8::HandleScope handle_scope(isolate_); | 208 v8::HandleScope handle_scope(isolate_); |
| 204 v8::Handle<v8::Function> callback = callback_.NewHandle(isolate_); | 209 v8::Handle<v8::Function> callback = callback_.NewHandle(isolate_); |
| 205 v8::Handle<v8::Context> context = callback->CreationContext(); | 210 v8::Handle<v8::Context> context = callback->CreationContext(); |
| 206 if (context.IsEmpty()) | 211 if (context.IsEmpty()) |
| 207 return; | 212 return; |
| 208 v8::Context::Scope context_scope(context); | 213 v8::Context::Scope context_scope(context); |
| 209 blink::WebScopedMicrotaskSuppression suppression; | 214 blink::WebScopedMicrotaskSuppression suppression; |
| 210 callback->Call(context->Global(), 0, NULL); | 215 callback->Call(context->Global(), 0, NULL); |
| 211 } | 216 } |
| 212 | 217 |
| 213 extensions::ScopedPersistent<v8::Object> object_; | 218 ScopedPersistent<v8::Object> object_; |
| 214 extensions::ScopedPersistent<v8::Function> callback_; | 219 ScopedPersistent<v8::Function> callback_; |
| 215 v8::Isolate* isolate_; | 220 v8::Isolate* isolate_; |
| 216 | 221 |
| 217 DISALLOW_COPY_AND_ASSIGN(GCCallback); | 222 DISALLOW_COPY_AND_ASSIGN(GCCallback); |
| 218 }; | 223 }; |
| 219 | 224 |
| 220 // void BindToGC(object, callback) | 225 // void BindToGC(object, callback) |
| 221 // | 226 // |
| 222 // Binds |callback| to be invoked *sometime after* |object| is garbage | 227 // Binds |callback| to be invoked *sometime after* |object| is garbage |
| 223 // collected. We don't call the method re-entrantly so as to avoid executing | 228 // collected. We don't call the method re-entrantly so as to avoid executing |
| 224 // JS in some bizarro undefined mid-GC state. | 229 // JS in some bizarro undefined mid-GC state. |
| 225 void BindToGC(const v8::FunctionCallbackInfo<v8::Value>& args) { | 230 void BindToGC(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 226 CHECK(args.Length() == 2 && args[0]->IsObject() && args[1]->IsFunction()); | 231 CHECK(args.Length() == 2 && args[0]->IsObject() && args[1]->IsFunction()); |
| 227 GCCallback::Bind(args[0].As<v8::Object>(), | 232 GCCallback::Bind(args[0].As<v8::Object>(), |
| 228 args[1].As<v8::Function>(), | 233 args[1].As<v8::Function>(), |
| 229 args.GetIsolate()); | 234 args.GetIsolate()); |
| 230 } | 235 } |
| 231 }; | 236 }; |
| 232 | 237 |
| 233 } // namespace | 238 } // namespace |
| 234 | 239 |
| 235 namespace extensions { | |
| 236 | |
| 237 ChromeV8Extension* MessagingBindings::Get( | 240 ChromeV8Extension* MessagingBindings::Get( |
| 238 Dispatcher* dispatcher, | 241 Dispatcher* dispatcher, |
| 239 ChromeV8Context* context) { | 242 ChromeV8Context* context) { |
| 240 return new ExtensionImpl(dispatcher, context); | 243 return new ExtensionImpl(dispatcher, context); |
| 241 } | 244 } |
| 242 | 245 |
| 243 // static | 246 // static |
| 244 void MessagingBindings::DispatchOnConnect( | 247 void MessagingBindings::DispatchOnConnect( |
| 245 const ChromeV8ContextSet::ContextSet& contexts, | 248 const ChromeV8ContextSet::ContextSet& contexts, |
| 246 int target_port_id, | 249 int target_port_id, |
| (...skipping 18 matching lines...) Expand all Loading... |
| 265 if (restrict_to_render_view && | 268 if (restrict_to_render_view && |
| 266 restrict_to_render_view != (*it)->GetRenderView()) { | 269 restrict_to_render_view != (*it)->GetRenderView()) { |
| 267 continue; | 270 continue; |
| 268 } | 271 } |
| 269 | 272 |
| 270 // TODO(kalman): remove when ContextSet::ForEach is available. | 273 // TODO(kalman): remove when ContextSet::ForEach is available. |
| 271 if ((*it)->v8_context().IsEmpty()) | 274 if ((*it)->v8_context().IsEmpty()) |
| 272 continue; | 275 continue; |
| 273 | 276 |
| 274 v8::Handle<v8::Value> tab = v8::Null(isolate); | 277 v8::Handle<v8::Value> tab = v8::Null(isolate); |
| 275 if (!source_tab.empty()) | 278 v8::Handle<v8::Value> tls_channel_id_value = v8::Undefined(isolate); |
| 276 tab = converter->ToV8Value(&source_tab, (*it)->v8_context()); | 279 const Extension* extension = (*it)->extension(); |
| 280 if (extension) { |
| 281 if (!source_tab.empty() && !extension->is_platform_app()) |
| 282 tab = converter->ToV8Value(&source_tab, (*it)->v8_context()); |
| 277 | 283 |
| 278 v8::Handle<v8::Value> tls_channel_id_value = v8::Undefined(isolate); | |
| 279 if ((*it)->extension()) { | |
| 280 ExternallyConnectableInfo* externally_connectable = | 284 ExternallyConnectableInfo* externally_connectable = |
| 281 ExternallyConnectableInfo::Get((*it)->extension()); | 285 ExternallyConnectableInfo::Get(extension); |
| 282 if (externally_connectable && | 286 if (externally_connectable && |
| 283 externally_connectable->accepts_tls_channel_id) { | 287 externally_connectable->accepts_tls_channel_id) { |
| 284 tls_channel_id_value = | 288 tls_channel_id_value = |
| 285 v8::String::NewFromUtf8(isolate, | 289 v8::String::NewFromUtf8(isolate, |
| 286 tls_channel_id.c_str(), | 290 tls_channel_id.c_str(), |
| 287 v8::String::kNormalString, | 291 v8::String::kNormalString, |
| 288 tls_channel_id.size()); | 292 tls_channel_id.size()); |
| 289 } | 293 } |
| 290 } | 294 } |
| 291 | 295 |
| 292 v8::Handle<v8::Value> arguments[] = { | 296 v8::Handle<v8::Value> arguments[] = { |
| 297 // portId |
| 293 v8::Integer::New(isolate, target_port_id), | 298 v8::Integer::New(isolate, target_port_id), |
| 299 // channelName |
| 294 v8::String::NewFromUtf8(isolate, | 300 v8::String::NewFromUtf8(isolate, |
| 295 channel_name.c_str(), | 301 channel_name.c_str(), |
| 296 v8::String::kNormalString, | 302 v8::String::kNormalString, |
| 297 channel_name.size()), | 303 channel_name.size()), |
| 298 tab, v8::String::NewFromUtf8(isolate, | 304 // sourceTab |
| 299 source_extension_id.c_str(), | 305 tab, |
| 300 v8::String::kNormalString, | 306 // sourceExtensionId |
| 301 source_extension_id.size()), | 307 v8::String::NewFromUtf8(isolate, |
| 308 source_extension_id.c_str(), |
| 309 v8::String::kNormalString, |
| 310 source_extension_id.size()), |
| 311 // targetExtensionId |
| 302 v8::String::NewFromUtf8(isolate, | 312 v8::String::NewFromUtf8(isolate, |
| 303 target_extension_id.c_str(), | 313 target_extension_id.c_str(), |
| 304 v8::String::kNormalString, | 314 v8::String::kNormalString, |
| 305 target_extension_id.size()), | 315 target_extension_id.size()), |
| 316 // sourceUrl |
| 306 v8::String::NewFromUtf8(isolate, | 317 v8::String::NewFromUtf8(isolate, |
| 307 source_url_spec.c_str(), | 318 source_url_spec.c_str(), |
| 308 v8::String::kNormalString, | 319 v8::String::kNormalString, |
| 309 source_url_spec.size()), | 320 source_url_spec.size()), |
| 321 // tlsChannelId |
| 310 tls_channel_id_value, | 322 tls_channel_id_value, |
| 311 }; | 323 }; |
| 312 | 324 |
| 313 v8::Handle<v8::Value> retval = (*it)->module_system()->CallModuleMethod( | 325 v8::Handle<v8::Value> retval = (*it)->module_system()->CallModuleMethod( |
| 314 "messaging", | 326 "messaging", |
| 315 "dispatchOnConnect", | 327 "dispatchOnConnect", |
| 316 arraysize(arguments), arguments); | 328 arraysize(arguments), arguments); |
| 317 | 329 |
| 318 if (retval.IsEmpty()) { | 330 if (retval.IsEmpty()) { |
| 319 LOG(ERROR) << "Empty return value from dispatchOnConnect."; | 331 LOG(ERROR) << "Empty return value from dispatchOnConnect."; |
| (...skipping 95 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 415 } else { | 427 } else { |
| 416 arguments.push_back(v8::Null(isolate)); | 428 arguments.push_back(v8::Null(isolate)); |
| 417 } | 429 } |
| 418 (*it)->module_system()->CallModuleMethod("messaging", | 430 (*it)->module_system()->CallModuleMethod("messaging", |
| 419 "dispatchOnDisconnect", | 431 "dispatchOnDisconnect", |
| 420 &arguments); | 432 &arguments); |
| 421 } | 433 } |
| 422 } | 434 } |
| 423 | 435 |
| 424 } // namespace extensions | 436 } // namespace extensions |
| OLD | NEW |