15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
5bb1529ce867d8845a77ec7cdf3e3003ef1771a40Ben Murdoch#include "content/renderer/npapi/plugin_channel_host.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/metrics/histogram.h"
8eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch#include "base/time/time.h"
9868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "content/child/child_process.h"
10ca12bfac764ba476d6cd062bf1dde12cc64c3f40Ben Murdoch#include "content/child/npapi/npobject_base.h"
11868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "content/child/plugin_messages.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_POSIX)
145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ipc/ipc_channel_posix.h"
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
177d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)#include "third_party/WebKit/public/web/WebBindings.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// TODO(shess): Debugging for http://crbug.com/97285
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// The hypothesis at #55 requires that RemoveRoute() be called between
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// sending ViewHostMsg_OpenChannelToPlugin to the browser, and calling
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// GetPluginChannelHost() on the result.  This code detects that case
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// and stores the backtrace of the RemoveRoute() in a breakpad key.
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// The specific RemoveRoute() is not tracked (there could be multiple,
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// and which is the one can't be known until the open completes), but
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// the backtrace from any such nested call should be sufficient to
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// drive a repro.
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_MACOSX)
302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "base/debug/crash_logging.h"
315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/debug/stack_trace.h"
32c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)#include "base/strings/sys_string_conversions.h"
335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Breakpad key for the RemoveRoute() backtrace.
375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char* kRemoveRouteTraceKey = "remove_route_bt";
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Breakpad key for the OnChannelError() backtrace.
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const char* kChannelErrorTraceKey = "channel_error_bt";
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// GetRemoveTrackingFlag() exposes this so that
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// WebPluginDelegateProxy::Initialize() can do scoped set/reset.  When
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// true, RemoveRoute() knows WBDP::Initialize() is on the stack, and
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// records the backtrace.
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool remove_tracking = false;
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace content {
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_MACOSX)
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool* PluginChannelHost::GetRemoveTrackingFlag() {
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return &remove_tracking;
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PluginChannelHost* PluginChannelHost::GetPluginChannelHost(
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const IPC::ChannelHandle& channel_handle,
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    base::MessageLoopProxy* ipc_message_loop) {
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PluginChannelHost* result =
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      static_cast<PluginChannelHost*>(NPChannelBase::GetChannel(
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          channel_handle,
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          IPC::Channel::MODE_CLIENT,
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          ClassFactory,
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          ipc_message_loop,
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          true,
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          ChildProcess::current()->GetShutDownEvent()));
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return result;
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PluginChannelHost::PluginChannelHost() : expecting_shutdown_(false) {
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)PluginChannelHost::~PluginChannelHost() {
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool PluginChannelHost::Init(base::MessageLoopProxy* ipc_message_loop,
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                             bool create_pipe_now,
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                             base::WaitableEvent* shutdown_event) {
84ca12bfac764ba476d6cd062bf1dde12cc64c3f40Ben Murdoch  return NPChannelBase::Init(ipc_message_loop, create_pipe_now, shutdown_event);
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int PluginChannelHost::GenerateRouteID() {
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int route_id = MSG_ROUTING_NONE;
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Send(new PluginMsg_GenerateRouteID(&route_id));
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return route_id;
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PluginChannelHost::AddRoute(int route_id,
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 IPC::Listener* listener,
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                                 NPObjectBase* npobject) {
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NPChannelBase::AddRoute(route_id, listener, npobject);
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!npobject)
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    proxies_[route_id] = listener;
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PluginChannelHost::RemoveRoute(int route_id) {
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_MACOSX)
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (remove_tracking) {
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    base::debug::StackTrace trace;
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    size_t count = 0;
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const void* const* addresses = trace.Addresses(&count);
1092a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    base::debug::SetCrashKeyFromAddresses(
1102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        kRemoveRouteTraceKey, addresses, count);
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  proxies_.erase(route_id);
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NPChannelBase::RemoveRoute(route_id);
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool PluginChannelHost::OnControlMessageReceived(const IPC::Message& message) {
1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool handled = true;
1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  IPC_BEGIN_MESSAGE_MAP(PluginChannelHost, message)
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IPC_MESSAGE_HANDLER(PluginHostMsg_SetException, OnSetException)
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IPC_MESSAGE_HANDLER(PluginHostMsg_PluginShuttingDown, OnPluginShuttingDown)
1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    IPC_MESSAGE_UNHANDLED(handled = false)
1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  IPC_END_MESSAGE_MAP()
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(handled);
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return handled;
1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PluginChannelHost::OnSetException(const std::string& message) {
130f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  blink::WebBindings::setException(NULL, message.c_str());
1315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PluginChannelHost::OnPluginShuttingDown() {
1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  expecting_shutdown_ = true;
1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool PluginChannelHost::Send(IPC::Message* msg) {
1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (msg->is_sync()) {
1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    base::TimeTicks start_time(base::TimeTicks::Now());
1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    bool result = NPChannelBase::Send(msg);
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    UMA_HISTOGRAM_TIMES("Plugin.SyncMessageTime",
1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)                        base::TimeTicks::Now() - start_time);
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return result;
1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return NPChannelBase::Send(msg);
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PluginChannelHost::OnChannelError() {
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(OS_MACOSX)
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (remove_tracking) {
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    base::debug::StackTrace trace;
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    size_t count = 0;
1535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const void* const* addresses = trace.Addresses(&count);
1542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    base::debug::SetCrashKeyFromAddresses(
1552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)        kChannelErrorTraceKey, addresses, count);
1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NPChannelBase::OnChannelError();
1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  for (ProxyMap::iterator iter = proxies_.begin();
1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)       iter != proxies_.end(); iter++) {
1635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    iter->second->OnChannelError();
1645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  proxies_.clear();
1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace content
170