axmol/cocos/2d/renderer/Renderer.cpp

243 lines
6.2 KiB
C++
Raw Normal View History

//
// Created by NiTe Luo on 10/31/13.
//
#include <OpenGL/OpenGL.h>
#include "Renderer.h"
#include "CCShaderCache.h"
#include "ccGLStateCache.h"
2013-11-09 04:06:39 +08:00
#include "CustomCommand.h"
2013-11-12 03:54:08 +08:00
#include "QuadCommand.h"
2013-11-09 05:57:21 +08:00
#include "CCGL.h"
2013-11-09 04:06:39 +08:00
NS_CC_BEGIN
using namespace std;
2013-11-08 08:50:53 +08:00
static Renderer*s_instance;
Renderer *Renderer::getInstance()
{
2013-11-08 08:50:53 +08:00
if(!s_instance)
{
2013-11-08 08:50:53 +08:00
s_instance = new Renderer();
if(!s_instance->init())
{
CC_SAFE_DELETE(s_instance);
}
}
2013-11-08 08:50:53 +08:00
return s_instance;
}
void Renderer::destroyInstance()
{
CC_SAFE_RELEASE_NULL(s_instance);
}
Renderer::Renderer()
:_lastMaterialID(0)
2013-11-08 07:48:37 +08:00
,_numQuads(0)
2013-11-11 16:14:29 +08:00
,_firstCommand(0)
,_lastCommand(0)
2013-11-08 07:48:37 +08:00
{
2013-11-08 08:50:53 +08:00
2013-11-08 07:48:37 +08:00
}
Renderer::~Renderer()
{
2013-11-08 08:50:53 +08:00
}
bool Renderer::init()
{
setupIndices();
setupVBOAndVAO();
return true;
}
void Renderer::setupIndices()
{
for( int i=0; i < VBO_SIZE; i++)
{
_indices[i*6+0] = (GLushort) (i*4+0);
_indices[i*6+1] = (GLushort) (i*4+1);
_indices[i*6+2] = (GLushort) (i*4+2);
_indices[i*6+3] = (GLushort) (i*4+3);
_indices[i*6+4] = (GLushort) (i*4+2);
_indices[i*6+5] = (GLushort) (i*4+1);
}
}
void Renderer::setupVBOAndVAO()
{
glGenVertexArrays(1, &_VAOname);
GL::bindVAO(_VAOname);
glGenBuffers(2, &_buffersVBO[0]);
glBindBuffer(GL_ARRAY_BUFFER, _buffersVBO[0]);
glBufferData(GL_ARRAY_BUFFER, sizeof(_quads[0]) * VBO_SIZE, _quads, GL_DYNAMIC_DRAW);
// vertices
glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_POSITION);
glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_POSITION, 3, GL_FLOAT, GL_FALSE, sizeof(V3F_C4B_T2F), (GLvoid*) offsetof( V3F_C4B_T2F, vertices));
// colors
glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_COLOR);
glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_COLOR, 4, GL_UNSIGNED_BYTE, GL_TRUE, sizeof(V3F_C4B_T2F), (GLvoid*) offsetof( V3F_C4B_T2F, colors));
// tex coords
glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_TEX_COORDS);
glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_TEX_COORDS, 2, GL_FLOAT, GL_FALSE, sizeof(V3F_C4B_T2F), (GLvoid*) offsetof( V3F_C4B_T2F, texCoords));
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, _buffersVBO[1]);
glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(_indices[0]) * VBO_SIZE * 6, _indices, GL_STATIC_DRAW);
// Must unbind the VAO before changing the element buffer.
GL::bindVAO(0);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
glBindBuffer(GL_ARRAY_BUFFER, 0);
CHECK_GL_ERROR_DEBUG();
}
void Renderer::addRenderCommand(RenderCommand *command)
{
2013-11-06 09:00:34 +08:00
command->generateID();
_renderQueue.push_back(command);
}
2013-11-06 09:00:34 +08:00
bool compareRenderCommand(RenderCommand* a, RenderCommand* b)
{
return a->getID() < b->getID();
}
void Renderer::render()
{
2013-11-07 06:57:42 +08:00
//Uncomment this once everything is rendered by new renderer
//glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
//TODO setup camera or MVP
//Process render commands
//1. Sort render commands based on ID
2013-11-06 09:00:34 +08:00
stable_sort(_renderQueue.begin(), _renderQueue.end(), compareRenderCommand);
2013-11-11 16:14:29 +08:00
size_t len = _renderQueue.size();
for (size_t i = 0; i < len; i++)
{
2013-11-11 16:14:29 +08:00
auto command = _renderQueue[i];
2013-11-12 03:54:08 +08:00
switch (command->getType())
2013-11-07 06:57:42 +08:00
{
2013-11-12 03:54:08 +08:00
case QUAD_COMMAND:
2013-11-11 16:14:29 +08:00
{
2013-11-12 03:54:08 +08:00
QuadCommand* cmd = static_cast<QuadCommand*>(command);
CCASSERT(cmd->getQuadCount()<VBO_SIZE, "VBO is not big enough for quad data, please break the quad data down or use customized render command");
2013-11-12 03:54:08 +08:00
//Batch quads
if(_numQuads + cmd->getQuadCount() < VBO_SIZE)
{
memcpy(_quads + _numQuads, cmd->getQuad(), sizeof(V3F_C4B_T2F_Quad) * cmd->getQuadCount());
_numQuads += cmd->getQuadCount();
_lastCommand = i;
}
else
{
//Draw batched quads if VBO is full
drawBatchedQuads();
}
break;
2013-11-11 16:14:29 +08:00
}
2013-11-12 03:54:08 +08:00
case CUSTOM_COMMAND:
2013-11-07 06:57:42 +08:00
{
2013-11-12 03:54:08 +08:00
flush();
CustomCommand* cmd = static_cast<CustomCommand*>(command);
cmd->execute();
2013-11-11 16:14:29 +08:00
}
2013-11-12 03:54:08 +08:00
default:
flush();
break;
2013-11-11 16:14:29 +08:00
}
}
2013-11-11 16:14:29 +08:00
//Draw the batched quads
drawBatchedQuads();
2013-11-12 03:54:08 +08:00
//TODO give command back to command pool
for_each(_renderQueue.begin(), _renderQueue.end(), [](RenderCommand* cmd){delete cmd;});
2013-11-11 16:14:29 +08:00
_firstCommand = _lastCommand = 0;
_lastMaterialID = 0;
_renderQueue.clear();
}
2013-11-09 04:06:39 +08:00
2013-11-11 16:14:29 +08:00
void Renderer::drawBatchedQuads()
{
int quadsToDraw = 0;
int startQuad = 0;
2013-11-11 16:14:29 +08:00
//Upload buffer to VBO
if(_numQuads <= 0)
return;
//Set VBO data
glBindBuffer(GL_ARRAY_BUFFER, _buffersVBO[0]);
glBufferData(GL_ARRAY_BUFFER, sizeof(_quads[0]) * (_numQuads), NULL, GL_DYNAMIC_DRAW);
void *buf = glMapBuffer(GL_ARRAY_BUFFER, GL_WRITE_ONLY);
memcpy(buf, _quads, sizeof(_quads[0])* (_numQuads));
glUnmapBuffer(GL_ARRAY_BUFFER);
glBindBuffer(GL_ARRAY_BUFFER, 0);
//Bind VAO
GL::bindVAO(_VAOname);
2013-11-12 03:54:08 +08:00
//Start drawing verties in batch
2013-11-11 16:14:29 +08:00
for(size_t i = _firstCommand; i <= _lastCommand; i++)
{
RenderCommand* command = _renderQueue[i];
if (command->getType() == QUAD_COMMAND)
{
QuadCommand* cmd = static_cast<QuadCommand*>(command);
if(_lastMaterialID != cmd->getMaterialID())
2013-11-09 04:06:39 +08:00
{
2013-11-11 16:14:29 +08:00
//Draw quads
if(quadsToDraw > 0)
{
glDrawElements(GL_TRIANGLES, (GLsizei) quadsToDraw*6, GL_UNSIGNED_SHORT, (GLvoid*) (startQuad*6*sizeof(_indices[0])) );
startQuad += quadsToDraw;
quadsToDraw = 0;
}
2013-11-09 04:06:39 +08:00
2013-11-11 16:14:29 +08:00
//Use new material
cmd->useMaterial();
_lastMaterialID = cmd->getMaterialID();
2013-11-09 04:06:39 +08:00
}
2013-11-11 16:14:29 +08:00
quadsToDraw += cmd->getQuadCount();
}
}
2013-11-12 03:54:08 +08:00
//Draw any remaining quad
if(quadsToDraw > 0)
{
glDrawElements(GL_TRIANGLES, (GLsizei) quadsToDraw*6, GL_UNSIGNED_SHORT, (GLvoid*) (startQuad*6*sizeof(_indices[0])) );
}
2013-11-08 07:48:37 +08:00
2013-11-12 03:54:08 +08:00
_firstCommand = _lastCommand;
2013-11-08 07:48:37 +08:00
_numQuads = 0;
}
2013-11-09 04:06:39 +08:00
void Renderer::flush()
{
2013-11-12 03:54:08 +08:00
drawBatchedQuads();
2013-11-09 04:06:39 +08:00
_lastMaterialID = 0;
}
NS_CC_END