// 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/geometry_binding.h" #include "cc/output/gl_renderer.h" // For the GLC() macro. #include "gpu/command_buffer/client/gles2_interface.h" #include "third_party/khronos/GLES2/gl2.h" #include "ui/gfx/rect_f.h" namespace cc { GeometryBinding::GeometryBinding(gpu::gles2::GLES2Interface* gl, const gfx::RectF& quad_vertex_rect) : gl_(gl), quad_vertices_vbo_(0), quad_elements_vbo_(0) { struct Vertex { float a_position[3]; float a_texCoord[2]; // Index of the vertex, divide by 4 to have the matrix for this quad. float a_index; }; struct Quad { Vertex v0, v1, v2, v3; }; struct QuadIndex { uint16 data[6]; }; COMPILE_ASSERT(sizeof(Quad) == 24 * sizeof(float), // NOLINT(runtime/sizeof) struct_is_densely_packed); COMPILE_ASSERT( sizeof(QuadIndex) == 6 * sizeof(uint16_t), // NOLINT(runtime/sizeof) struct_is_densely_packed); Quad quad_list[8]; QuadIndex quad_index_list[8]; for (int i = 0; i < 8; i++) { Vertex v0 = {{quad_vertex_rect.x(), quad_vertex_rect.bottom(), 0.0f, }, {0.0f, 1.0f, }, i * 4.0f + 0.0f}; Vertex v1 = {{quad_vertex_rect.x(), quad_vertex_rect.y(), 0.0f, }, {0.0f, 0.0f, }, i * 4.0f + 1.0f}; Vertex v2 = {{quad_vertex_rect.right(), quad_vertex_rect.y(), 0.0f, }, {1.0f, .0f, }, i * 4.0f + 2.0f}; Vertex v3 = {{quad_vertex_rect.right(), quad_vertex_rect.bottom(), 0.0f, }, {1.0f, 1.0f, }, i * 4.0f + 3.0f}; Quad x = {v0, v1, v2, v3}; quad_list[i] = x; QuadIndex y = { {static_cast<uint16>(0 + 4 * i), static_cast<uint16>(1 + 4 * i), static_cast<uint16>(2 + 4 * i), static_cast<uint16>(3 + 4 * i), static_cast<uint16>(0 + 4 * i), static_cast<uint16>(2 + 4 * i)}}; quad_index_list[i] = y; } gl_->GenBuffers(1, &quad_vertices_vbo_); gl_->GenBuffers(1, &quad_elements_vbo_); GLC(gl_, gl_->BindBuffer(GL_ARRAY_BUFFER, quad_vertices_vbo_)); GLC(gl_, gl_->BufferData( GL_ARRAY_BUFFER, sizeof(quad_list), quad_list, GL_STATIC_DRAW)); GLC(gl_, gl_->BindBuffer(GL_ELEMENT_ARRAY_BUFFER, quad_elements_vbo_)); GLC(gl_, gl_->BufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(quad_index_list), quad_index_list, GL_STATIC_DRAW)); } GeometryBinding::~GeometryBinding() { gl_->DeleteBuffers(1, &quad_vertices_vbo_); gl_->DeleteBuffers(1, &quad_elements_vbo_); } void GeometryBinding::PrepareForDraw() { GLC(gl_, gl_->BindBuffer(GL_ELEMENT_ARRAY_BUFFER, quad_elements_vbo_)); GLC(gl_, gl_->BindBuffer(GL_ARRAY_BUFFER, quad_vertices_vbo_)); // OpenGL defines the last parameter to VertexAttribPointer as type // "const GLvoid*" even though it is actually an offset into the buffer // object's data store and not a pointer to the client's address space. const void* offsets[3] = { 0, reinterpret_cast<const void*>( 3 * sizeof(float)), // NOLINT(runtime/sizeof) reinterpret_cast<const void*>(5 * sizeof(float)), // NOLINT(runtime/sizeof) }; GLC(gl_, gl_->VertexAttribPointer(PositionAttribLocation(), 3, GL_FLOAT, false, 6 * sizeof(float), // NOLINT(runtime/sizeof) offsets[0])); GLC(gl_, gl_->VertexAttribPointer(TexCoordAttribLocation(), 2, GL_FLOAT, false, 6 * sizeof(float), // NOLINT(runtime/sizeof) offsets[1])); GLC(gl_, gl_->VertexAttribPointer(TriangleIndexAttribLocation(), 1, GL_FLOAT, false, 6 * sizeof(float), // NOLINT(runtime/sizeof) offsets[2])); GLC(gl_, gl_->EnableVertexAttribArray(PositionAttribLocation())); GLC(gl_, gl_->EnableVertexAttribArray(TexCoordAttribLocation())); GLC(gl_, gl_->EnableVertexAttribArray(TriangleIndexAttribLocation())); } } // namespace cc