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/gpu/gpu_child_thread.h"
6
7 #include "base/bind.h"
8 #include "base/lazy_instance.h"
9 #include "base/threading/worker_pool.h"
10 #include "build/build_config.h"
11 #include "content/child/child_process.h"
12 #include "content/child/thread_safe_sender.h"
13 #include "content/common/gpu/gpu_messages.h"
14 #include "content/gpu/gpu_watchdog_thread.h"
15 #include "content/public/common/content_client.h"
16 #include "content/public/common/content_switches.h"
17 #include "gpu/config/gpu_info_collector.h"
18 #include "ipc/ipc_channel_handle.h"
19 #include "ipc/ipc_sync_message_filter.h"
20 #include "ui/gl/gl_implementation.h"
21
22 namespace content {
23 namespace {
24
25 static base::LazyInstance<scoped_refptr<ThreadSafeSender> >
26 g_thread_safe_sender = LAZY_INSTANCE_INITIALIZER;
27
GpuProcessLogMessageHandler(int severity,const char * file,int line,size_t message_start,const std::string & str)28 bool GpuProcessLogMessageHandler(int severity,
29 const char* file, int line,
30 size_t message_start,
31 const std::string& str) {
32 std::string header = str.substr(0, message_start);
33 std::string message = str.substr(message_start);
34
35 g_thread_safe_sender.Get()->Send(new GpuHostMsg_OnLogMessage(
36 severity, header, message));
37
38 return false;
39 }
40
41 } // namespace
42
GpuChildThread(GpuWatchdogThread * watchdog_thread,bool dead_on_arrival,const gpu::GPUInfo & gpu_info,const DeferredMessages & deferred_messages)43 GpuChildThread::GpuChildThread(GpuWatchdogThread* watchdog_thread,
44 bool dead_on_arrival,
45 const gpu::GPUInfo& gpu_info,
46 const DeferredMessages& deferred_messages)
47 : dead_on_arrival_(dead_on_arrival),
48 gpu_info_(gpu_info),
49 deferred_messages_(deferred_messages),
50 in_browser_process_(false) {
51 watchdog_thread_ = watchdog_thread;
52 #if defined(OS_WIN)
53 target_services_ = NULL;
54 #endif
55 g_thread_safe_sender.Get() = thread_safe_sender();
56 }
57
GpuChildThread(const std::string & channel_id)58 GpuChildThread::GpuChildThread(const std::string& channel_id)
59 : ChildThread(channel_id),
60 dead_on_arrival_(false),
61 in_browser_process_(true) {
62 #if defined(OS_WIN)
63 target_services_ = NULL;
64 #endif
65 DCHECK(
66 CommandLine::ForCurrentProcess()->HasSwitch(switches::kSingleProcess) ||
67 CommandLine::ForCurrentProcess()->HasSwitch(switches::kInProcessGPU));
68 // For single process and in-process GPU mode, we need to load and
69 // initialize the GL implementation and locate the GL entry points here.
70 if (!gfx::GLSurface::InitializeOneOff()) {
71 VLOG(1) << "gfx::GLSurface::InitializeOneOff()";
72 }
73 g_thread_safe_sender.Get() = thread_safe_sender();
74 }
75
~GpuChildThread()76 GpuChildThread::~GpuChildThread() {
77 }
78
Shutdown()79 void GpuChildThread::Shutdown() {
80 ChildThread::Shutdown();
81 logging::SetLogMessageHandler(NULL);
82 }
83
Init(const base::Time & process_start_time)84 void GpuChildThread::Init(const base::Time& process_start_time) {
85 process_start_time_ = process_start_time;
86 }
87
Send(IPC::Message * msg)88 bool GpuChildThread::Send(IPC::Message* msg) {
89 // The GPU process must never send a synchronous IPC message to the browser
90 // process. This could result in deadlock.
91 DCHECK(!msg->is_sync());
92
93 return ChildThread::Send(msg);
94 }
95
OnControlMessageReceived(const IPC::Message & msg)96 bool GpuChildThread::OnControlMessageReceived(const IPC::Message& msg) {
97 bool msg_is_ok = true;
98 bool handled = true;
99 IPC_BEGIN_MESSAGE_MAP_EX(GpuChildThread, msg, msg_is_ok)
100 IPC_MESSAGE_HANDLER(GpuMsg_Initialize, OnInitialize)
101 IPC_MESSAGE_HANDLER(GpuMsg_CollectGraphicsInfo, OnCollectGraphicsInfo)
102 IPC_MESSAGE_HANDLER(GpuMsg_GetVideoMemoryUsageStats,
103 OnGetVideoMemoryUsageStats)
104 IPC_MESSAGE_HANDLER(GpuMsg_Clean, OnClean)
105 IPC_MESSAGE_HANDLER(GpuMsg_Crash, OnCrash)
106 IPC_MESSAGE_HANDLER(GpuMsg_Hang, OnHang)
107 IPC_MESSAGE_HANDLER(GpuMsg_DisableWatchdog, OnDisableWatchdog)
108 IPC_MESSAGE_UNHANDLED(handled = false)
109 IPC_END_MESSAGE_MAP_EX()
110
111 if (handled)
112 return true;
113
114 return gpu_channel_manager_.get() &&
115 gpu_channel_manager_->OnMessageReceived(msg);
116 }
117
OnInitialize()118 void GpuChildThread::OnInitialize() {
119 Send(new GpuHostMsg_Initialized(!dead_on_arrival_, gpu_info_));
120 while (!deferred_messages_.empty()) {
121 Send(deferred_messages_.front());
122 deferred_messages_.pop();
123 }
124
125 if (dead_on_arrival_) {
126 VLOG(1) << "Exiting GPU process due to errors during initialization";
127 base::MessageLoop::current()->Quit();
128 return;
129 }
130
131 #if defined(OS_ANDROID)
132 base::PlatformThread::SetThreadPriority(
133 base::PlatformThread::CurrentHandle(),
134 base::kThreadPriority_Display);
135 #endif
136
137 // We don't need to pipe log messages if we are running the GPU thread in
138 // the browser process.
139 if (!in_browser_process_)
140 logging::SetLogMessageHandler(GpuProcessLogMessageHandler);
141
142 // Record initialization only after collecting the GPU info because that can
143 // take a significant amount of time.
144 gpu_info_.initialization_time = base::Time::Now() - process_start_time_;
145
146 // Defer creation of the render thread. This is to prevent it from handling
147 // IPC messages before the sandbox has been enabled and all other necessary
148 // initialization has succeeded.
149 gpu_channel_manager_.reset(
150 new GpuChannelManager(this,
151 watchdog_thread_.get(),
152 ChildProcess::current()->io_message_loop_proxy(),
153 ChildProcess::current()->GetShutDownEvent()));
154
155 // Ensure the browser process receives the GPU info before a reply to any
156 // subsequent IPC it might send.
157 if (!in_browser_process_)
158 Send(new GpuHostMsg_GraphicsInfoCollected(gpu_info_));
159 }
160
StopWatchdog()161 void GpuChildThread::StopWatchdog() {
162 if (watchdog_thread_.get()) {
163 watchdog_thread_->Stop();
164 }
165 }
166
OnCollectGraphicsInfo()167 void GpuChildThread::OnCollectGraphicsInfo() {
168 #if defined(OS_WIN)
169 // GPU full info collection should only happen on un-sandboxed GPU process
170 // or single process/in-process gpu mode on Windows.
171 CommandLine* command_line = CommandLine::ForCurrentProcess();
172 DCHECK(command_line->HasSwitch(switches::kDisableGpuSandbox) ||
173 in_browser_process_);
174 #endif // OS_WIN
175
176 if (!gpu::CollectContextGraphicsInfo(&gpu_info_))
177 VLOG(1) << "gpu::CollectGraphicsInfo failed";
178 GetContentClient()->SetGpuInfo(gpu_info_);
179
180 #if defined(OS_WIN)
181 // This is slow, but it's the only thing the unsandboxed GPU process does,
182 // and GpuDataManager prevents us from sending multiple collecting requests,
183 // so it's OK to be blocking.
184 gpu::GetDxDiagnostics(&gpu_info_.dx_diagnostics);
185 gpu_info_.finalized = true;
186 #endif // OS_WIN
187
188 Send(new GpuHostMsg_GraphicsInfoCollected(gpu_info_));
189
190 #if defined(OS_WIN)
191 if (!in_browser_process_) {
192 // The unsandboxed GPU process fulfilled its duty. Rest in peace.
193 base::MessageLoop::current()->Quit();
194 }
195 #endif // OS_WIN
196 }
197
OnGetVideoMemoryUsageStats()198 void GpuChildThread::OnGetVideoMemoryUsageStats() {
199 GPUVideoMemoryUsageStats video_memory_usage_stats;
200 if (gpu_channel_manager_)
201 gpu_channel_manager_->gpu_memory_manager()->GetVideoMemoryUsageStats(
202 &video_memory_usage_stats);
203 Send(new GpuHostMsg_VideoMemoryUsageStats(video_memory_usage_stats));
204 }
205
OnClean()206 void GpuChildThread::OnClean() {
207 VLOG(1) << "GPU: Removing all contexts";
208 if (gpu_channel_manager_)
209 gpu_channel_manager_->LoseAllContexts();
210 }
211
OnCrash()212 void GpuChildThread::OnCrash() {
213 VLOG(1) << "GPU: Simulating GPU crash";
214 // Good bye, cruel world.
215 volatile int* it_s_the_end_of_the_world_as_we_know_it = NULL;
216 *it_s_the_end_of_the_world_as_we_know_it = 0xdead;
217 }
218
OnHang()219 void GpuChildThread::OnHang() {
220 VLOG(1) << "GPU: Simulating GPU hang";
221 for (;;) {
222 // Do not sleep here. The GPU watchdog timer tracks the amount of user
223 // time this thread is using and it doesn't use much while calling Sleep.
224 }
225 }
226
OnDisableWatchdog()227 void GpuChildThread::OnDisableWatchdog() {
228 VLOG(1) << "GPU: Disabling watchdog thread";
229 if (watchdog_thread_.get()) {
230 // Disarm the watchdog before shutting down the message loop. This prevents
231 // the future posting of tasks to the message loop.
232 if (watchdog_thread_->message_loop())
233 watchdog_thread_->PostAcknowledge();
234 // Prevent rearming.
235 watchdog_thread_->Stop();
236 }
237 }
238
239 } // namespace content
240
241