blob: 308bb177290b650b62e23ba35d1f40ffdf0e7610 [file] [log] [blame]
[email protected]f85f0702010-01-30 09:31:011// Copyright (c) 2010 The Chromium Authors. All rights reserved.
license.botbf09a502008-08-24 00:55:552// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
initial.commit09911bf2008-07-26 23:55:294
[email protected]1d97d2e2008-12-18 23:39:025#ifndef CHROME_RENDERER_RENDER_THREAD_H_
6#define CHROME_RENDERER_RENDER_THREAD_H_
[email protected]32b76ef2010-07-26 23:08:247#pragma once
initial.commit09911bf2008-07-26 23:55:298
[email protected]b68d5ed2009-04-16 02:41:289#include <string>
[email protected]1bc83062009-02-06 00:16:3710#include <vector>
11
initial.commit09911bf2008-07-26 23:55:2912#include "base/shared_memory.h"
[email protected]71d6d852009-12-07 22:12:3613#include "base/time.h"
14#include "base/timer.h"
[email protected]037fce02009-01-22 01:42:1515#include "build/build_config.h"
[email protected]8930d472009-02-21 08:05:2816#include "chrome/common/child_thread.h"
[email protected]b9ab10c2009-08-07 18:09:5517#include "chrome/common/css_colors.h"
[email protected]3e90d4a2009-07-03 17:38:3918#include "chrome/renderer/visitedlink_slave.h"
[email protected]5c7293a2010-03-17 06:40:5719#include "gfx/native_widget_types.h"
[email protected]a83d42292010-08-17 22:51:1020#include "ipc/ipc_channel_proxy.h"
[email protected]cb6037d2009-11-16 22:55:1721#include "ipc/ipc_platform_file.h"
initial.commit09911bf2008-07-26 23:55:2922
[email protected]1edc16b82009-04-07 17:45:5423class AppCacheDispatcher;
[email protected]dd9241932010-02-24 19:23:1324class CookieMessageFilter;
[email protected]017022b2009-07-27 23:06:3425class DBMessageFilter;
[email protected]a8624712009-04-17 00:51:3526class DevToolsAgentFilter;
[email protected]39008c02009-02-11 23:59:2527class FilePath;
[email protected]e13ad79b2010-07-22 21:36:5028class GpuChannelHost;
[email protected]70c19a932010-05-14 12:59:1129class IndexedDBDispatcher;
[email protected]dfcb62a2009-06-17 19:32:4330class ListValue;
[email protected]b7c7bcf2009-10-03 07:07:3431class NullableString16;
[email protected]55e57d42009-02-25 06:10:1732class RendererHistogram;
[email protected]e13ad79b2010-07-22 21:36:5033class RendererHistogramSnapshots;
[email protected]74be069e82010-06-25 00:12:4934class RendererNetPredictor;
[email protected]8d86fce2009-02-26 23:37:5535class RendererWebKitClientImpl;
[email protected]85c55dc2009-11-06 03:05:4636class SpellCheck;
[email protected]39008c02009-02-11 23:59:2537class SkBitmap;
[email protected]4d395d092009-02-11 21:40:4038class UserScriptSlave;
[email protected]cccf90932009-08-23 17:56:2539class URLPattern;
[email protected]2b437e232010-04-02 01:30:0840class WebDatabaseObserverImpl;
[email protected]cccf90932009-08-23 17:56:2541
[email protected]f85f0702010-01-30 09:31:0142struct ContentSettings;
[email protected]9b6f40e2009-06-11 15:54:2643struct RendererPreferences;
[email protected]c61cc652009-11-04 05:44:4044struct ViewMsg_DOMStorageEvent_Params;
[email protected]942690b132010-05-11 06:42:1445struct ViewMsg_ExtensionExtentsUpdated_Params;
[email protected]4e6419c2010-01-15 04:50:3446struct ViewMsg_New_Params;
[email protected]39008c02009-02-11 23:59:2547struct WebPreferences;
initial.commit09911bf2008-07-26 23:55:2948
[email protected]c6a7b862010-08-20 22:19:3849namespace base {
50class MessageLoopProxy;
51class Thread;
52}
53
[email protected]46f36a492010-07-28 19:36:4154namespace IPC {
55struct ChannelHandle;
56}
57
[email protected]b7c7bcf2009-10-03 07:07:3458namespace WebKit {
59class WebStorageEventDispatcher;
60}
61
[email protected]81a34412009-01-05 19:17:2462// The RenderThreadBase is the minimal interface that a RenderView/Widget
63// expects from a render thread. The interface basically abstracts a way to send
64// and receive messages.
[email protected]00c39612010-03-06 02:53:2865//
[email protected]f3ede412010-06-21 22:52:1666// TODO(brettw): This has two different and opposing usage patterns which
67// make it confusing.
68//
69// In the first mode, callers call RenderThread::current() to get the one and
70// only global RenderThread (bug 10837: this should be renamed get()). Then
71// they access it. Since RenderThread is a concrete class, this can be NULL
72// during unit tests. Callers need to NULL check this every time. Some callers
73// don't happen to get called during unit tests and don't do the NULL checks,
74// which is also confusing since it's not clear if you need to or not.
75//
76// In the second mode, the abstract base class RenderThreadBase is passed to
77// RenderView and RenderWidget. Normally, this points to
78// RenderThread::current() so it's quite confusing which accessing mode should
79// be used. However, during unit testing, this class is replaced with a mock
80// to support testing functions, and is guaranteed non-NULL.
81//
82// It might be nice not to have the ::current() call and put all of the
83// functions on the abstract class so they can be mocked. However, there are
84// some standalone functions like in ChromiumBridge that are not associated
85// with a view that need to access the current thread to send messages to the
86// browser process. These need the ::current() paradigm. So instead, we should
87// probably remove the render_thread_ parameter to RenderView/Widget in
88// preference to just getting the global singleton. We can make it easier to
89// understand by moving everything to the abstract interface and saying that
90// there should never be a NULL RenderThread::current(). Tests would be
91// responsible for setting up the mock one.
[email protected]8930d472009-02-21 08:05:2892class RenderThreadBase {
[email protected]8085dbc82008-09-26 22:53:4493 public:
94 virtual ~RenderThreadBase() {}
95
[email protected]8930d472009-02-21 08:05:2896 virtual bool Send(IPC::Message* msg) = 0;
97
[email protected]8085dbc82008-09-26 22:53:4498 // Called to add or remove a listener for a particular message routing ID.
99 // These methods normally get delegated to a MessageRouter.
100 virtual void AddRoute(int32 routing_id, IPC::Channel::Listener* listener) = 0;
101 virtual void RemoveRoute(int32 routing_id) = 0;
[email protected]81a34412009-01-05 19:17:24102
103 virtual void AddFilter(IPC::ChannelProxy::MessageFilter* filter) = 0;
104 virtual void RemoveFilter(IPC::ChannelProxy::MessageFilter* filter) = 0;
[email protected]bee16aab2009-08-26 15:55:03105
106 // Called by a RenderWidget when it is hidden or restored.
107 virtual void WidgetHidden() = 0;
108 virtual void WidgetRestored() = 0;
[email protected]b8f41a192010-04-19 18:25:04109
110 // True if this process should be treated as an extension process.
111 virtual bool IsExtensionProcess() const = 0;
[email protected]8085dbc82008-09-26 22:53:44112};
113
initial.commit09911bf2008-07-26 23:55:29114// The RenderThread class represents a background thread where RenderView
115// instances live. The RenderThread supports an API that is used by its
116// consumer to talk indirectly to the RenderViews and supporting objects.
117// Likewise, it provides an API for the RenderViews to talk back to the main
[email protected]57c6a652009-05-04 07:58:34118// process (i.e., their corresponding TabContents).
initial.commit09911bf2008-07-26 23:55:29119//
120// Most of the communication occurs in the form of IPC messages. They are
121// routed to the RenderThread according to the routing IDs of the messages.
122// The routing IDs correspond to RenderView instances.
[email protected]8930d472009-02-21 08:05:28123class RenderThread : public RenderThreadBase,
124 public ChildThread {
initial.commit09911bf2008-07-26 23:55:29125 public:
[email protected]8930d472009-02-21 08:05:28126 // Grabs the IPC channel name from the command line.
127 RenderThread();
128 // Constructor that's used when running in single process mode.
[email protected]11f4857282009-11-13 19:56:17129 explicit RenderThread(const std::string& channel_name);
[email protected]8085dbc82008-09-26 22:53:44130 virtual ~RenderThread();
initial.commit09911bf2008-07-26 23:55:29131
[email protected]8930d472009-02-21 08:05:28132 // Returns the one render thread for this process. Note that this should only
133 // be accessed when running on the render thread itself
[email protected]00c39612010-03-06 02:53:28134 //
135 // TODO(brettw) this should be on the abstract base class instead of here,
136 // and return the base class' interface instead. Currently this causes
137 // problems with testing. See the comment above RenderThreadBase above.
[email protected]8930d472009-02-21 08:05:28138 static RenderThread* current();
initial.commit09911bf2008-07-26 23:55:29139
[email protected]c1f50aa2010-02-18 03:46:57140 // Returns the routing ID of the RenderWidget containing the current script
141 // execution context (corresponding to WebFrame::frameForCurrentContext).
142 static int32 RoutingIDForCurrentContext();
143
[email protected]45776222009-07-15 20:21:58144 // Overridden from RenderThreadBase.
[email protected]c1f50aa2010-02-18 03:46:57145 virtual bool Send(IPC::Message* msg);
146 virtual void AddRoute(int32 routing_id, IPC::Channel::Listener* listener);
147 virtual void RemoveRoute(int32 routing_id);
[email protected]81a34412009-01-05 19:17:24148 virtual void AddFilter(IPC::ChannelProxy::MessageFilter* filter);
149 virtual void RemoveFilter(IPC::ChannelProxy::MessageFilter* filter);
[email protected]bee16aab2009-08-26 15:55:03150 virtual void WidgetHidden();
151 virtual void WidgetRestored();
152
[email protected]c1f50aa2010-02-18 03:46:57153 // These methods modify how the next message is sent. Normally, when sending
154 // a synchronous message that runs a nested message loop, we need to suspend
155 // callbacks into WebKit. This involves disabling timers and deferring
156 // resource loads. However, there are exceptions when we need to customize
157 // the behavior.
158 void DoNotSuspendWebKitSharedTimer();
159 void DoNotNotifyWebKitOfModalLoop();
160
[email protected]8d86fce2009-02-26 23:37:55161 VisitedLinkSlave* visited_link_slave() const {
162 return visited_link_slave_.get();
163 }
initial.commit09911bf2008-07-26 23:55:29164
[email protected]8d86fce2009-02-26 23:37:55165 UserScriptSlave* user_script_slave() const {
166 return user_script_slave_.get();
167 }
[email protected]1e0f70402008-10-16 23:57:47168
[email protected]f430b5712009-08-21 21:46:31169 AppCacheDispatcher* appcache_dispatcher() const {
170 return appcache_dispatcher_.get();
171 }
172
[email protected]70c19a932010-05-14 12:59:11173 IndexedDBDispatcher* indexed_db_dispatcher() const {
174 return indexed_db_dispatcher_.get();
175 }
176
[email protected]85c55dc2009-11-06 03:05:46177 SpellCheck* spellchecker() const {
178 return spellchecker_.get();
179 }
[email protected]85c55dc2009-11-06 03:05:46180
[email protected]b547fd42009-04-23 23:16:27181 bool plugin_refresh_allowed() const { return plugin_refresh_allowed_; }
182
[email protected]b8f41a192010-04-19 18:25:04183 virtual bool IsExtensionProcess() const { return is_extension_process_; }
[email protected]71d6d852009-12-07 22:12:36184
[email protected]b2a74ca2010-03-12 17:57:09185 bool is_incognito_process() const { return is_incognito_process_; }
186
initial.commit09911bf2008-07-26 23:55:29187 // Do DNS prefetch resolution of a hostname.
188 void Resolve(const char* name, size_t length);
189
[email protected]55e57d42009-02-25 06:10:17190 // Send all the Histogram data to browser.
[email protected]c9a3ef82009-05-28 22:02:46191 void SendHistograms(int sequence_number);
[email protected]55e57d42009-02-25 06:10:17192
initial.commit09911bf2008-07-26 23:55:29193 // Invokes InformHostOfCacheStats after a short delay. Used to move this
194 // bookkeeping operation off the critical latency path.
195 void InformHostOfCacheStatsLater();
196
[email protected]c40acc32010-01-14 01:02:53197 // Sends a message to the browser to close all connections.
198 void CloseCurrentConnections();
[email protected]b07f29092009-06-05 07:33:21199
200 // Sends a message to the browser to enable or disable the disk cache.
201 void SetCacheMode(bool enabled);
202
[email protected]c5d79342010-06-05 01:27:34203 // Sends a message to the browser to clear the disk cache.
204 void ClearCache();
205
[email protected]12893c32010-08-19 17:30:54206 // Sends a message to the browser to enable/disable spdy.
207 void EnableSpdy(bool enable);
208
[email protected]c8865962009-12-16 07:47:39209 // Update the list of active extensions that will be reported when we crash.
210 void UpdateActiveExtensions();
211
[email protected]6217d392010-03-25 22:08:35212 // Asynchronously establish a channel to the GPU plugin if not previously
213 // established or if it has been lost (for example if the GPU plugin crashed).
214 // Use GetGpuChannel() to determine when the channel is ready for use.
215 void EstablishGpuChannel();
216
[email protected]3bf4d532010-03-27 00:23:34217 // Synchronously establish a channel to the GPU plugin if not previously
218 // established or if it has been lost (for example if the GPU plugin crashed).
219 // If there is a pending asynchronous request, it will be completed by the
220 // time this routine returns.
221 GpuChannelHost* EstablishGpuChannelSync();
222
[email protected]6217d392010-03-25 22:08:35223 // Get the GPU channel. Returns NULL if the channel is not established or
224 // has been lost.
225 GpuChannelHost* GetGpuChannel();
226
[email protected]3a8eecb2010-04-22 23:56:30227 // Returns the extension ID that the given URL is a part of, or empty if
228 // none. This includes web URLs that are part of an extension's web extent.
229 // TODO(mpcomplete): this doesn't feel like it belongs here. Find a better
230 // place.
[email protected]4fdbc1492010-07-01 01:20:59231 std::string GetExtensionIdByURL(const GURL& url);
232
233 std::string GetExtensionIdByBrowseExtent(const GURL& url);
[email protected]3a8eecb2010-04-22 23:56:30234
[email protected]c6a7b862010-08-20 22:19:38235 // Returns a MessageLoopProxy instance corresponding to the message loop
236 // of the thread on which file operations should be run. Must be called
237 // on the renderer's main thread.
238 scoped_refptr<base::MessageLoopProxy> GetFileThreadMessageLoopProxy();
239
[email protected]8930d472009-02-21 08:05:28240 private:
241 virtual void OnControlMessageReceived(const IPC::Message& msg);
initial.commit09911bf2008-07-26 23:55:29242
[email protected]42f1d7822009-07-23 18:17:55243 void Init();
initial.commit09911bf2008-07-26 23:55:29244
[email protected]176aa482008-11-14 03:25:15245 void OnUpdateVisitedLinks(base::SharedMemoryHandle table);
[email protected]3e90d4a2009-07-03 17:38:39246 void OnAddVisitedLinks(const VisitedLinkSlave::Fingerprints& fingerprints);
247 void OnResetVisitedLinks();
[email protected]9d797f32010-04-23 07:17:54248 void OnSetZoomLevelForCurrentURL(const GURL& url, int zoom_level);
[email protected]0314ae02010-04-08 09:18:29249 void OnSetContentSettingsForCurrentURL(
250 const GURL& url, const ContentSettings& content_settings);
[email protected]b2a74ca2010-03-12 17:57:09251 void OnUpdateUserScripts(base::SharedMemoryHandle table);
[email protected]703e807a2009-03-28 19:56:51252 void OnSetExtensionFunctionNames(const std::vector<std::string>& names);
[email protected]3a8eecb2010-04-22 23:56:30253 void OnExtensionExtentsUpdated(
254 const ViewMsg_ExtensionExtentsUpdated_Params& params);
[email protected]45776222009-07-15 20:21:58255 void OnPageActionsUpdated(const std::string& extension_id,
[email protected]b7c7bcf2009-10-03 07:07:34256 const std::vector<std::string>& page_actions);
[email protected]c61cc652009-11-04 05:44:40257 void OnDOMStorageEvent(const ViewMsg_DOMStorageEvent_Params& params);
[email protected]75e126b932009-09-28 19:38:49258 void OnExtensionSetAPIPermissions(
259 const std::string& extension_id,
260 const std::vector<std::string>& permissions);
261 void OnExtensionSetHostPermissions(
262 const GURL& extension_url,
263 const std::vector<URLPattern>& permissions);
[email protected]db7331a2010-02-25 22:10:50264 void OnExtensionSetIncognitoEnabled(
265 const std::string& extension_id,
266 bool enabled);
initial.commit09911bf2008-07-26 23:55:29267 void OnSetNextPageID(int32 next_page_id);
[email protected]b2a74ca2010-03-12 17:57:09268 void OnSetIsIncognitoProcess(bool is_incognito_process);
[email protected]b9ab10c2009-08-07 18:09:55269 void OnSetCSSColors(const std::vector<CSSColors::CSSColorMapping>& colors);
[email protected]4e6419c2010-01-15 04:50:34270 void OnCreateNewView(const ViewMsg_New_Params& params);
initial.commit09911bf2008-07-26 23:55:29271 void OnTransferBitmap(const SkBitmap& bitmap, int resource_id);
272 void OnSetCacheCapacities(size_t min_dead_capacity,
273 size_t max_dead_capacity,
274 size_t capacity);
275 void OnGetCacheResourceStats();
276
[email protected]55e57d42009-02-25 06:10:17277 // Send all histograms to browser.
[email protected]c9a3ef82009-05-28 22:02:46278 void OnGetRendererHistograms(int sequence_number);
[email protected]55e57d42009-02-25 06:10:17279
[email protected]d41041092009-10-08 06:56:57280 // Send tcmalloc info to browser.
281 void OnGetRendererTcmalloc();
[email protected]38b48a82009-11-11 01:51:32282 void OnGetV8HeapStats();
[email protected]d41041092009-10-08 06:56:57283
[email protected]dfcb62a2009-06-17 19:32:43284 void OnExtensionMessageInvoke(const std::string& function_name,
[email protected]d7259472010-03-24 08:40:49285 const ListValue& args,
[email protected]a807bbe2010-04-14 10:51:19286 bool requires_incognito_access,
287 const GURL& event_url);
[email protected]fede6ca12009-10-08 18:24:26288 void OnPurgeMemory();
[email protected]b78e168b2009-09-21 22:05:45289 void OnPurgePluginListCache(bool reload_pages);
[email protected]75e5a872009-04-02 23:56:11290
[email protected]cb6037d2009-11-16 22:55:17291 void OnInitSpellChecker(IPC::PlatformFileForTransit bdict_file,
[email protected]85c55dc2009-11-06 03:05:46292 const std::vector<std::string>& custom_words,
293 const std::string& language,
294 bool auto_spell_correct);
295 void OnSpellCheckWordAdded(const std::string& word);
296 void OnSpellCheckEnableAutoSpellCorrect(bool enable);
[email protected]85c55dc2009-11-06 03:05:46297
[email protected]6217d392010-03-25 22:08:35298 void OnGpuChannelEstablished(const IPC::ChannelHandle& channel_handle);
299
[email protected]b3df5a42010-05-11 14:31:09300 void OnGetAccessibilityTree();
301
initial.commit09911bf2008-07-26 23:55:29302 // Gather usage statistics from the in-memory cache and inform our host.
303 // These functions should be call periodically so that the host can make
304 // decisions about how to allocation resources using current information.
305 void InformHostOfCacheStats();
306
[email protected]90a3fbb12009-02-28 01:13:47307 // We initialize WebKit as late as possible.
308 void EnsureWebKitInitialized();
309
[email protected]bee16aab2009-08-26 15:55:03310 // A task we invoke periodically to assist with idle cleanup.
311 void IdleHandler();
312
[email protected]71d6d852009-12-07 22:12:36313 // Schedule a call to IdleHandler with the given initial delay.
314 void ScheduleIdleHandler(double initial_delay_s);
315
initial.commit09911bf2008-07-26 23:55:29316 // These objects live solely on the render thread.
[email protected]bee16aab2009-08-26 15:55:03317 scoped_ptr<ScopedRunnableMethodFactory<RenderThread> > task_factory_;
[email protected]42f1d7822009-07-23 18:17:55318 scoped_ptr<VisitedLinkSlave> visited_link_slave_;
319 scoped_ptr<UserScriptSlave> user_script_slave_;
[email protected]74be069e82010-06-25 00:12:49320 scoped_ptr<RendererNetPredictor> renderer_net_predictor_;
[email protected]f430b5712009-08-21 21:46:31321 scoped_ptr<AppCacheDispatcher> appcache_dispatcher_;
[email protected]70c19a932010-05-14 12:59:11322 scoped_ptr<IndexedDBDispatcher> indexed_db_dispatcher_;
[email protected]9291ed12009-07-23 17:33:22323 scoped_refptr<DevToolsAgentFilter> devtools_agent_filter_;
[email protected]42f1d7822009-07-23 18:17:55324 scoped_ptr<RendererHistogramSnapshots> histogram_snapshots_;
325 scoped_ptr<RendererWebKitClientImpl> webkit_client_;
[email protected]b7c7bcf2009-10-03 07:07:34326 scoped_ptr<WebKit::WebStorageEventDispatcher> dom_storage_event_dispatcher_;
[email protected]2b437e232010-04-02 01:30:08327 scoped_ptr<WebDatabaseObserverImpl> web_database_observer_impl_;
[email protected]85c55dc2009-11-06 03:05:46328 scoped_ptr<SpellCheck> spellchecker_;
[email protected]9291ed12009-07-23 17:33:22329
[email protected]e2b2d4a2009-10-24 03:32:59330 // Used on the renderer and IPC threads.
[email protected]017022b2009-07-27 23:06:34331 scoped_refptr<DBMessageFilter> db_message_filter_;
[email protected]dd9241932010-02-24 19:23:13332 scoped_refptr<CookieMessageFilter> cookie_message_filter_;
[email protected]017022b2009-07-27 23:06:34333
[email protected]5fa1c542009-05-05 20:36:07334#if defined(OS_POSIX)
335 scoped_refptr<IPC::ChannelProxy::MessageFilter>
336 suicide_on_channel_error_filter_;
337#endif
338
[email protected]b547fd42009-04-23 23:16:27339 // If true, then a GetPlugins call is allowed to rescan the disk.
340 bool plugin_refresh_allowed_;
341
[email protected]bee16aab2009-08-26 15:55:03342 // Is there a pending task for doing CacheStats.
343 bool cache_stats_task_pending_;
344
345 // The count of RenderWidgets running through this thread.
346 int widget_count_;
347
348 // The count of hidden RenderWidgets running through this thread.
349 int hidden_widget_count_;
350
351 // The current value of the idle notification timer delay.
352 double idle_notification_delay_in_s_;
353
[email protected]71d6d852009-12-07 22:12:36354 // True if this renderer is running extensions.
355 bool is_extension_process_;
356
[email protected]b2a74ca2010-03-12 17:57:09357 // True if this renderer is incognito.
358 bool is_incognito_process_;
359
[email protected]80fc08c52010-03-09 07:43:50360 bool suspend_webkit_shared_timer_;
361 bool notify_webkit_of_modal_loop_;
[email protected]c1f50aa2010-02-18 03:46:57362
[email protected]71d6d852009-12-07 22:12:36363 // Timer that periodically calls IdleHandler.
364 base::RepeatingTimer<RenderThread> idle_timer_;
365
366 // Same as above, but on a longer timer and will run even if the process is
367 // not idle, to ensure that IdleHandle gets called eventually.
368 base::RepeatingTimer<RenderThread> forced_idle_timer_;
369
[email protected]6217d392010-03-25 22:08:35370 // The channel from the renderer process to the GPU process.
371 scoped_refptr<GpuChannelHost> gpu_channel_;
372
[email protected]3a8eecb2010-04-22 23:56:30373 // A list of extension web extents, which tells us which URLs belong to an
374 // installed app.
[email protected]e13ad79b2010-07-22 21:36:50375 struct ExtensionInfo;
[email protected]3a8eecb2010-04-22 23:56:30376 std::vector<ExtensionInfo> extension_extents_;
377
[email protected]c6a7b862010-08-20 22:19:38378 // A lazily initiated thread on which file operations are run.
379 scoped_ptr<base::Thread> file_thread_;
380
[email protected]1bc83062009-02-06 00:16:37381 DISALLOW_COPY_AND_ASSIGN(RenderThread);
initial.commit09911bf2008-07-26 23:55:29382};
383
[email protected]1d97d2e2008-12-18 23:39:02384#endif // CHROME_RENDERER_RENDER_THREAD_H_