// Copyright 2011 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 "cc/output/program_binding.h" #include "base/debug/trace_event.h" #include "cc/output/geometry_binding.h" #include "gpu/command_buffer/client/gles2_interface.h" #include "third_party/khronos/GLES2/gl2.h" using gpu::gles2::GLES2Interface; namespace cc { ProgramBindingBase::ProgramBindingBase() : program_(0), vertex_shader_id_(0), fragment_shader_id_(0), initialized_(false) {} ProgramBindingBase::~ProgramBindingBase() { // If you hit these asserts, you initialized but forgot to call Cleanup(). DCHECK(!program_); DCHECK(!vertex_shader_id_); DCHECK(!fragment_shader_id_); DCHECK(!initialized_); } bool ProgramBindingBase::Init(GLES2Interface* context, const std::string& vertex_shader, const std::string& fragment_shader) { TRACE_EVENT0("cc", "ProgramBindingBase::init"); vertex_shader_id_ = LoadShader(context, GL_VERTEX_SHADER, vertex_shader); if (!vertex_shader_id_) return false; fragment_shader_id_ = LoadShader(context, GL_FRAGMENT_SHADER, fragment_shader); if (!fragment_shader_id_) { context->DeleteShader(vertex_shader_id_); vertex_shader_id_ = 0; return false; } program_ = CreateShaderProgram(context, vertex_shader_id_, fragment_shader_id_); return !!program_; } bool ProgramBindingBase::Link(GLES2Interface* context) { context->LinkProgram(program_); CleanupShaders(context); if (!program_) return false; #ifndef NDEBUG int linked = 0; context->GetProgramiv(program_, GL_LINK_STATUS, &linked); if (!linked) return false; #endif return true; } void ProgramBindingBase::Cleanup(GLES2Interface* context) { initialized_ = false; if (!program_) return; DCHECK(context); context->DeleteProgram(program_); program_ = 0; CleanupShaders(context); } unsigned ProgramBindingBase::LoadShader(GLES2Interface* context, unsigned type, const std::string& shader_source) { unsigned shader = context->CreateShader(type); if (!shader) return 0u; const char* shader_source_str[] = { shader_source.data() }; int shader_length[] = { static_cast(shader_source.length()) }; context->ShaderSource( shader, 1, shader_source_str, shader_length); context->CompileShader(shader); #ifndef NDEBUG int compiled = 0; context->GetShaderiv(shader, GL_COMPILE_STATUS, &compiled); if (!compiled) return 0u; #endif return shader; } unsigned ProgramBindingBase::CreateShaderProgram(GLES2Interface* context, unsigned vertex_shader, unsigned fragment_shader) { unsigned program_object = context->CreateProgram(); if (!program_object) return 0; context->AttachShader(program_object, vertex_shader); context->AttachShader(program_object, fragment_shader); // Bind the common attrib locations. context->BindAttribLocation( program_object, GeometryBinding::PositionAttribLocation(), "a_position"); context->BindAttribLocation( program_object, GeometryBinding::TexCoordAttribLocation(), "a_texCoord"); context->BindAttribLocation(program_object, GeometryBinding::TriangleIndexAttribLocation(), "a_index"); return program_object; } void ProgramBindingBase::CleanupShaders(GLES2Interface* context) { if (vertex_shader_id_) { context->DeleteShader(vertex_shader_id_); vertex_shader_id_ = 0; } if (fragment_shader_id_) { context->DeleteShader(fragment_shader_id_); fragment_shader_id_ = 0; } } } // namespace cc