Pin Chrome's shortcut to the Win10 Start menu on install and OS upgrade.
[chromium-blink-merge.git] / content / browser / gpu / browser_gpu_memory_buffer_manager.cc
blob67797d42937129b8dcc17254764621bb84db3316
1 // Copyright 2014 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.
5 #include "content/browser/gpu/browser_gpu_memory_buffer_manager.h"
7 #include "base/atomic_sequence_num.h"
8 #include "base/bind.h"
9 #include "base/command_line.h"
10 #include "base/strings/stringprintf.h"
11 #include "base/synchronization/waitable_event.h"
12 #include "base/threading/thread_restrictions.h"
13 #include "base/trace_event/memory_dump_manager.h"
14 #include "base/trace_event/process_memory_dump.h"
15 #include "base/trace_event/trace_event.h"
16 #include "content/browser/gpu/gpu_process_host.h"
17 #include "content/common/gpu/client/gpu_memory_buffer_impl.h"
18 #include "content/common/gpu/client/gpu_memory_buffer_impl_shared_memory.h"
19 #include "content/common/gpu/gpu_memory_buffer_factory_shared_memory.h"
20 #include "content/public/browser/browser_thread.h"
21 #include "content/public/common/content_switches.h"
22 #include "gpu/GLES2/gl2extchromium.h"
24 #if defined(OS_MACOSX)
25 #include "content/common/gpu/gpu_memory_buffer_factory_io_surface.h"
26 #endif
28 #if defined(OS_ANDROID)
29 #include "content/common/gpu/gpu_memory_buffer_factory_surface_texture.h"
30 #endif
32 #if defined(USE_OZONE)
33 #include "content/common/gpu/gpu_memory_buffer_factory_ozone_native_pixmap.h"
34 #endif
36 namespace content {
37 namespace {
39 void GpuMemoryBufferDeleted(
40 scoped_refptr<base::SingleThreadTaskRunner> destruction_task_runner,
41 const GpuMemoryBufferImpl::DestructionCallback& destruction_callback,
42 uint32 sync_point) {
43 destruction_task_runner->PostTask(
44 FROM_HERE, base::Bind(destruction_callback, sync_point));
47 bool IsGpuMemoryBufferFactoryConfigurationSupported(
48 gfx::GpuMemoryBufferType type,
49 const GpuMemoryBufferFactory::Configuration& configuration) {
50 switch (type) {
51 case gfx::SHARED_MEMORY_BUFFER:
52 return GpuMemoryBufferFactorySharedMemory::
53 IsGpuMemoryBufferConfigurationSupported(configuration.format,
54 configuration.usage);
55 #if defined(OS_MACOSX)
56 case gfx::IO_SURFACE_BUFFER:
57 return GpuMemoryBufferFactoryIOSurface::
58 IsGpuMemoryBufferConfigurationSupported(configuration.format,
59 configuration.usage);
60 #endif
61 #if defined(OS_ANDROID)
62 case gfx::SURFACE_TEXTURE_BUFFER:
63 return GpuMemoryBufferFactorySurfaceTexture::
64 IsGpuMemoryBufferConfigurationSupported(configuration.format,
65 configuration.usage);
66 #endif
67 #if defined(USE_OZONE)
68 case gfx::OZONE_NATIVE_PIXMAP:
69 return GpuMemoryBufferFactoryOzoneNativePixmap::
70 IsGpuMemoryBufferConfigurationSupported(configuration.format,
71 configuration.usage);
72 #endif
73 default:
74 NOTREACHED();
75 return false;
79 gfx::GpuMemoryBufferType GetGpuMemoryBufferFactoryType() {
80 std::vector<gfx::GpuMemoryBufferType> supported_types;
81 GpuMemoryBufferFactory::GetSupportedTypes(&supported_types);
82 DCHECK(!supported_types.empty());
84 // The GPU service will always use the preferred type.
85 return supported_types[0];
88 std::vector<GpuMemoryBufferFactory::Configuration>
89 GetSupportedGpuMemoryBufferConfigurations(gfx::GpuMemoryBufferType type) {
90 std::vector<GpuMemoryBufferFactory::Configuration> configurations;
91 if (base::CommandLine::ForCurrentProcess()->HasSwitch(
92 switches::kEnableNativeGpuMemoryBuffers)) {
93 const GpuMemoryBufferFactory::Configuration kNativeConfigurations[] = {
94 {gfx::GpuMemoryBuffer::R_8, gfx::GpuMemoryBuffer::MAP},
95 {gfx::GpuMemoryBuffer::R_8, gfx::GpuMemoryBuffer::PERSISTENT_MAP},
96 {gfx::GpuMemoryBuffer::RGBA_4444, gfx::GpuMemoryBuffer::MAP},
97 {gfx::GpuMemoryBuffer::RGBA_4444, gfx::GpuMemoryBuffer::PERSISTENT_MAP},
98 {gfx::GpuMemoryBuffer::RGBA_8888, gfx::GpuMemoryBuffer::MAP},
99 {gfx::GpuMemoryBuffer::RGBA_8888, gfx::GpuMemoryBuffer::PERSISTENT_MAP},
100 {gfx::GpuMemoryBuffer::BGRA_8888, gfx::GpuMemoryBuffer::MAP},
101 {gfx::GpuMemoryBuffer::BGRA_8888,
102 gfx::GpuMemoryBuffer::PERSISTENT_MAP}};
103 for (auto& configuration : kNativeConfigurations) {
104 if (IsGpuMemoryBufferFactoryConfigurationSupported(type, configuration))
105 configurations.push_back(configuration);
109 #if defined(USE_OZONE) || defined(OS_MACOSX)
110 const GpuMemoryBufferFactory::Configuration kScanoutConfigurations[] = {
111 {gfx::GpuMemoryBuffer::BGRA_8888, gfx::GpuMemoryBuffer::SCANOUT},
112 {gfx::GpuMemoryBuffer::RGBX_8888, gfx::GpuMemoryBuffer::SCANOUT}};
113 for (auto& configuration : kScanoutConfigurations) {
114 if (IsGpuMemoryBufferFactoryConfigurationSupported(type, configuration))
115 configurations.push_back(configuration);
117 #endif
119 return configurations;
122 BrowserGpuMemoryBufferManager* g_gpu_memory_buffer_manager = nullptr;
124 // Global atomic to generate gpu memory buffer unique IDs.
125 base::StaticAtomicSequenceNumber g_next_gpu_memory_buffer_id;
127 } // namespace
129 struct BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferRequest {
130 AllocateGpuMemoryBufferRequest(const gfx::Size& size,
131 gfx::GpuMemoryBuffer::Format format,
132 gfx::GpuMemoryBuffer::Usage usage,
133 int client_id,
134 int surface_id)
135 : event(true, false),
136 size(size),
137 format(format),
138 usage(usage),
139 client_id(client_id),
140 surface_id(surface_id) {}
141 ~AllocateGpuMemoryBufferRequest() {}
142 base::WaitableEvent event;
143 gfx::Size size;
144 gfx::GpuMemoryBuffer::Format format;
145 gfx::GpuMemoryBuffer::Usage usage;
146 int client_id;
147 int surface_id;
148 scoped_ptr<gfx::GpuMemoryBuffer> result;
151 BrowserGpuMemoryBufferManager::BrowserGpuMemoryBufferManager(int gpu_client_id)
152 : factory_type_(GetGpuMemoryBufferFactoryType()),
153 supported_configurations_(
154 GetSupportedGpuMemoryBufferConfigurations(factory_type_)),
155 gpu_client_id_(gpu_client_id),
156 gpu_host_id_(0) {
157 DCHECK(!g_gpu_memory_buffer_manager);
158 g_gpu_memory_buffer_manager = this;
161 BrowserGpuMemoryBufferManager::~BrowserGpuMemoryBufferManager() {
162 g_gpu_memory_buffer_manager = nullptr;
165 // static
166 BrowserGpuMemoryBufferManager* BrowserGpuMemoryBufferManager::current() {
167 return g_gpu_memory_buffer_manager;
170 // static
171 uint32 BrowserGpuMemoryBufferManager::GetImageTextureTarget(
172 gfx::GpuMemoryBuffer::Format format,
173 gfx::GpuMemoryBuffer::Usage usage) {
174 gfx::GpuMemoryBufferType type = GetGpuMemoryBufferFactoryType();
175 for (auto& configuration : GetSupportedGpuMemoryBufferConfigurations(type)) {
176 if (configuration.format != format || configuration.usage != usage)
177 continue;
179 switch (type) {
180 case gfx::SURFACE_TEXTURE_BUFFER:
181 case gfx::OZONE_NATIVE_PIXMAP:
182 // GPU memory buffers that are shared with the GL using EGLImages
183 // require TEXTURE_EXTERNAL_OES.
184 return GL_TEXTURE_EXTERNAL_OES;
185 case gfx::IO_SURFACE_BUFFER:
186 // IOSurface backed images require GL_TEXTURE_RECTANGLE_ARB.
187 return GL_TEXTURE_RECTANGLE_ARB;
188 default:
189 return GL_TEXTURE_2D;
193 return GL_TEXTURE_2D;
196 scoped_ptr<gfx::GpuMemoryBuffer>
197 BrowserGpuMemoryBufferManager::AllocateGpuMemoryBuffer(
198 const gfx::Size& size,
199 gfx::GpuMemoryBuffer::Format format,
200 gfx::GpuMemoryBuffer::Usage usage) {
201 return AllocateGpuMemoryBufferForSurface(size, format, usage, 0);
204 scoped_ptr<gfx::GpuMemoryBuffer>
205 BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForScanout(
206 const gfx::Size& size,
207 gfx::GpuMemoryBuffer::Format format,
208 int32 surface_id) {
209 DCHECK_GT(surface_id, 0);
210 return AllocateGpuMemoryBufferForSurface(
211 size, format, gfx::GpuMemoryBuffer::SCANOUT, surface_id);
214 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForChildProcess(
215 const gfx::Size& size,
216 gfx::GpuMemoryBuffer::Format format,
217 gfx::GpuMemoryBuffer::Usage usage,
218 base::ProcessHandle child_process_handle,
219 int child_client_id,
220 const AllocationCallback& callback) {
221 DCHECK_CURRENTLY_ON(BrowserThread::IO);
223 gfx::GpuMemoryBufferId new_id = g_next_gpu_memory_buffer_id.GetNext();
225 // Use service side allocation if this is a supported configuration.
226 if (IsGpuMemoryBufferConfigurationSupported(format, usage)) {
227 AllocateGpuMemoryBufferOnIO(new_id, size, format, usage, child_client_id, 0,
228 false, callback);
229 return;
232 // Early out if we cannot fallback to shared memory buffer.
233 if (!GpuMemoryBufferImplSharedMemory::IsFormatSupported(format) ||
234 !GpuMemoryBufferImplSharedMemory::IsUsageSupported(usage) ||
235 !GpuMemoryBufferImplSharedMemory::IsSizeValidForFormat(size, format)) {
236 callback.Run(gfx::GpuMemoryBufferHandle());
237 return;
240 BufferMap& buffers = clients_[child_client_id];
241 DCHECK(buffers.find(new_id) == buffers.end());
243 // Allocate shared memory buffer as fallback.
244 buffers[new_id] =
245 BufferInfo(size, gfx::SHARED_MEMORY_BUFFER, format, usage, 0);
246 callback.Run(GpuMemoryBufferImplSharedMemory::AllocateForChildProcess(
247 new_id, size, format, child_process_handle));
250 gfx::GpuMemoryBuffer*
251 BrowserGpuMemoryBufferManager::GpuMemoryBufferFromClientBuffer(
252 ClientBuffer buffer) {
253 return GpuMemoryBufferImpl::FromClientBuffer(buffer);
256 void BrowserGpuMemoryBufferManager::SetDestructionSyncPoint(
257 gfx::GpuMemoryBuffer* buffer,
258 uint32 sync_point) {
259 static_cast<GpuMemoryBufferImpl*>(buffer)
260 ->set_destruction_sync_point(sync_point);
263 bool BrowserGpuMemoryBufferManager::OnMemoryDump(
264 base::trace_event::ProcessMemoryDump* pmd) {
265 DCHECK_CURRENTLY_ON(BrowserThread::IO);
267 for (const auto& client : clients_) {
268 int client_id = client.first;
270 for (const auto& buffer : client.second) {
271 if (buffer.second.type == gfx::EMPTY_BUFFER)
272 continue;
274 gfx::GpuMemoryBufferId buffer_id = buffer.first;
275 base::trace_event::MemoryAllocatorDump* dump =
276 pmd->CreateAllocatorDump(base::StringPrintf(
277 "gpumemorybuffer/client_%d/buffer_%d", client_id, buffer_id));
278 if (!dump)
279 return false;
281 size_t buffer_size_in_bytes = 0;
282 // Note: BufferSizeInBytes returns an approximated size for the buffer
283 // but the factory can be made to return the exact size if this
284 // approximation is not good enough.
285 bool valid_size = GpuMemoryBufferImpl::BufferSizeInBytes(
286 buffer.second.size, buffer.second.format, &buffer_size_in_bytes);
287 DCHECK(valid_size);
289 dump->AddScalar(base::trace_event::MemoryAllocatorDump::kNameSize,
290 base::trace_event::MemoryAllocatorDump::kUnitsBytes,
291 buffer_size_in_bytes);
293 // Create the cross-process ownership edge. If the client creates a
294 // corresponding dump for the same buffer, this will avoid to
295 // double-count them in tracing. If, instead, no other process will emit a
296 // dump with the same guid, the segment will be accounted to the browser.
297 const uint64 client_tracing_process_id = base::trace_event::
298 MemoryDumpManager::ChildProcessIdToTracingProcessId(client_id);
299 base::trace_event::MemoryAllocatorDumpGuid shared_buffer_guid =
300 gfx::GetGpuMemoryBufferGUIDForTracing(client_tracing_process_id,
301 buffer_id);
302 pmd->CreateSharedGlobalAllocatorDump(shared_buffer_guid);
303 pmd->AddOwnershipEdge(dump->guid(), shared_buffer_guid);
307 return true;
310 void BrowserGpuMemoryBufferManager::ChildProcessDeletedGpuMemoryBuffer(
311 gfx::GpuMemoryBufferId id,
312 base::ProcessHandle child_process_handle,
313 int child_client_id,
314 uint32 sync_point) {
315 DCHECK_CURRENTLY_ON(BrowserThread::IO);
317 DestroyGpuMemoryBufferOnIO(id, child_client_id, sync_point);
320 void BrowserGpuMemoryBufferManager::ProcessRemoved(
321 base::ProcessHandle process_handle,
322 int client_id) {
323 DCHECK_CURRENTLY_ON(BrowserThread::IO);
325 ClientMap::iterator client_it = clients_.find(client_id);
326 if (client_it == clients_.end())
327 return;
329 for (const auto& buffer : client_it->second) {
330 // This might happen if buffer is currenlty in the process of being
331 // allocated. The buffer will in that case be cleaned up when allocation
332 // completes.
333 if (buffer.second.type == gfx::EMPTY_BUFFER)
334 continue;
336 GpuProcessHost* host = GpuProcessHost::FromID(buffer.second.gpu_host_id);
337 if (host)
338 host->DestroyGpuMemoryBuffer(buffer.first, client_id, 0);
341 clients_.erase(client_it);
344 scoped_ptr<gfx::GpuMemoryBuffer>
345 BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForSurface(
346 const gfx::Size& size,
347 gfx::GpuMemoryBuffer::Format format,
348 gfx::GpuMemoryBuffer::Usage usage,
349 int32 surface_id) {
350 DCHECK(!BrowserThread::CurrentlyOn(BrowserThread::IO));
352 AllocateGpuMemoryBufferRequest request(size, format, usage, gpu_client_id_,
353 surface_id);
354 BrowserThread::PostTask(
355 BrowserThread::IO, FROM_HERE,
356 base::Bind(
357 &BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForSurfaceOnIO,
358 base::Unretained(this), // Safe as we wait for result below.
359 base::Unretained(&request)));
361 // We're blocking the UI thread, which is generally undesirable.
362 TRACE_EVENT0(
363 "browser",
364 "BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForSurface");
365 base::ThreadRestrictions::ScopedAllowWait allow_wait;
366 request.event.Wait();
367 return request.result.Pass();
370 bool BrowserGpuMemoryBufferManager::IsGpuMemoryBufferConfigurationSupported(
371 gfx::GpuMemoryBuffer::Format format,
372 gfx::GpuMemoryBuffer::Usage usage) const {
373 for (auto& configuration : supported_configurations_) {
374 if (configuration.format == format && configuration.usage == usage)
375 return true;
377 return false;
380 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForSurfaceOnIO(
381 AllocateGpuMemoryBufferRequest* request) {
382 DCHECK_CURRENTLY_ON(BrowserThread::IO);
384 gfx::GpuMemoryBufferId new_id = g_next_gpu_memory_buffer_id.GetNext();
386 // Use service side allocation if this is a supported configuration.
387 if (IsGpuMemoryBufferConfigurationSupported(request->format,
388 request->usage)) {
389 // Note: Unretained is safe as this is only used for synchronous allocation
390 // from a non-IO thread.
391 AllocateGpuMemoryBufferOnIO(
392 new_id, request->size, request->format, request->usage,
393 request->client_id, request->surface_id, false,
394 base::Bind(&BrowserGpuMemoryBufferManager::
395 GpuMemoryBufferAllocatedForSurfaceOnIO,
396 base::Unretained(this), base::Unretained(request)));
397 return;
400 DCHECK(GpuMemoryBufferImplSharedMemory::IsFormatSupported(request->format))
401 << request->format;
402 DCHECK(GpuMemoryBufferImplSharedMemory::IsUsageSupported(request->usage))
403 << request->usage;
405 BufferMap& buffers = clients_[request->client_id];
406 DCHECK(buffers.find(new_id) == buffers.end());
408 // Allocate shared memory buffer as fallback.
409 buffers[new_id] = BufferInfo(request->size, gfx::SHARED_MEMORY_BUFFER,
410 request->format, request->usage, 0);
411 // Note: Unretained is safe as IO thread is stopped before manager is
412 // destroyed.
413 request->result = GpuMemoryBufferImplSharedMemory::Create(
414 new_id, request->size, request->format,
415 base::Bind(
416 &GpuMemoryBufferDeleted,
417 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::IO),
418 base::Bind(&BrowserGpuMemoryBufferManager::DestroyGpuMemoryBufferOnIO,
419 base::Unretained(this), new_id, request->client_id)));
420 request->event.Signal();
423 void BrowserGpuMemoryBufferManager::GpuMemoryBufferAllocatedForSurfaceOnIO(
424 AllocateGpuMemoryBufferRequest* request,
425 const gfx::GpuMemoryBufferHandle& handle) {
426 DCHECK_CURRENTLY_ON(BrowserThread::IO);
428 // Early out if factory failed to allocate the buffer.
429 if (handle.is_null()) {
430 request->event.Signal();
431 return;
434 // Note: Unretained is safe as IO thread is stopped before manager is
435 // destroyed.
436 request->result = GpuMemoryBufferImpl::CreateFromHandle(
437 handle, request->size, request->format, request->usage,
438 base::Bind(
439 &GpuMemoryBufferDeleted,
440 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::IO),
441 base::Bind(&BrowserGpuMemoryBufferManager::DestroyGpuMemoryBufferOnIO,
442 base::Unretained(this), handle.id, request->client_id)));
443 request->event.Signal();
446 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferOnIO(
447 gfx::GpuMemoryBufferId id,
448 const gfx::Size& size,
449 gfx::GpuMemoryBuffer::Format format,
450 gfx::GpuMemoryBuffer::Usage usage,
451 int client_id,
452 int surface_id,
453 bool reused_gpu_process,
454 const AllocationCallback& callback) {
455 DCHECK_CURRENTLY_ON(BrowserThread::IO);
457 BufferMap& buffers = clients_[client_id];
458 DCHECK(buffers.find(id) == buffers.end());
460 GpuProcessHost* host = GpuProcessHost::FromID(gpu_host_id_);
461 if (!host) {
462 host = GpuProcessHost::Get(GpuProcessHost::GPU_PROCESS_KIND_SANDBOXED,
463 CAUSE_FOR_GPU_LAUNCH_GPU_MEMORY_BUFFER_ALLOCATE);
464 if (!host) {
465 LOG(ERROR) << "Failed to launch GPU process.";
466 callback.Run(gfx::GpuMemoryBufferHandle());
467 return;
469 gpu_host_id_ = host->host_id();
470 reused_gpu_process = false;
471 } else {
472 if (reused_gpu_process) {
473 // We come here if we retried to allocate the buffer because of a
474 // failure in GpuMemoryBufferAllocatedOnIO, but we ended up with the
475 // same process ID, meaning the failure was not because of a channel
476 // error, but another reason. So fail now.
477 LOG(ERROR) << "Failed to allocate GpuMemoryBuffer.";
478 callback.Run(gfx::GpuMemoryBufferHandle());
479 return;
481 reused_gpu_process = true;
484 // Note: Handling of cases where the client is removed before the allocation
485 // completes is less subtle if we set the buffer type to EMPTY_BUFFER here
486 // and verify that this has not changed when allocation completes.
487 buffers[id] = BufferInfo(size, gfx::EMPTY_BUFFER, format, usage, 0);
489 // Note: Unretained is safe as IO thread is stopped before manager is
490 // destroyed.
491 host->CreateGpuMemoryBuffer(
492 id, size, format, usage, client_id, surface_id,
493 base::Bind(&BrowserGpuMemoryBufferManager::GpuMemoryBufferAllocatedOnIO,
494 base::Unretained(this), id, client_id, surface_id,
495 gpu_host_id_, reused_gpu_process, callback));
498 void BrowserGpuMemoryBufferManager::GpuMemoryBufferAllocatedOnIO(
499 gfx::GpuMemoryBufferId id,
500 int client_id,
501 int surface_id,
502 int gpu_host_id,
503 bool reused_gpu_process,
504 const AllocationCallback& callback,
505 const gfx::GpuMemoryBufferHandle& handle) {
506 DCHECK_CURRENTLY_ON(BrowserThread::IO);
508 ClientMap::iterator client_it = clients_.find(client_id);
510 // This can happen if client is removed while the buffer is being allocated.
511 if (client_it == clients_.end()) {
512 if (!handle.is_null()) {
513 GpuProcessHost* host = GpuProcessHost::FromID(gpu_host_id);
514 if (host)
515 host->DestroyGpuMemoryBuffer(handle.id, client_id, 0);
517 callback.Run(gfx::GpuMemoryBufferHandle());
518 return;
521 BufferMap& buffers = client_it->second;
523 BufferMap::iterator buffer_it = buffers.find(id);
524 DCHECK(buffer_it != buffers.end());
525 DCHECK_EQ(buffer_it->second.type, gfx::EMPTY_BUFFER);
527 // If the handle isn't valid, that means that the GPU process crashed or is
528 // misbehaving.
529 bool valid_handle = !handle.is_null() && handle.id == id;
530 if (!valid_handle) {
531 // If we failed after re-using the GPU process, it may have died in the
532 // mean time. Retry to have a chance to create a fresh GPU process.
533 if (handle.is_null() && reused_gpu_process) {
534 DVLOG(1) << "Failed to create buffer through existing GPU process. "
535 "Trying to restart GPU process.";
536 // If the GPU process has already been restarted, retry without failure
537 // when GPU process host ID already exists.
538 if (gpu_host_id != gpu_host_id_)
539 reused_gpu_process = false;
540 gfx::Size size = buffer_it->second.size;
541 gfx::GpuMemoryBuffer::Format format = buffer_it->second.format;
542 gfx::GpuMemoryBuffer::Usage usage = buffer_it->second.usage;
543 // Remove the buffer entry and call AllocateGpuMemoryBufferOnIO again.
544 buffers.erase(buffer_it);
545 AllocateGpuMemoryBufferOnIO(id, size, format, usage, client_id,
546 surface_id, reused_gpu_process, callback);
547 } else {
548 // Remove the buffer entry and run the allocation callback with an empty
549 // handle to indicate failure.
550 buffers.erase(buffer_it);
551 callback.Run(gfx::GpuMemoryBufferHandle());
553 return;
556 // Store the type and host id of this buffer so it can be cleaned up if the
557 // client is removed.
558 buffer_it->second.type = handle.type;
559 buffer_it->second.gpu_host_id = gpu_host_id;
561 callback.Run(handle);
564 void BrowserGpuMemoryBufferManager::DestroyGpuMemoryBufferOnIO(
565 gfx::GpuMemoryBufferId id,
566 int client_id,
567 uint32 sync_point) {
568 DCHECK_CURRENTLY_ON(BrowserThread::IO);
569 DCHECK(clients_.find(client_id) != clients_.end());
571 BufferMap& buffers = clients_[client_id];
573 BufferMap::iterator buffer_it = buffers.find(id);
574 if (buffer_it == buffers.end()) {
575 LOG(ERROR) << "Invalid GpuMemoryBuffer ID for client.";
576 return;
579 // This can happen if a client managed to call this while a buffer is in the
580 // process of being allocated.
581 if (buffer_it->second.type == gfx::EMPTY_BUFFER) {
582 LOG(ERROR) << "Invalid GpuMemoryBuffer type.";
583 return;
586 GpuProcessHost* host = GpuProcessHost::FromID(buffer_it->second.gpu_host_id);
587 if (host)
588 host->DestroyGpuMemoryBuffer(id, client_id, sync_point);
590 buffers.erase(buffer_it);
593 } // namespace content