1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
// Copyright (c) 2006-2008 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 <windows.h>
#include "chrome/plugin/plugin_channel.h"
#include "base/command_line.h"
#include "base/process_util.h"
#include "base/string_util.h"
#include "chrome/common/child_process.h"
#include "chrome/common/plugin_messages.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/plugin/plugin_thread.h"
PluginChannel* PluginChannel::GetPluginChannel(MessageLoop* ipc_message_loop) {
static int next_id;
std::wstring channel_name = StringPrintf(
L"%d.r%d", GetCurrentProcessId(), ++next_id);
return static_cast<PluginChannel*>(PluginChannelBase::GetChannel(
channel_name,
IPC::Channel::MODE_SERVER,
ClassFactory,
ipc_message_loop,
false));
}
PluginChannel::PluginChannel() : in_send_(0), off_the_record_(false) {
SendUnblockingOnlyDuringDispatch();
ChildProcess::current()->AddRefProcess();
const CommandLine* command_line = CommandLine::ForCurrentProcess();
log_messages_ = command_line->HasSwitch(switches::kLogPluginMessages);
}
PluginChannel::~PluginChannel() {
ChildProcess::current()->ReleaseProcess();
}
bool PluginChannel::Send(IPC::Message* msg) {
in_send_++;
if (log_messages_) {
LOG(INFO) << "sending message @" << msg << " on channel @" << this
<< " with type " << msg->type();
}
bool result = PluginChannelBase::Send(msg);
in_send_--;
return result;
}
void PluginChannel::OnMessageReceived(const IPC::Message& msg) {
if (log_messages_) {
LOG(INFO) << "received message @" << &msg << " on channel @" << this
<< " with type " << msg.type();
}
PluginChannelBase::OnMessageReceived(msg);
}
void PluginChannel::OnControlMessageReceived(const IPC::Message& msg) {
IPC_BEGIN_MESSAGE_MAP(PluginChannel, msg)
IPC_MESSAGE_HANDLER(PluginMsg_CreateInstance, OnCreateInstance)
IPC_MESSAGE_HANDLER_DELAY_REPLY(PluginMsg_DestroyInstance,
OnDestroyInstance)
IPC_MESSAGE_HANDLER(PluginMsg_GenerateRouteID, OnGenerateRouteID)
IPC_MESSAGE_UNHANDLED_ERROR()
IPC_END_MESSAGE_MAP()
}
void PluginChannel::OnCreateInstance(const std::string& mime_type,
int* instance_id) {
*instance_id = GenerateRouteID();
scoped_refptr<WebPluginDelegateStub> stub = new WebPluginDelegateStub(
mime_type, *instance_id, this);
AddRoute(*instance_id, stub, false);
plugin_stubs_.push_back(stub);
}
void PluginChannel::OnDestroyInstance(int instance_id,
IPC::Message* reply_msg) {
for (size_t i = 0; i < plugin_stubs_.size(); ++i) {
if (plugin_stubs_[i]->instance_id() == instance_id) {
plugin_stubs_.erase(plugin_stubs_.begin() + i);
RemoveRoute(instance_id);
Send(reply_msg);
return;
}
}
NOTREACHED() << "Couldn't find WebPluginDelegateStub to destroy";
}
void PluginChannel::OnGenerateRouteID(int* route_id) {
*route_id = GenerateRouteID();
}
int PluginChannel::GenerateRouteID() {
static LONG last_id = 0;
return InterlockedIncrement(&last_id);
}
void PluginChannel::OnChannelConnected(int32 peer_pid) {
renderer_handle_.Set(base::OpenProcessHandle(peer_pid));
PluginChannelBase::OnChannelConnected(peer_pid);
}
void PluginChannel::OnChannelError() {
renderer_handle_.Set(NULL);
PluginChannelBase::OnChannelError();
CleanUp();
}
void PluginChannel::CleanUp() {
// We need to clean up the stubs so that they call NPPDestroy. This will
// also lead to them releasing their reference on this object so that it can
// be deleted.
for (size_t i = 0; i < plugin_stubs_.size(); ++i)
RemoveRoute(plugin_stubs_[i]->instance_id());
// Need to addref this object temporarily because otherwise removing the last
// stub will cause the destructor of this object to be called, however at
// that point plugin_stubs_ will have one element and its destructor will be
// called twice.
scoped_refptr<PluginChannel> me(this);
plugin_stubs_.clear();
}
|