// 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