1 /*
2 * Copyright (C) 2010 Apple Inc. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23 * THE POSSIBILITY OF SUCH DAMAGE.
24 */
25
26 #include "config.h"
27 #include "PluginProcess.h"
28
29 #if ENABLE(PLUGIN_PROCESS)
30
31 #include "ArgumentCoders.h"
32 #include "NetscapePluginModule.h"
33 #include "PluginProcessProxyMessages.h"
34 #include "PluginProcessCreationParameters.h"
35 #include "WebProcessConnection.h"
36
37 #if PLATFORM(MAC)
38 #include "MachPort.h"
39 #endif
40
41 namespace WebKit {
42
43 static const double shutdownTimeout = 15.0;
44
shared()45 PluginProcess& PluginProcess::shared()
46 {
47 DEFINE_STATIC_LOCAL(PluginProcess, pluginProcess, ());
48 return pluginProcess;
49 }
50
PluginProcess()51 PluginProcess::PluginProcess()
52 : ChildProcess(shutdownTimeout)
53 #if PLATFORM(MAC)
54 , m_compositingRenderServerPort(MACH_PORT_NULL)
55 #endif
56 {
57 }
58
~PluginProcess()59 PluginProcess::~PluginProcess()
60 {
61 }
62
initialize(CoreIPC::Connection::Identifier serverIdentifier,RunLoop * runLoop)63 void PluginProcess::initialize(CoreIPC::Connection::Identifier serverIdentifier, RunLoop* runLoop)
64 {
65 ASSERT(!m_connection);
66
67 m_connection = CoreIPC::Connection::createClientConnection(serverIdentifier, this, runLoop);
68 m_connection->setDidCloseOnConnectionWorkQueueCallback(didCloseOnConnectionWorkQueue);
69 m_connection->open();
70 }
71
removeWebProcessConnection(WebProcessConnection * webProcessConnection)72 void PluginProcess::removeWebProcessConnection(WebProcessConnection* webProcessConnection)
73 {
74 size_t vectorIndex = m_webProcessConnections.find(webProcessConnection);
75 ASSERT(vectorIndex != notFound);
76
77 m_webProcessConnections.remove(vectorIndex);
78
79 if (m_webProcessConnections.isEmpty() && m_pluginModule) {
80 // Decrement the load count. This is balanced by a call to incrementLoadCount in createWebProcessConnection.
81 m_pluginModule->decrementLoadCount();
82 }
83
84 enableTermination();
85 }
86
netscapePluginModule()87 NetscapePluginModule* PluginProcess::netscapePluginModule()
88 {
89 if (!m_pluginModule) {
90 ASSERT(!m_pluginPath.isNull());
91 m_pluginModule = NetscapePluginModule::getOrCreate(m_pluginPath);
92
93 #if PLATFORM(MAC)
94 if (m_pluginModule) {
95 if (m_pluginModule->pluginQuirks().contains(PluginQuirks::PrognameShouldBeWebKitPluginHost))
96 setprogname("WebKitPluginHost");
97 }
98 #endif
99 }
100
101 return m_pluginModule.get();
102 }
103
shouldTerminate()104 bool PluginProcess::shouldTerminate()
105 {
106 ASSERT(m_webProcessConnections.isEmpty());
107
108 return true;
109 }
110
didReceiveMessage(CoreIPC::Connection * connection,CoreIPC::MessageID messageID,CoreIPC::ArgumentDecoder * arguments)111 void PluginProcess::didReceiveMessage(CoreIPC::Connection* connection, CoreIPC::MessageID messageID, CoreIPC::ArgumentDecoder* arguments)
112 {
113 didReceivePluginProcessMessage(connection, messageID, arguments);
114 }
115
didClose(CoreIPC::Connection *)116 void PluginProcess::didClose(CoreIPC::Connection*)
117 {
118 // The UI process has crashed, just go ahead and quit.
119 // FIXME: If the plug-in is spinning in the main loop, we'll never get this message.
120 RunLoop::current()->stop();
121 }
122
didReceiveInvalidMessage(CoreIPC::Connection *,CoreIPC::MessageID)123 void PluginProcess::didReceiveInvalidMessage(CoreIPC::Connection*, CoreIPC::MessageID)
124 {
125 }
126
syncMessageSendTimedOut(CoreIPC::Connection *)127 void PluginProcess::syncMessageSendTimedOut(CoreIPC::Connection*)
128 {
129 }
130
initializePluginProcess(const PluginProcessCreationParameters & parameters)131 void PluginProcess::initializePluginProcess(const PluginProcessCreationParameters& parameters)
132 {
133 ASSERT(!m_pluginModule);
134
135 m_pluginPath = parameters.pluginPath;
136
137 platformInitialize(parameters);
138 }
139
createWebProcessConnection()140 void PluginProcess::createWebProcessConnection()
141 {
142 bool didHaveAnyWebProcessConnections = !m_webProcessConnections.isEmpty();
143
144 #if PLATFORM(MAC)
145 // Create the listening port.
146 mach_port_t listeningPort;
147 mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &listeningPort);
148
149 // Create a listening connection.
150 RefPtr<WebProcessConnection> connection = WebProcessConnection::create(listeningPort);
151 m_webProcessConnections.append(connection.release());
152
153 CoreIPC::MachPort clientPort(listeningPort, MACH_MSG_TYPE_MAKE_SEND);
154 m_connection->send(Messages::PluginProcessProxy::DidCreateWebProcessConnection(clientPort), 0);
155 #else
156 // FIXME: Implement.
157 ASSERT_NOT_REACHED();
158 #endif
159
160 if (NetscapePluginModule* module = netscapePluginModule()) {
161 if (!didHaveAnyWebProcessConnections) {
162 // Increment the load count. This is matched by a call to decrementLoadCount in removeWebProcessConnection.
163 // We do this so that the plug-in module's NP_Shutdown won't be called until right before exiting.
164 module->incrementLoadCount();
165 }
166 }
167
168 disableTermination();
169 }
170
getSitesWithData(uint64_t callbackID)171 void PluginProcess::getSitesWithData(uint64_t callbackID)
172 {
173 LocalTerminationDisabler terminationDisabler(*this);
174
175 Vector<String> sites;
176 if (NetscapePluginModule* module = netscapePluginModule())
177 sites = module->sitesWithData();
178
179 m_connection->send(Messages::PluginProcessProxy::DidGetSitesWithData(sites, callbackID), 0);
180 }
181
clearSiteData(const Vector<String> & sites,uint64_t flags,uint64_t maxAgeInSeconds,uint64_t callbackID)182 void PluginProcess::clearSiteData(const Vector<String>& sites, uint64_t flags, uint64_t maxAgeInSeconds, uint64_t callbackID)
183 {
184 LocalTerminationDisabler terminationDisabler(*this);
185
186 if (NetscapePluginModule* module = netscapePluginModule()) {
187 if (sites.isEmpty()) {
188 // Clear everything.
189 module->clearSiteData(String(), flags, maxAgeInSeconds);
190 } else {
191 for (size_t i = 0; i < sites.size(); ++i)
192 module->clearSiteData(sites[i], flags, maxAgeInSeconds);
193 }
194 }
195
196 m_connection->send(Messages::PluginProcessProxy::DidClearSiteData(callbackID), 0);
197 }
198
199 } // namespace WebKit
200
201 #endif // ENABLE(PLUGIN_PROCESS)
202
203