1 // Copyright 2013 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 "gpu/command_buffer/service/async_pixel_transfer_manager_share_group.h"
10 #include "base/debug/trace_event.h"
11 #include "base/debug/trace_event_synthetic_delay.h"
12 #include "base/lazy_instance.h"
13 #include "base/logging.h"
14 #include "base/memory/ref_counted.h"
15 #include "base/memory/weak_ptr.h"
16 #include "base/synchronization/cancellation_flag.h"
17 #include "base/synchronization/lock.h"
18 #include "base/synchronization/waitable_event.h"
19 #include "base/threading/thread.h"
20 #include "base/threading/thread_checker.h"
21 #include "gpu/command_buffer/service/async_pixel_transfer_delegate.h"
22 #include "ui/gl/gl_bindings.h"
23 #include "ui/gl/gl_context.h"
24 #include "ui/gl/gl_surface.h"
25 #include "ui/gl/gpu_preference.h"
26 #include "ui/gl/scoped_binders.h"
32 const char kAsyncTransferThreadName
[] = "AsyncTransferThread";
34 void PerformNotifyCompletion(
35 AsyncMemoryParams mem_params
,
36 scoped_refptr
<AsyncPixelTransferCompletionObserver
> observer
) {
37 TRACE_EVENT0("gpu", "PerformNotifyCompletion");
38 observer
->DidComplete(mem_params
);
41 // TODO(backer): Factor out common thread scheduling logic from the EGL and
42 // ShareGroup implementations. http://crbug.com/239889
43 class TransferThread
: public base::Thread
{
46 : base::Thread(kAsyncTransferThreadName
),
49 #if defined(OS_ANDROID) || defined(OS_LINUX)
50 SetPriority(base::kThreadPriority_Background
);
54 virtual ~TransferThread() {
55 // The only instance of this class was declared leaky.
59 void InitializeOnMainThread(gfx::GLContext
* parent_context
) {
60 TRACE_EVENT0("gpu", "TransferThread::InitializeOnMainThread");
64 base::WaitableEvent
wait_for_init(true, false);
65 message_loop_proxy()->PostTask(
67 base::Bind(&TransferThread::InitializeOnTransferThread
,
68 base::Unretained(this),
69 base::Unretained(parent_context
),
74 virtual void CleanUp() OVERRIDE
{
82 scoped_refptr
<gfx::GLSurface
> surface_
;
83 scoped_refptr
<gfx::GLContext
> context_
;
85 void InitializeOnTransferThread(gfx::GLContext
* parent_context
,
86 base::WaitableEvent
* caller_wait
) {
87 TRACE_EVENT0("gpu", "InitializeOnTransferThread");
89 if (!parent_context
) {
90 LOG(ERROR
) << "No parent context provided.";
91 caller_wait
->Signal();
95 surface_
= gfx::GLSurface::CreateOffscreenGLSurface(gfx::Size(1, 1));
96 if (!surface_
.get()) {
97 LOG(ERROR
) << "Unable to create GLSurface";
98 caller_wait
->Signal();
102 // TODO(backer): This is coded for integrated GPUs. For discrete GPUs
103 // we would probably want to use a PBO texture upload for a true async
104 // upload (that would hopefully be optimized as a DMA transfer by the
106 context_
= gfx::GLContext::CreateGLContext(parent_context
->share_group(),
108 gfx::PreferIntegratedGpu
);
109 if (!context_
.get()) {
110 LOG(ERROR
) << "Unable to create GLContext.";
111 caller_wait
->Signal();
115 context_
->MakeCurrent(surface_
.get());
117 caller_wait
->Signal();
120 DISALLOW_COPY_AND_ASSIGN(TransferThread
);
123 base::LazyInstance
<TransferThread
>::Leaky
124 g_transfer_thread
= LAZY_INSTANCE_INITIALIZER
;
126 base::MessageLoopProxy
* transfer_message_loop_proxy() {
127 return g_transfer_thread
.Pointer()->message_loop_proxy().get();
130 class PendingTask
: public base::RefCountedThreadSafe
<PendingTask
> {
132 explicit PendingTask(const base::Closure
& task
)
133 : task_(task
), task_pending_(true, false) {}
136 // This is meant to be called on the main thread where the texture
138 DCHECK(checker_
.CalledOnValidThread());
139 if (task_lock_
.Try()) {
141 if (!task_
.is_null())
145 task_lock_
.Release();
146 task_pending_
.Signal();
152 void BindAndRun(GLuint texture_id
) {
153 // This is meant to be called on the upload thread where we don't have to
154 // restore the previous texture binding.
155 DCHECK(!checker_
.CalledOnValidThread());
156 base::AutoLock
locked(task_lock_
);
157 if (!task_
.is_null()) {
158 glBindTexture(GL_TEXTURE_2D
, texture_id
);
161 glBindTexture(GL_TEXTURE_2D
, 0);
162 // Flush for synchronization between threads.
164 task_pending_
.Signal();
169 base::AutoLock
locked(task_lock_
);
171 task_pending_
.Signal();
174 bool TaskIsInProgress() {
175 return !task_pending_
.IsSignaled();
179 task_pending_
.Wait();
183 friend class base::RefCountedThreadSafe
<PendingTask
>;
185 virtual ~PendingTask() {}
187 base::ThreadChecker checker_
;
189 base::Lock task_lock_
;
191 base::WaitableEvent task_pending_
;
193 DISALLOW_COPY_AND_ASSIGN(PendingTask
);
196 // Class which holds async pixel transfers state.
197 // The texture_id is accessed by either thread, but everything
198 // else accessed only on the main thread.
199 class TransferStateInternal
200 : public base::RefCountedThreadSafe
<TransferStateInternal
> {
202 TransferStateInternal(GLuint texture_id
,
203 const AsyncTexImage2DParams
& define_params
)
204 : texture_id_(texture_id
), define_params_(define_params
) {}
206 bool TransferIsInProgress() {
207 return pending_upload_task_
.get() &&
208 pending_upload_task_
->TaskIsInProgress();
211 void BindTransfer() {
212 TRACE_EVENT2("gpu", "BindAsyncTransfer",
213 "width", define_params_
.width
,
214 "height", define_params_
.height
);
217 glBindTexture(GL_TEXTURE_2D
, texture_id_
);
218 bind_callback_
.Run();
221 void WaitForTransferCompletion() {
222 TRACE_EVENT0("gpu", "WaitForTransferCompletion");
223 DCHECK(pending_upload_task_
.get());
224 if (!pending_upload_task_
->TryRun()) {
225 pending_upload_task_
->WaitForTask();
227 pending_upload_task_
= NULL
;
230 void CancelUpload() {
231 TRACE_EVENT0("gpu", "CancelUpload");
232 if (pending_upload_task_
.get())
233 pending_upload_task_
->Cancel();
234 pending_upload_task_
= NULL
;
237 void ScheduleAsyncTexImage2D(
238 const AsyncTexImage2DParams tex_params
,
239 const AsyncMemoryParams mem_params
,
240 scoped_refptr
<AsyncPixelTransferUploadStats
> texture_upload_stats
,
241 const base::Closure
& bind_callback
) {
242 TRACE_EVENT_SYNTHETIC_DELAY_BEGIN("gpu.AsyncTexImage");
243 pending_upload_task_
= new PendingTask(base::Bind(
244 &TransferStateInternal::PerformAsyncTexImage2D
,
248 texture_upload_stats
));
249 transfer_message_loop_proxy()->PostTask(
252 &PendingTask::BindAndRun
, pending_upload_task_
, texture_id_
));
254 // Save the late bind callback, so we can notify the client when it is
256 bind_callback_
= bind_callback
;
259 void ScheduleAsyncTexSubImage2D(
260 AsyncTexSubImage2DParams tex_params
,
261 AsyncMemoryParams mem_params
,
262 scoped_refptr
<AsyncPixelTransferUploadStats
> texture_upload_stats
) {
263 TRACE_EVENT_SYNTHETIC_DELAY_BEGIN("gpu.AsyncTexImage");
264 pending_upload_task_
= new PendingTask(base::Bind(
265 &TransferStateInternal::PerformAsyncTexSubImage2D
,
269 texture_upload_stats
));
270 transfer_message_loop_proxy()->PostTask(
273 &PendingTask::BindAndRun
, pending_upload_task_
, texture_id_
));
277 friend class base::RefCountedThreadSafe
<TransferStateInternal
>;
279 virtual ~TransferStateInternal() {
282 void PerformAsyncTexImage2D(
283 AsyncTexImage2DParams tex_params
,
284 AsyncMemoryParams mem_params
,
285 scoped_refptr
<AsyncPixelTransferUploadStats
> texture_upload_stats
) {
287 "PerformAsyncTexImage",
292 DCHECK_EQ(0, tex_params
.level
);
294 base::TimeTicks begin_time
;
295 if (texture_upload_stats
.get())
296 begin_time
= base::TimeTicks::HighResNow();
298 void* data
= mem_params
.GetDataAddress();
301 TRACE_EVENT0("gpu", "glTexImage2D");
302 glTexImage2D(GL_TEXTURE_2D
,
304 tex_params
.internal_format
,
311 TRACE_EVENT_SYNTHETIC_DELAY_END("gpu.AsyncTexImage");
314 if (texture_upload_stats
.get()) {
315 texture_upload_stats
->AddUpload(base::TimeTicks::HighResNow() -
320 void PerformAsyncTexSubImage2D(
321 AsyncTexSubImage2DParams tex_params
,
322 AsyncMemoryParams mem_params
,
323 scoped_refptr
<AsyncPixelTransferUploadStats
> texture_upload_stats
) {
325 "PerformAsyncTexSubImage2D",
330 DCHECK_EQ(0, tex_params
.level
);
332 base::TimeTicks begin_time
;
333 if (texture_upload_stats
.get())
334 begin_time
= base::TimeTicks::HighResNow();
336 void* data
= mem_params
.GetDataAddress();
338 TRACE_EVENT0("gpu", "glTexSubImage2D");
339 glTexSubImage2D(GL_TEXTURE_2D
,
348 TRACE_EVENT_SYNTHETIC_DELAY_END("gpu.AsyncTexImage");
351 if (texture_upload_stats
.get()) {
352 texture_upload_stats
->AddUpload(base::TimeTicks::HighResNow() -
357 scoped_refptr
<PendingTask
> pending_upload_task_
;
361 // Definition params for texture that needs binding.
362 AsyncTexImage2DParams define_params_
;
364 // Callback to invoke when AsyncTexImage2D is complete
365 // and the client can safely use the texture. This occurs
366 // during BindCompletedAsyncTransfers().
367 base::Closure bind_callback_
;
372 class AsyncPixelTransferDelegateShareGroup
373 : public AsyncPixelTransferDelegate
,
374 public base::SupportsWeakPtr
<AsyncPixelTransferDelegateShareGroup
> {
376 AsyncPixelTransferDelegateShareGroup(
377 AsyncPixelTransferManagerShareGroup::SharedState
* shared_state
,
379 const AsyncTexImage2DParams
& define_params
);
380 virtual ~AsyncPixelTransferDelegateShareGroup();
382 void BindTransfer() { state_
->BindTransfer(); }
384 // Implement AsyncPixelTransferDelegate:
385 virtual void AsyncTexImage2D(
386 const AsyncTexImage2DParams
& tex_params
,
387 const AsyncMemoryParams
& mem_params
,
388 const base::Closure
& bind_callback
) OVERRIDE
;
389 virtual void AsyncTexSubImage2D(
390 const AsyncTexSubImage2DParams
& tex_params
,
391 const AsyncMemoryParams
& mem_params
) OVERRIDE
;
392 virtual bool TransferIsInProgress() OVERRIDE
;
393 virtual void WaitForTransferCompletion() OVERRIDE
;
396 // A raw pointer is safe because the SharedState is owned by the Manager,
397 // which owns this Delegate.
398 AsyncPixelTransferManagerShareGroup::SharedState
* shared_state_
;
399 scoped_refptr
<TransferStateInternal
> state_
;
401 DISALLOW_COPY_AND_ASSIGN(AsyncPixelTransferDelegateShareGroup
);
404 AsyncPixelTransferDelegateShareGroup::AsyncPixelTransferDelegateShareGroup(
405 AsyncPixelTransferManagerShareGroup::SharedState
* shared_state
,
407 const AsyncTexImage2DParams
& define_params
)
408 : shared_state_(shared_state
),
409 state_(new TransferStateInternal(texture_id
, define_params
)) {}
411 AsyncPixelTransferDelegateShareGroup::~AsyncPixelTransferDelegateShareGroup() {
412 TRACE_EVENT0("gpu", " ~AsyncPixelTransferDelegateShareGroup");
413 state_
->CancelUpload();
416 bool AsyncPixelTransferDelegateShareGroup::TransferIsInProgress() {
417 return state_
->TransferIsInProgress();
420 void AsyncPixelTransferDelegateShareGroup::WaitForTransferCompletion() {
421 if (state_
->TransferIsInProgress()) {
422 state_
->WaitForTransferCompletion();
423 DCHECK(!state_
->TransferIsInProgress());
426 // Fast track the BindTransfer, if applicable.
427 for (AsyncPixelTransferManagerShareGroup::SharedState::TransferQueue::iterator
428 iter
= shared_state_
->pending_allocations
.begin();
429 iter
!= shared_state_
->pending_allocations
.end();
431 if (iter
->get() != this)
434 shared_state_
->pending_allocations
.erase(iter
);
440 void AsyncPixelTransferDelegateShareGroup::AsyncTexImage2D(
441 const AsyncTexImage2DParams
& tex_params
,
442 const AsyncMemoryParams
& mem_params
,
443 const base::Closure
& bind_callback
) {
444 DCHECK(!state_
->TransferIsInProgress());
445 DCHECK_EQ(static_cast<GLenum
>(GL_TEXTURE_2D
), tex_params
.target
);
446 DCHECK_EQ(tex_params
.level
, 0);
448 shared_state_
->pending_allocations
.push_back(AsWeakPtr());
449 state_
->ScheduleAsyncTexImage2D(tex_params
,
451 shared_state_
->texture_upload_stats
,
455 void AsyncPixelTransferDelegateShareGroup::AsyncTexSubImage2D(
456 const AsyncTexSubImage2DParams
& tex_params
,
457 const AsyncMemoryParams
& mem_params
) {
458 TRACE_EVENT2("gpu", "AsyncTexSubImage2D",
459 "width", tex_params
.width
,
460 "height", tex_params
.height
);
461 DCHECK(!state_
->TransferIsInProgress());
462 DCHECK_EQ(static_cast<GLenum
>(GL_TEXTURE_2D
), tex_params
.target
);
463 DCHECK_EQ(tex_params
.level
, 0);
465 state_
->ScheduleAsyncTexSubImage2D(
466 tex_params
, mem_params
, shared_state_
->texture_upload_stats
);
469 AsyncPixelTransferManagerShareGroup::SharedState::SharedState()
470 // TODO(reveman): Skip this if --enable-gpu-benchmarking is not present.
471 : texture_upload_stats(new AsyncPixelTransferUploadStats
) {}
473 AsyncPixelTransferManagerShareGroup::SharedState::~SharedState() {}
475 AsyncPixelTransferManagerShareGroup::AsyncPixelTransferManagerShareGroup(
476 gfx::GLContext
* context
) {
477 g_transfer_thread
.Pointer()->InitializeOnMainThread(context
);
480 AsyncPixelTransferManagerShareGroup::~AsyncPixelTransferManagerShareGroup() {}
482 void AsyncPixelTransferManagerShareGroup::BindCompletedAsyncTransfers() {
483 scoped_ptr
<gfx::ScopedTextureBinder
> texture_binder
;
485 while (!shared_state_
.pending_allocations
.empty()) {
486 if (!shared_state_
.pending_allocations
.front().get()) {
487 shared_state_
.pending_allocations
.pop_front();
490 AsyncPixelTransferDelegateShareGroup
* delegate
=
491 shared_state_
.pending_allocations
.front().get();
492 // Terminate early, as all transfers finish in order, currently.
493 if (delegate
->TransferIsInProgress())
497 texture_binder
.reset(new gfx::ScopedTextureBinder(GL_TEXTURE_2D
, 0));
499 // Used to set tex info from the gles2 cmd decoder once upload has
500 // finished (it'll bind the texture and call a callback).
501 delegate
->BindTransfer();
503 shared_state_
.pending_allocations
.pop_front();
507 void AsyncPixelTransferManagerShareGroup::AsyncNotifyCompletion(
508 const AsyncMemoryParams
& mem_params
,
509 AsyncPixelTransferCompletionObserver
* observer
) {
510 // Post a PerformNotifyCompletion task to the upload thread. This task
511 // will run after all async transfers are complete.
512 transfer_message_loop_proxy()->PostTask(
514 base::Bind(&PerformNotifyCompletion
,
516 make_scoped_refptr(observer
)));
519 uint32
AsyncPixelTransferManagerShareGroup::GetTextureUploadCount() {
520 return shared_state_
.texture_upload_stats
->GetStats(NULL
);
524 AsyncPixelTransferManagerShareGroup::GetTotalTextureUploadTime() {
525 base::TimeDelta total_texture_upload_time
;
526 shared_state_
.texture_upload_stats
->GetStats(&total_texture_upload_time
);
527 return total_texture_upload_time
;
530 void AsyncPixelTransferManagerShareGroup::ProcessMorePendingTransfers() {
533 bool AsyncPixelTransferManagerShareGroup::NeedsProcessMorePendingTransfers() {
537 void AsyncPixelTransferManagerShareGroup::WaitAllAsyncTexImage2D() {
538 if (shared_state_
.pending_allocations
.empty())
541 AsyncPixelTransferDelegateShareGroup
* delegate
=
542 shared_state_
.pending_allocations
.back().get();
544 delegate
->WaitForTransferCompletion();
547 AsyncPixelTransferDelegate
*
548 AsyncPixelTransferManagerShareGroup::CreatePixelTransferDelegateImpl(
549 gles2::TextureRef
* ref
,
550 const AsyncTexImage2DParams
& define_params
) {
551 return new AsyncPixelTransferDelegateShareGroup(
552 &shared_state_
, ref
->service_id(), define_params
);