// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "gpu/command_buffer/service/texture_owner.h"
#include <memory>
#include "base/atomic_sequence_num.h"
#include "base/feature_list.h"
#include "base/functional/bind.h"
#include "base/task/single_thread_task_runner.h"
#include "base/trace_event/memory_dump_manager.h"
#include "gpu/command_buffer/service/abstract_texture_android.h"
#include "gpu/command_buffer/service/decoder_context.h"
#include "gpu/command_buffer/service/feature_info.h"
#include "gpu/command_buffer/service/image_reader_gl_owner.h"
#include "gpu/command_buffer/service/surface_texture_gl_owner.h"
#include "gpu/command_buffer/service/texture_base.h"
#include "ui/gl/scoped_binders.h"
#include "ui/gl/scoped_make_current.h"
namespace gpu {
namespace {
// Generates process-unique IDs to use for tracing resources.
base::AtomicSequenceNumber g_next_texture_owner_tracing_id;
std::unique_ptr<AbstractTextureAndroid> CreateTexture(
SharedContextState* context_state) {
DCHECK(context_state);
gles2::FeatureInfo* feature_info = context_state->feature_info();
if (feature_info && feature_info->is_passthrough_cmd_decoder()) {
return AbstractTextureAndroid::CreateForPassthrough(gfx::Size());
}
return AbstractTextureAndroid::CreateForValidating(gfx::Size());
}
} // namespace
TextureOwner::TextureOwner(bool binds_texture_on_update,
std::unique_ptr<AbstractTextureAndroid> texture,
scoped_refptr<SharedContextState> context_state)
: base::RefCountedDeleteOnSequence<TextureOwner>(
base::SingleThreadTaskRunner::GetCurrentDefault()),
binds_texture_on_update_(binds_texture_on_update),
context_state_(std::move(context_state)),
texture_(std::move(texture)),
task_runner_(base::SingleThreadTaskRunner::GetCurrentDefault()),
tracing_id_(g_next_texture_owner_tracing_id.GetNext()) {
DCHECK(context_state_);
context_state_->AddContextLostObserver(this);
base::trace_event::MemoryDumpManager::GetInstance()->RegisterDumpProvider(
this, "TextureOwner", base::SingleThreadTaskRunner::GetCurrentDefault());
}
TextureOwner::TextureOwner(bool binds_texture_on_update,
std::unique_ptr<AbstractTextureAndroid> texture)
: base::RefCountedDeleteOnSequence<TextureOwner>(
base::SingleThreadTaskRunner::GetCurrentDefault()),
binds_texture_on_update_(binds_texture_on_update),
texture_(std::move(texture)),
task_runner_(base::SingleThreadTaskRunner::GetCurrentDefault()),
tracing_id_(g_next_texture_owner_tracing_id.GetNext()) {
base::trace_event::MemoryDumpManager::GetInstance()->RegisterDumpProvider(
this, "TextureOwner", base::SingleThreadTaskRunner::GetCurrentDefault());
}
TextureOwner::~TextureOwner() {
base::trace_event::MemoryDumpManager::GetInstance()->UnregisterDumpProvider(
this);
bool have_context = true;
std::optional<ui::ScopedMakeCurrent> scoped_make_current;
if (!context_state_) {
have_context = false;
} else {
if (!context_state_->IsCurrent(nullptr, /*needs_gl=*/true)) {
scoped_make_current.emplace(context_state_->context(),
context_state_->surface());
have_context = scoped_make_current->IsContextCurrent();
}
context_state_->RemoveContextLostObserver(this);
}
if (!have_context)
texture_->NotifyOnContextLost();
// Reset texture and context state here while the |context_state_| is current.
texture_.reset();
context_state_.reset();
}
// static
scoped_refptr<TextureOwner> TextureOwner::Create(
Mode mode,
scoped_refptr<SharedContextState> context_state,
scoped_refptr<RefCountedLock> drdc_lock,
TextureOwnerCodecType type_for_metrics) {
auto texture = CreateTexture(context_state.get());
switch (mode) {
case Mode::kAImageReaderInsecure:
case Mode::kAImageReaderInsecureSurfaceControl:
case Mode::kAImageReaderSecureSurfaceControl:
return new ImageReaderGLOwner(std::move(texture), mode,
std::move(context_state),
std::move(drdc_lock), type_for_metrics);
case Mode::kSurfaceTextureInsecure:
DCHECK(!drdc_lock);
return new SurfaceTextureGLOwner(std::move(texture),
std::move(context_state));
}
NOTREACHED_IN_MIGRATION();
return nullptr;
}
GLuint TextureOwner::GetTextureId() const {
return texture_->service_id();
}
TextureBase* TextureOwner::GetTextureBase() const {
return texture_->GetTextureBase();
}
void TextureOwner::OnContextLost() {
ReleaseResources();
context_state_->RemoveContextLostObserver(this);
context_state_.reset();
}
} // namespace gpu