// Copyright (c) 2012 The Chromium Authors. All rights reserved. // 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/buffer_manager.h" #include "base/debug/trace_event.h" #include "base/logging.h" #include "gpu/command_buffer/common/gles2_cmd_utils.h" #include "gpu/command_buffer/service/gles2_cmd_decoder.h" namespace gpu { namespace gles2 { BufferManager::BufferManager() : allow_buffers_on_multiple_targets_(false), mem_represented_(0), last_reported_mem_represented_(1), buffer_info_count_(0), have_context_(true) { UpdateMemRepresented(); } BufferManager::~BufferManager() { DCHECK(buffer_infos_.empty()); CHECK_EQ(buffer_info_count_, 0u); } void BufferManager::Destroy(bool have_context) { have_context_ = have_context; buffer_infos_.clear(); DCHECK_EQ(0u, mem_represented_); UpdateMemRepresented(); } void BufferManager::UpdateMemRepresented() { if (mem_represented_ != last_reported_mem_represented_) { last_reported_mem_represented_ = mem_represented_; TRACE_COUNTER_ID1( "BufferManager", "BufferMemory", this, mem_represented_); } } void BufferManager::CreateBufferInfo(GLuint client_id, GLuint service_id) { BufferInfo::Ref buffer(new BufferInfo(this, service_id)); std::pair result = buffer_infos_.insert(std::make_pair(client_id, buffer)); DCHECK(result.second); } BufferManager::BufferInfo* BufferManager::GetBufferInfo( GLuint client_id) { BufferInfoMap::iterator it = buffer_infos_.find(client_id); return it != buffer_infos_.end() ? it->second : NULL; } void BufferManager::RemoveBufferInfo(GLuint client_id) { BufferInfoMap::iterator it = buffer_infos_.find(client_id); if (it != buffer_infos_.end()) { BufferInfo* buffer = it->second; buffer->MarkAsDeleted(); buffer_infos_.erase(it); } } void BufferManager::StartTracking(BufferManager::BufferInfo* /* buffer */) { ++buffer_info_count_; } void BufferManager::StopTracking(BufferManager::BufferInfo* buffer) { mem_represented_ -= buffer->size(); --buffer_info_count_; UpdateMemRepresented(); } BufferManager::BufferInfo::BufferInfo(BufferManager* manager, GLuint service_id) : manager_(manager), deleted_(false), service_id_(service_id), target_(0), size_(0), usage_(GL_STATIC_DRAW), shadowed_(false) { manager_->StartTracking(this); } BufferManager::BufferInfo::~BufferInfo() { if (manager_) { if (manager_->have_context_) { GLuint id = service_id(); glDeleteBuffersARB(1, &id); } manager_->StopTracking(this); manager_ = NULL; } } void BufferManager::BufferInfo::SetInfo( GLsizeiptr size, GLenum usage, bool shadow) { usage_ = usage; if (size != size_ || shadow != shadowed_) { shadowed_ = shadow; size_ = size; ClearCache(); if (shadowed_) { shadow_.reset(new int8[size]); memset(shadow_.get(), 0, size); } } } bool BufferManager::BufferInfo::SetRange( GLintptr offset, GLsizeiptr size, const GLvoid * data) { if (offset < 0 || offset + size < offset || offset + size > size_) { return false; } if (shadowed_) { memcpy(shadow_.get() + offset, data, size); ClearCache(); } return true; } const void* BufferManager::BufferInfo::GetRange( GLintptr offset, GLsizeiptr size) const { if (!shadowed_) { return NULL; } if (offset < 0 || offset + size < offset || offset + size > size_) { return NULL; } return shadow_.get() + offset; } void BufferManager::BufferInfo::ClearCache() { range_set_.clear(); } template GLuint GetMaxValue(const void* data, GLuint offset, GLsizei count) { GLuint max_value = 0; const T* element = reinterpret_cast( static_cast(data) + offset); const T* end = element + count; for (; element < end; ++element) { if (*element > max_value) { max_value = *element; } } return max_value; } bool BufferManager::BufferInfo::GetMaxValueForRange( GLuint offset, GLsizei count, GLenum type, GLuint* max_value) { Range range(offset, count, type); RangeToMaxValueMap::iterator it = range_set_.find(range); if (it != range_set_.end()) { *max_value = it->second; return true; } uint32 size; if (!SafeMultiplyUint32( count, GLES2Util::GetGLTypeSizeForTexturesAndBuffers(type), &size)) { return false; } if (!SafeAddUint32(offset, size, &size)) { return false; } if (size > static_cast(size_)) { return false; } if (!shadowed_) { return false; } // Scan the range for the max value and store GLuint max_v = 0; switch (type) { case GL_UNSIGNED_BYTE: max_v = GetMaxValue(shadow_.get(), offset, count); break; case GL_UNSIGNED_SHORT: // Check we are not accessing an odd byte for a 2 byte value. if ((offset & 1) != 0) { return false; } max_v = GetMaxValue(shadow_.get(), offset, count); break; case GL_UNSIGNED_INT: // Check we are not accessing a non aligned address for a 4 byte value. if ((offset & 3) != 0) { return false; } max_v = GetMaxValue(shadow_.get(), offset, count); break; default: NOTREACHED(); // should never get here by validation. break; } range_set_.insert(std::make_pair(range, max_v)); *max_value = max_v; return true; } bool BufferManager::GetClientId(GLuint service_id, GLuint* client_id) const { // This doesn't need to be fast. It's only used during slow queries. for (BufferInfoMap::const_iterator it = buffer_infos_.begin(); it != buffer_infos_.end(); ++it) { if (it->second->service_id() == service_id) { *client_id = it->first; return true; } } return false; } void BufferManager::SetInfo( BufferManager::BufferInfo* info, GLsizeiptr size, GLenum usage) { DCHECK(info); mem_represented_ -= info->size(); info->SetInfo(size, usage, info->target() == GL_ELEMENT_ARRAY_BUFFER || allow_buffers_on_multiple_targets_); mem_represented_ += info->size(); } bool BufferManager::SetTarget(BufferManager::BufferInfo* info, GLenum target) { // Check that we are not trying to bind it to a different target. if (info->target() != 0 && info->target() != target && !allow_buffers_on_multiple_targets_) { return false; } if (info->target() == 0) { info->set_target(target); } return true; } } // namespace gles2 } // namespace gpu