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 3 // found in the LICENSE file. 4 5 #include "content/renderer/pepper/pepper_in_process_router.h" 6 7 #include "base/bind.h" 8 #include "base/message_loop/message_loop.h" 9 #include "content/public/renderer/render_thread.h" 10 #include "content/renderer/pepper/renderer_ppapi_host_impl.h" 11 #include "content/renderer/render_frame_impl.h" 12 #include "ipc/ipc_message.h" 13 #include "ipc/ipc_sender.h" 14 #include "ppapi/proxy/ppapi_messages.h" 15 #include "ppapi/shared_impl/ppapi_globals.h" 16 #include "ppapi/shared_impl/resource_tracker.h" 17 18 using ppapi::UnpackMessage; 19 20 namespace content { 21 22 class PepperInProcessRouter::Channel : public IPC::Sender { 23 public: 24 Channel(const base::Callback<bool(IPC::Message*)>& callback) 25 : callback_(callback) {} 26 27 virtual ~Channel() {} 28 29 virtual bool Send(IPC::Message* message) OVERRIDE { 30 return callback_.Run(message); 31 } 32 33 private: 34 base::Callback<bool(IPC::Message*)> callback_; 35 }; 36 37 PepperInProcessRouter::PepperInProcessRouter( 38 RendererPpapiHostImpl* host_impl) 39 : host_impl_(host_impl), 40 pending_message_id_(0), 41 reply_result_(false), 42 weak_factory_(this) { 43 browser_channel_.reset( 44 new Channel(base::Bind(&PepperInProcessRouter::SendToBrowser, 45 base::Unretained(this)))); 46 host_to_plugin_router_.reset( 47 new Channel(base::Bind(&PepperInProcessRouter::SendToPlugin, 48 base::Unretained(this)))); 49 plugin_to_host_router_.reset( 50 new Channel(base::Bind(&PepperInProcessRouter::SendToHost, 51 base::Unretained(this)))); 52 } 53 54 PepperInProcessRouter::~PepperInProcessRouter() { 55 } 56 57 IPC::Sender* PepperInProcessRouter::GetPluginToRendererSender() { 58 return plugin_to_host_router_.get(); 59 } 60 61 IPC::Sender* PepperInProcessRouter::GetRendererToPluginSender() { 62 return host_to_plugin_router_.get(); 63 } 64 65 ppapi::proxy::Connection PepperInProcessRouter::GetPluginConnection( 66 PP_Instance instance) { 67 int routing_id = 0; 68 RenderFrame* frame = host_impl_->GetRenderFrameForInstance(instance); 69 if (frame) 70 routing_id = frame->GetRoutingID(); 71 return ppapi::proxy::Connection(browser_channel_.get(), 72 plugin_to_host_router_.get(), 73 routing_id); 74 } 75 76 // static 77 bool PepperInProcessRouter::OnPluginMsgReceived(const IPC::Message& msg) { 78 // Emulate the proxy by dispatching the relevant message here. 79 ppapi::proxy::ResourceMessageReplyParams reply_params; 80 IPC::Message nested_msg; 81 82 if (msg.type() == PpapiPluginMsg_ResourceReply::ID) { 83 // Resource reply from the renderer (no routing id). 84 if (!UnpackMessage<PpapiPluginMsg_ResourceReply>(msg, &reply_params, 85 &nested_msg)) { 86 NOTREACHED(); 87 return false; 88 } 89 } else if (msg.type() == PpapiHostMsg_InProcessResourceReply::ID) { 90 // Resource reply from the browser (has a routing id). 91 if (!UnpackMessage<PpapiHostMsg_InProcessResourceReply>(msg, &reply_params, 92 &nested_msg)) { 93 NOTREACHED(); 94 return false; 95 } 96 } else { 97 return false; 98 } 99 ppapi::Resource* resource = 100 ppapi::PpapiGlobals::Get()->GetResourceTracker()->GetResource( 101 reply_params.pp_resource()); 102 // If the resource doesn't exist, it may have been destroyed so just ignore 103 // the message. 104 if (resource) 105 resource->OnReplyReceived(reply_params, nested_msg); 106 return true; 107 } 108 109 bool PepperInProcessRouter::SendToHost(IPC::Message* msg) { 110 scoped_ptr<IPC::Message> message(msg); 111 112 if (!message->is_sync()) { 113 // If this is a resource destroyed message, post a task to dispatch it. 114 // Dispatching it synchronously can cause the host to re-enter the proxy 115 // code while we're still in the resource destructor, leading to a crash. 116 // http://crbug.com/276368. 117 // This won't cause message reordering problems because the resource 118 // destroyed message is always the last one sent for a resource. 119 if (message->type() == PpapiHostMsg_ResourceDestroyed::ID) { 120 base::MessageLoop::current()->PostTask( 121 FROM_HERE, 122 base::Bind(&PepperInProcessRouter::DispatchHostMsg, 123 weak_factory_.GetWeakPtr(), 124 base::Owned(message.release()))); 125 return true; 126 } else { 127 bool result = host_impl_->GetPpapiHost()->OnMessageReceived(*message); 128 DCHECK(result) << "The message was not handled by the host."; 129 return true; 130 } 131 } 132 133 pending_message_id_ = IPC::SyncMessage::GetMessageId(*message); 134 reply_deserializer_.reset( 135 static_cast<IPC::SyncMessage*>(message.get())->GetReplyDeserializer()); 136 reply_result_ = false; 137 138 bool result = host_impl_->GetPpapiHost()->OnMessageReceived(*message); 139 DCHECK(result) << "The message was not handled by the host."; 140 141 pending_message_id_ = 0; 142 reply_deserializer_.reset(NULL); 143 return reply_result_; 144 } 145 146 bool PepperInProcessRouter::SendToPlugin(IPC::Message* msg) { 147 scoped_ptr<IPC::Message> message(msg); 148 CHECK(!msg->is_sync()); 149 if (IPC::SyncMessage::IsMessageReplyTo(*message, pending_message_id_)) { 150 if (!msg->is_reply_error()) 151 reply_result_ = reply_deserializer_->SerializeOutputParameters(*message); 152 } else { 153 CHECK(!pending_message_id_); 154 // Dispatch plugin messages from the message loop. 155 base::MessageLoop::current()->PostTask( 156 FROM_HERE, 157 base::Bind(&PepperInProcessRouter::DispatchPluginMsg, 158 weak_factory_.GetWeakPtr(), 159 base::Owned(message.release()))); 160 } 161 return true; 162 } 163 164 void PepperInProcessRouter::DispatchHostMsg(IPC::Message* msg) { 165 bool handled = host_impl_->GetPpapiHost()->OnMessageReceived(*msg); 166 DCHECK(handled); 167 } 168 169 void PepperInProcessRouter::DispatchPluginMsg(IPC::Message* msg) { 170 bool handled = OnPluginMsgReceived(*msg); 171 DCHECK(handled); 172 } 173 174 bool PepperInProcessRouter::SendToBrowser(IPC::Message *msg) { 175 return RenderThread::Get()->Send(msg); 176 } 177 178 } // namespace content 179