// 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/vertex_attrib_manager.h" #include #include "base/command_line.h" #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "base/string_number_conversions.h" #include "build/build_config.h" #define GLES2_GPU_SERVICE 1 #include "gpu/command_buffer/common/gles2_cmd_format.h" #include "gpu/command_buffer/common/gles2_cmd_utils.h" #include "gpu/command_buffer/service/buffer_manager.h" #include "gpu/command_buffer/service/feature_info.h" #include "gpu/command_buffer/service/gl_utils.h" #include "gpu/command_buffer/service/gles2_cmd_decoder.h" #include "gpu/command_buffer/service/gpu_switches.h" #include "gpu/command_buffer/service/program_manager.h" #include "gpu/command_buffer/service/vertex_array_manager.h" namespace gpu { namespace gles2 { VertexAttrib::VertexAttrib() : index_(0), enabled_(false), size_(4), type_(GL_FLOAT), offset_(0), normalized_(GL_FALSE), gl_stride_(0), real_stride_(16), divisor_(0), is_client_side_array_(false), list_(NULL) { } VertexAttrib::~VertexAttrib() { } void VertexAttrib::SetInfo( Buffer* buffer, GLint size, GLenum type, GLboolean normalized, GLsizei gl_stride, GLsizei real_stride, GLsizei offset) { DCHECK_GT(real_stride, 0); buffer_ = buffer; size_ = size; type_ = type; normalized_ = normalized; gl_stride_ = gl_stride; real_stride_ = real_stride; offset_ = offset; } void VertexAttrib::Unbind(Buffer* buffer) { if (buffer_ == buffer) { buffer_ = NULL; } } bool VertexAttrib::CanAccess(GLuint index) const { if (!enabled_) { return true; } if (!buffer_ || buffer_->IsDeleted()) { return false; } // The number of elements that can be accessed. GLsizeiptr buffer_size = buffer_->size(); if (offset_ > buffer_size || real_stride_ == 0) { return false; } uint32 usable_size = buffer_size - offset_; GLuint num_elements = usable_size / real_stride_ + ((usable_size % real_stride_) >= (GLES2Util::GetGLTypeSizeForTexturesAndBuffers(type_) * size_) ? 1 : 0); return index < num_elements; } VertexAttribManager::VertexAttribManager() : num_fixed_attribs_(0), element_array_buffer_(NULL), manager_(NULL), deleted_(false), service_id_(0) { } VertexAttribManager::VertexAttribManager( VertexArrayManager* manager, GLuint service_id, uint32 num_vertex_attribs) : num_fixed_attribs_(0), element_array_buffer_(NULL), manager_(manager), deleted_(false), service_id_(service_id) { manager_->StartTracking(this); Initialize(num_vertex_attribs, false); } VertexAttribManager::~VertexAttribManager() { if (manager_) { if (manager_->have_context_) { if (service_id_ != 0) // 0 indicates an emulated VAO glDeleteVertexArraysOES(1, &service_id_); } manager_->StopTracking(this); manager_ = NULL; } } void VertexAttribManager::Initialize( uint32 max_vertex_attribs, bool init_attribs) { vertex_attrib_infos_.resize(max_vertex_attribs); bool disable_workarounds = CommandLine::ForCurrentProcess()->HasSwitch( switches::kDisableGpuDriverBugWorkarounds); for (uint32 vv = 0; vv < vertex_attrib_infos_.size(); ++vv) { vertex_attrib_infos_[vv].set_index(vv); vertex_attrib_infos_[vv].SetList(&disabled_vertex_attribs_); if (!disable_workarounds && init_attribs) { glVertexAttrib4f(vv, 0.0f, 0.0f, 0.0f, 1.0f); } } } void VertexAttribManager::SetElementArrayBuffer(Buffer* buffer) { element_array_buffer_ = buffer; } bool VertexAttribManager::Enable(GLuint index, bool enable) { if (index >= vertex_attrib_infos_.size()) { return false; } VertexAttrib& info = vertex_attrib_infos_[index]; if (info.enabled() != enable) { info.set_enabled(enable); info.SetList(enable ? &enabled_vertex_attribs_ : &disabled_vertex_attribs_); } return true; } void VertexAttribManager::Unbind(Buffer* buffer) { if (element_array_buffer_ == buffer) { element_array_buffer_ = NULL; } for (uint32 vv = 0; vv < vertex_attrib_infos_.size(); ++vv) { vertex_attrib_infos_[vv].Unbind(buffer); } } bool VertexAttribManager::ValidateBindings( const char* function_name, GLES2Decoder* decoder, FeatureInfo* feature_info, Program* current_program, GLuint max_vertex_accessed, GLsizei primcount) { // true if any enabled, used divisor is zero bool divisor0 = false; const GLuint kInitialBufferId = 0xFFFFFFFFU; GLuint current_buffer_id = kInitialBufferId; bool use_client_side_arrays_for_stream_buffers = feature_info->workarounds( ).use_client_side_arrays_for_stream_buffers; // Validate all attribs currently enabled. If they are used by the current // program then check that they have enough elements to handle the draw call. // If they are not used by the current program check that they have a buffer // assigned. for (VertexAttribInfoList::iterator it = enabled_vertex_attribs_.begin(); it != enabled_vertex_attribs_.end(); ++it) { VertexAttrib* attrib = *it; const Program::VertexAttrib* attrib_info = current_program->GetAttribInfoByLocation(attrib->index()); if (attrib_info) { divisor0 |= (attrib->divisor() == 0); GLuint count = attrib->MaxVertexAccessed(primcount, max_vertex_accessed); // This attrib is used in the current program. if (!attrib->CanAccess(count)) { decoder->SetGLError( GL_INVALID_OPERATION, function_name, (std::string( "attempt to access out of range vertices in attribute ") + base::IntToString(attrib->index())).c_str()); return false; } if (use_client_side_arrays_for_stream_buffers) { Buffer* buffer = attrib->buffer(); glEnableVertexAttribArray(attrib->index()); if (buffer->IsClientSideArray()) { if (current_buffer_id != 0) { current_buffer_id = 0; glBindBuffer(GL_ARRAY_BUFFER, 0); } attrib->set_is_client_side_array(true); const void* ptr = buffer->GetRange(attrib->offset(), 0); DCHECK(ptr); glVertexAttribPointer( attrib->index(), attrib->size(), attrib->type(), attrib->normalized(), attrib->gl_stride(), ptr); } else if (attrib->is_client_side_array()) { attrib->set_is_client_side_array(false); GLuint new_buffer_id = buffer->service_id(); if (new_buffer_id != current_buffer_id) { current_buffer_id = new_buffer_id; glBindBuffer(GL_ARRAY_BUFFER, current_buffer_id); } const void* ptr = reinterpret_cast(attrib->offset()); glVertexAttribPointer( attrib->index(), attrib->size(), attrib->type(), attrib->normalized(), attrib->gl_stride(), ptr); } } } else { // This attrib is not used in the current program. if (!attrib->buffer()) { decoder->SetGLError( GL_INVALID_OPERATION, function_name, (std::string( "attempt to render with no buffer attached to " "enabled attribute ") + base::IntToString(attrib->index())).c_str()); return false; } else if (use_client_side_arrays_for_stream_buffers) { Buffer* buffer = attrib->buffer(); // Disable client side arrays for unused attributes else we'll // read bad memory if (buffer->IsClientSideArray()) { // Don't disable attrib 0 since it's special. if (attrib->index() > 0) { glDisableVertexAttribArray(attrib->index()); } } } } } if (primcount && !divisor0) { decoder->SetGLError( GL_INVALID_OPERATION, function_name, "attempt instanced render with all attributes having " "non-zero divisors"); return false; } if (current_buffer_id != kInitialBufferId) { // Restore the buffer binding. decoder->RestoreBufferBindings(); } return true; } } // namespace gles2 } // namespace gpu