Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "content/browser/gpu/browser_gpu_memory_buffer_manager.h" | 5 #include "content/browser/gpu/browser_gpu_memory_buffer_manager.h" |
| 6 | 6 |
| 7 #include "base/atomic_sequence_num.h" | 7 #include "base/atomic_sequence_num.h" |
| 8 #include "base/bind.h" | 8 #include "base/bind.h" |
| 9 #include "base/debug/trace_event.h" | 9 #include "base/debug/trace_event.h" |
| 10 #include "base/lazy_instance.h" | 10 #include "base/lazy_instance.h" |
| 11 #include "base/synchronization/waitable_event.h" | 11 #include "base/synchronization/waitable_event.h" |
| 12 #include "base/threading/thread_restrictions.h" | 12 #include "base/threading/thread_restrictions.h" |
| 13 #include "content/common/gpu/client/gpu_memory_buffer_impl.h" | 13 #include "content/common/gpu/client/gpu_memory_buffer_impl.h" |
| 14 #include "content/public/browser/browser_thread.h" | 14 #include "content/public/browser/browser_thread.h" |
| 15 | 15 |
| 16 namespace content { | 16 namespace content { |
| 17 namespace { | 17 namespace { |
| 18 | 18 |
| 19 void GpuMemoryBufferAllocatedForChildProcess( | |
| 20 const BrowserGpuMemoryBufferManager::AllocationCallback& callback, | |
| 21 const gfx::GpuMemoryBufferHandle& handle) { | |
| 22 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
| 23 | |
| 24 callback.Run(handle); | |
| 25 } | |
| 26 | |
| 27 BrowserGpuMemoryBufferManager* g_gpu_memory_buffer_manager = nullptr; | 19 BrowserGpuMemoryBufferManager* g_gpu_memory_buffer_manager = nullptr; |
| 28 | 20 |
| 29 // Global atomic to generate gpu memory buffer unique IDs. | 21 // Global atomic to generate gpu memory buffer unique IDs. |
| 30 base::StaticAtomicSequenceNumber g_next_gpu_memory_buffer_id; | 22 base::StaticAtomicSequenceNumber g_next_gpu_memory_buffer_id; |
| 31 | 23 |
| 32 } // namespace | 24 } // namespace |
| 33 | 25 |
| 34 struct BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferRequest { | 26 struct BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferRequest { |
| 35 AllocateGpuMemoryBufferRequest(const gfx::Size& size, | 27 AllocateGpuMemoryBufferRequest(const gfx::Size& size, |
| 36 gfx::GpuMemoryBuffer::Format format, | 28 gfx::GpuMemoryBuffer::Format format, |
| (...skipping 52 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 89 | 81 |
| 90 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForChildProcess( | 82 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferForChildProcess( |
| 91 const gfx::Size& size, | 83 const gfx::Size& size, |
| 92 gfx::GpuMemoryBuffer::Format format, | 84 gfx::GpuMemoryBuffer::Format format, |
| 93 gfx::GpuMemoryBuffer::Usage usage, | 85 gfx::GpuMemoryBuffer::Usage usage, |
| 94 base::ProcessHandle child_process_handle, | 86 base::ProcessHandle child_process_handle, |
| 95 int child_client_id, | 87 int child_client_id, |
| 96 const AllocationCallback& callback) { | 88 const AllocationCallback& callback) { |
| 97 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 89 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
| 98 | 90 |
| 91 gfx::GpuMemoryBufferId new_id = g_next_gpu_memory_buffer_id.GetNext(); | |
| 92 | |
| 93 BufferMap& buffers = clients_[child_client_id]; | |
| 94 DCHECK(buffers.find(new_id) == buffers.end()); | |
| 95 | |
| 96 // Note: Handling of cases where the child process is removed before the | |
| 97 // allocation completes is less subtle if we set the buffer type to | |
| 98 // EMPTY_BUFFER here and verify that this has not changed when allocation | |
| 99 // completes. | |
| 100 buffers[new_id] = gfx::EMPTY_BUFFER; | |
| 101 | |
| 99 GpuMemoryBufferImpl::AllocateForChildProcess( | 102 GpuMemoryBufferImpl::AllocateForChildProcess( |
| 100 g_next_gpu_memory_buffer_id.GetNext(), | 103 new_id, |
| 101 size, | 104 size, |
| 102 format, | 105 format, |
| 103 usage, | 106 usage, |
| 104 child_process_handle, | 107 child_process_handle, |
| 105 child_client_id, | 108 child_client_id, |
| 106 base::Bind(&GpuMemoryBufferAllocatedForChildProcess, callback)); | 109 base::Bind(&BrowserGpuMemoryBufferManager:: |
| 110 GpuMemoryBufferAllocatedForChildProcess, | |
| 111 base::Unretained(this), | |
| 112 child_process_handle, | |
| 113 child_client_id, | |
| 114 callback)); | |
| 107 } | 115 } |
| 108 | 116 |
| 109 gfx::GpuMemoryBuffer* | 117 gfx::GpuMemoryBuffer* |
| 110 BrowserGpuMemoryBufferManager::GpuMemoryBufferFromClientBuffer( | 118 BrowserGpuMemoryBufferManager::GpuMemoryBufferFromClientBuffer( |
| 111 ClientBuffer buffer) { | 119 ClientBuffer buffer) { |
| 112 return GpuMemoryBufferImpl::FromClientBuffer(buffer); | 120 return GpuMemoryBufferImpl::FromClientBuffer(buffer); |
| 113 } | 121 } |
| 114 | 122 |
| 123 void BrowserGpuMemoryBufferManager::SetDestructionSyncPoint( | |
| 124 gfx::GpuMemoryBuffer* buffer, | |
| 125 uint32 sync_point) { | |
| 126 static_cast<GpuMemoryBufferImpl*>(buffer) | |
| 127 ->set_destruction_sync_point(sync_point); | |
| 128 } | |
| 129 | |
| 115 void BrowserGpuMemoryBufferManager::ChildProcessDeletedGpuMemoryBuffer( | 130 void BrowserGpuMemoryBufferManager::ChildProcessDeletedGpuMemoryBuffer( |
| 116 gfx::GpuMemoryBufferType type, | |
| 117 gfx::GpuMemoryBufferId id, | 131 gfx::GpuMemoryBufferId id, |
| 118 base::ProcessHandle child_process_handle, | 132 base::ProcessHandle child_process_handle, |
| 119 int child_client_id, | 133 int child_client_id, |
| 120 uint32 sync_point) { | 134 uint32 sync_point) { |
| 121 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 135 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
| 136 DCHECK(clients_.find(child_client_id) != clients_.end()); | |
| 137 | |
| 138 BufferMap& buffers = clients_[child_client_id]; | |
| 139 | |
| 140 BufferMap::iterator buffer_it = buffers.find(id); | |
| 141 if (buffer_it == buffers.end()) { | |
| 142 LOG(ERROR) << "Invalid GpuMemoryBuffer ID for child process."; | |
| 143 return; | |
| 144 } | |
| 145 | |
| 146 // This can happen if a child process manage to trigger a call to this while | |
|
alexst (slow to review)
2014/11/05 13:51:23
nit: ..if a child process managed (or manages).
reveman
2014/11/05 16:22:51
Done.
| |
| 147 // a buffer is in the process of being allocated. | |
| 148 if (buffer_it->second == gfx::EMPTY_BUFFER) { | |
| 149 LOG(ERROR) << "Invalid GpuMemoryBuffer type."; | |
| 150 return; | |
| 151 } | |
| 122 | 152 |
| 123 GpuMemoryBufferImpl::DeletedByChildProcess( | 153 GpuMemoryBufferImpl::DeletedByChildProcess( |
| 124 type, id, child_process_handle, child_client_id, sync_point); | 154 buffer_it->second, id, child_process_handle, child_client_id, sync_point); |
| 155 | |
| 156 buffers.erase(buffer_it); | |
| 125 } | 157 } |
| 126 | 158 |
| 127 void BrowserGpuMemoryBufferManager::ProcessRemoved( | 159 void BrowserGpuMemoryBufferManager::ProcessRemoved( |
| 128 base::ProcessHandle process_handle) { | 160 base::ProcessHandle process_handle, |
| 129 // TODO(reveman): Handle child process removal correctly. | 161 int client_id) { |
| 162 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
| 163 | |
| 164 ClientMap::iterator client_it = clients_.find(client_id); | |
| 165 if (client_it == clients_.end()) | |
| 166 return; | |
| 167 | |
| 168 for (auto &buffer_it : client_it->second) { | |
| 169 // This might happen if buffer is currenlty in the process of being | |
| 170 // allocated. The buffer will in that case be cleaned up when allocation | |
| 171 // completes. | |
| 172 if (buffer_it.second == gfx::EMPTY_BUFFER) | |
| 173 continue; | |
| 174 | |
| 175 GpuMemoryBufferImpl::DeletedByChildProcess(buffer_it.second, | |
| 176 buffer_it.first, | |
| 177 process_handle, | |
| 178 client_id, | |
| 179 0); | |
| 180 } | |
| 181 | |
| 182 clients_.erase(client_it); | |
| 183 } | |
| 184 | |
| 185 void BrowserGpuMemoryBufferManager::GpuMemoryBufferAllocatedForChildProcess( | |
| 186 base::ProcessHandle child_process_handle, | |
| 187 int child_client_id, | |
| 188 const AllocationCallback& callback, | |
| 189 const gfx::GpuMemoryBufferHandle& handle) { | |
| 190 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | |
| 191 | |
| 192 ClientMap::iterator client_it = clients_.find(child_client_id); | |
| 193 | |
| 194 // This can happen if the child process is removed while the buffer is being | |
| 195 // allocated. | |
| 196 if (client_it == clients_.end()) { | |
| 197 if (!handle.is_null()) { | |
| 198 GpuMemoryBufferImpl::DeletedByChildProcess(handle.type, | |
| 199 handle.id, | |
| 200 child_process_handle, | |
| 201 child_client_id, | |
| 202 0); | |
| 203 } | |
| 204 callback.Run(gfx::GpuMemoryBufferHandle()); | |
| 205 return; | |
| 206 } | |
| 207 | |
| 208 BufferMap& buffers = client_it->second; | |
| 209 | |
| 210 BufferMap::iterator buffer_it = buffers.find(handle.id); | |
| 211 DCHECK(buffer_it != buffers.end()); | |
| 212 DCHECK_EQ(buffer_it->second, gfx::EMPTY_BUFFER); | |
| 213 | |
| 214 if (handle.is_null()) { | |
| 215 buffers.erase(buffer_it); | |
| 216 callback.Run(gfx::GpuMemoryBufferHandle()); | |
| 217 return; | |
| 218 } | |
| 219 | |
| 220 // Store the type for this buffer so it can be cleaned up if the child | |
| 221 // process is removed. | |
| 222 buffer_it->second = handle.type; | |
| 223 | |
| 224 callback.Run(handle); | |
| 130 } | 225 } |
| 131 | 226 |
| 132 // static | 227 // static |
| 133 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferOnIO( | 228 void BrowserGpuMemoryBufferManager::AllocateGpuMemoryBufferOnIO( |
| 134 AllocateGpuMemoryBufferRequest* request) { | 229 AllocateGpuMemoryBufferRequest* request) { |
| 135 GpuMemoryBufferImpl::Create( | 230 GpuMemoryBufferImpl::Create( |
| 136 g_next_gpu_memory_buffer_id.GetNext(), | 231 g_next_gpu_memory_buffer_id.GetNext(), |
| 137 request->size, | 232 request->size, |
| 138 request->format, | 233 request->format, |
| 139 request->usage, | 234 request->usage, |
| 140 request->client_id, | 235 request->client_id, |
| 141 base::Bind(&BrowserGpuMemoryBufferManager::GpuMemoryBufferCreatedOnIO, | 236 base::Bind(&BrowserGpuMemoryBufferManager::GpuMemoryBufferCreatedOnIO, |
| 142 base::Unretained(request))); | 237 base::Unretained(request))); |
| 143 } | 238 } |
| 144 | 239 |
| 145 // static | 240 // static |
| 146 void BrowserGpuMemoryBufferManager::GpuMemoryBufferCreatedOnIO( | 241 void BrowserGpuMemoryBufferManager::GpuMemoryBufferCreatedOnIO( |
| 147 AllocateGpuMemoryBufferRequest* request, | 242 AllocateGpuMemoryBufferRequest* request, |
| 148 scoped_ptr<GpuMemoryBufferImpl> buffer) { | 243 scoped_ptr<GpuMemoryBufferImpl> buffer) { |
| 149 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); | 244 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
| 150 | 245 |
| 151 request->result = buffer.Pass(); | 246 request->result = buffer.Pass(); |
| 152 request->event.Signal(); | 247 request->event.Signal(); |
| 153 } | 248 } |
| 154 | 249 |
| 155 void BrowserGpuMemoryBufferManager::SetDestructionSyncPoint( | |
| 156 gfx::GpuMemoryBuffer* buffer, | |
| 157 uint32 sync_point) { | |
| 158 static_cast<GpuMemoryBufferImpl*>(buffer) | |
| 159 ->set_destruction_sync_point(sync_point); | |
| 160 } | |
| 161 | |
| 162 } // namespace content | 250 } // namespace content |
| OLD | NEW |