2019-11-23 20:27:39 +08:00
|
|
|
|
/****************************************************************************
|
|
|
|
|
Copyright (c) 2018-2019 Xiamen Yaji Software Co., Ltd.
|
2022-03-13 12:05:22 +08:00
|
|
|
|
Copyright (c) 2021-2022 Bytedance Inc.
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
2022-10-01 16:24:52 +08:00
|
|
|
|
https://axmolengine.github.io/
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
THE SOFTWARE.
|
|
|
|
|
****************************************************************************/
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
#include "CommandBufferGL.h"
|
|
|
|
|
#include "BufferGL.h"
|
|
|
|
|
#include "RenderPipelineGL.h"
|
|
|
|
|
#include "TextureGL.h"
|
|
|
|
|
#include "DepthStencilStateGL.h"
|
|
|
|
|
#include "ProgramGL.h"
|
2023-06-11 13:08:08 +08:00
|
|
|
|
#include "base/EventDispatcher.h"
|
|
|
|
|
#include "base/EventType.h"
|
|
|
|
|
#include "base/Director.h"
|
2021-07-07 13:52:56 +08:00
|
|
|
|
#include "renderer/backend/opengl/MacrosGL.h"
|
2019-11-23 20:27:39 +08:00
|
|
|
|
#include "renderer/backend/opengl/UtilsGL.h"
|
2020-09-21 22:10:50 +08:00
|
|
|
|
#include "RenderTargetGL.h"
|
2019-11-23 20:27:39 +08:00
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
|
NS_AX_BACKEND_BEGIN
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
|
|
namespace
|
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
|
void applyTexture(TextureBackend* texture, int slot, int index)
|
|
|
|
|
{
|
|
|
|
|
switch (texture->getTextureType())
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
|
case TextureType::TEXTURE_2D:
|
|
|
|
|
static_cast<Texture2DGL*>(texture)->apply(slot, index);
|
|
|
|
|
break;
|
|
|
|
|
case TextureType::TEXTURE_CUBE:
|
|
|
|
|
static_cast<TextureCubeGL*>(texture)->apply(slot, index);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
assert(false);
|
|
|
|
|
return;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
} // namespace
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
CommandBufferGL::CommandBufferGL() {}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
|
|
CommandBufferGL::~CommandBufferGL()
|
|
|
|
|
{
|
|
|
|
|
cleanResources();
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-22 22:01:47 +08:00
|
|
|
|
bool CommandBufferGL::beginFrame()
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
2021-04-22 22:01:47 +08:00
|
|
|
|
return true;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-22 22:01:47 +08:00
|
|
|
|
void CommandBufferGL::beginRenderPass(const RenderTarget* rt, const RenderPassDescriptor& descirptor)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
2020-09-21 22:10:50 +08:00
|
|
|
|
auto rtGL = static_cast<const RenderTargetGL*>(rt);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2020-09-21 22:10:50 +08:00
|
|
|
|
rtGL->bindFrameBuffer();
|
2022-06-24 14:18:48 +08:00
|
|
|
|
rtGL->update();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2020-09-21 22:10:50 +08:00
|
|
|
|
auto clearFlags = descirptor.flags.clear;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
|
|
// set clear color, depth and stencil
|
|
|
|
|
GLbitfield mask = 0;
|
2020-09-21 22:10:50 +08:00
|
|
|
|
if (bitmask::any(clearFlags, TargetBufferFlags::COLOR))
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
mask |= GL_COLOR_BUFFER_BIT;
|
|
|
|
|
const auto& clearColor = descirptor.clearColorValue;
|
|
|
|
|
glClearColor(clearColor[0], clearColor[1], clearColor[2], clearColor[3]);
|
|
|
|
|
}
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
CHECK_GL_ERROR_DEBUG();
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
GLboolean oldDepthWrite = GL_FALSE;
|
|
|
|
|
GLboolean oldDepthTest = GL_FALSE;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
GLfloat oldDepthClearValue = 0.f;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
GLint oldDepthFunc = GL_LESS;
|
2020-09-21 22:10:50 +08:00
|
|
|
|
if (bitmask::any(clearFlags, TargetBufferFlags::DEPTH))
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
glGetBooleanv(GL_DEPTH_WRITEMASK, &oldDepthWrite);
|
|
|
|
|
glGetBooleanv(GL_DEPTH_TEST, &oldDepthTest);
|
|
|
|
|
glGetFloatv(GL_DEPTH_CLEAR_VALUE, &oldDepthClearValue);
|
|
|
|
|
glGetIntegerv(GL_DEPTH_FUNC, &oldDepthFunc);
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
mask |= GL_DEPTH_BUFFER_BIT;
|
|
|
|
|
glClearDepth(descirptor.clearDepthValue);
|
|
|
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
|
glDepthMask(GL_TRUE);
|
|
|
|
|
glDepthFunc(GL_ALWAYS);
|
|
|
|
|
}
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
CHECK_GL_ERROR_DEBUG();
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
|
|
|
|
if (bitmask::any(clearFlags, TargetBufferFlags::STENCIL))
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
mask |= GL_STENCIL_BUFFER_BIT;
|
|
|
|
|
glClearStencil(descirptor.clearStencilValue);
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (mask)
|
|
|
|
|
glClear(mask);
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
CHECK_GL_ERROR_DEBUG();
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
// restore depth test
|
2020-09-21 22:10:50 +08:00
|
|
|
|
if (bitmask::any(clearFlags, TargetBufferFlags::DEPTH))
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
if (!oldDepthTest)
|
|
|
|
|
glDisable(GL_DEPTH_TEST);
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
glDepthMask(oldDepthWrite);
|
|
|
|
|
glDepthFunc(oldDepthFunc);
|
|
|
|
|
glClearDepth(oldDepthClearValue);
|
|
|
|
|
}
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
CHECK_GL_ERROR_DEBUG();
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-22 16:32:17 +08:00
|
|
|
|
void CommandBufferGL::setDepthStencilState(DepthStencilState* depthStencilState)
|
|
|
|
|
{
|
|
|
|
|
_depthStencilStateGL = static_cast<DepthStencilStateGL*>(depthStencilState);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
void CommandBufferGL::setRenderPipeline(RenderPipeline* renderPipeline)
|
|
|
|
|
{
|
2020-09-22 16:32:17 +08:00
|
|
|
|
_renderPipeline = static_cast<RenderPipelineGL*>(renderPipeline);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
|
* Update depthStencil status, improvment: for metal backend cache it
|
|
|
|
|
* @param depthStencilState Specifies the depth and stencil status
|
|
|
|
|
*/
|
2020-09-22 16:32:17 +08:00
|
|
|
|
void CommandBufferGL::updateDepthStencilState(const DepthStencilDescriptor& descriptor)
|
|
|
|
|
{
|
|
|
|
|
_depthStencilStateGL->update(descriptor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Update render pipeline status
|
|
|
|
|
* @param depthStencilState Specifies the depth and stencil status
|
|
|
|
|
*/
|
|
|
|
|
void CommandBufferGL::updatePipelineState(const RenderTarget* rt, const PipelineDescriptor& descriptor)
|
|
|
|
|
{
|
|
|
|
|
_renderPipeline->update(rt, descriptor);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setViewport(int x, int y, unsigned int w, unsigned int h)
|
|
|
|
|
{
|
|
|
|
|
glViewport(x, y, w, h);
|
|
|
|
|
_viewPort.x = x;
|
|
|
|
|
_viewPort.y = y;
|
|
|
|
|
_viewPort.w = w;
|
|
|
|
|
_viewPort.h = h;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setCullMode(CullMode mode)
|
|
|
|
|
{
|
|
|
|
|
_cullMode = mode;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setWinding(Winding winding)
|
|
|
|
|
{
|
|
|
|
|
glFrontFace(UtilsGL::toGLFrontFace(winding));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setIndexBuffer(Buffer* buffer)
|
|
|
|
|
{
|
|
|
|
|
assert(buffer != nullptr);
|
2022-03-13 12:05:22 +08:00
|
|
|
|
if (buffer == nullptr || _indexBuffer == buffer)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
return;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
buffer->retain();
|
2022-07-16 10:43:05 +08:00
|
|
|
|
AX_SAFE_RELEASE(_indexBuffer);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
_indexBuffer = static_cast<BufferGL*>(buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setVertexBuffer(Buffer* buffer)
|
|
|
|
|
{
|
|
|
|
|
assert(buffer != nullptr);
|
|
|
|
|
if (buffer == nullptr || _vertexBuffer == buffer)
|
|
|
|
|
return;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
buffer->retain();
|
2022-07-16 10:43:05 +08:00
|
|
|
|
AX_SAFE_RELEASE(_vertexBuffer);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
_vertexBuffer = static_cast<BufferGL*>(buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setProgramState(ProgramState* programState)
|
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
|
AX_SAFE_RETAIN(programState);
|
|
|
|
|
AX_SAFE_RELEASE(_programState);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
_programState = programState;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-06 16:17:55 +08:00
|
|
|
|
void CommandBufferGL::drawArrays(PrimitiveType primitiveType, std::size_t start, std::size_t count, bool wireframe)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
prepareDrawing();
|
2022-11-10 22:03:40 +08:00
|
|
|
|
#ifndef AX_USE_GLES // glPolygonMode is only supported in Desktop OpenGL
|
2022-08-06 16:17:55 +08:00
|
|
|
|
if (wireframe) glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
|
|
|
|
|
#else
|
|
|
|
|
if (wireframe) primitiveType = PrimitiveType::LINE;
|
|
|
|
|
#endif
|
2019-11-23 20:27:39 +08:00
|
|
|
|
glDrawArrays(UtilsGL::toGLPrimitiveType(primitiveType), start, count);
|
2022-11-10 22:03:40 +08:00
|
|
|
|
#ifndef AX_USE_GLES // glPolygonMode is only supported in Desktop OpenGL
|
2022-08-06 16:17:55 +08:00
|
|
|
|
if (wireframe) glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
|
|
|
|
|
#endif
|
2019-11-23 20:27:39 +08:00
|
|
|
|
cleanResources();
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
void CommandBufferGL::drawElements(PrimitiveType primitiveType,
|
|
|
|
|
IndexFormat indexType,
|
|
|
|
|
std::size_t count,
|
2022-08-06 16:17:55 +08:00
|
|
|
|
std::size_t offset,
|
|
|
|
|
bool wireframe)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
prepareDrawing();
|
2022-11-10 22:03:40 +08:00
|
|
|
|
#ifndef AX_USE_GLES // glPolygonMode is only supported in Desktop OpenGL
|
2022-08-06 16:17:55 +08:00
|
|
|
|
if (wireframe) glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
|
|
|
|
|
#else
|
|
|
|
|
if (wireframe) primitiveType = PrimitiveType::LINE;
|
|
|
|
|
#endif
|
2019-11-23 20:27:39 +08:00
|
|
|
|
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, _indexBuffer->getHandler());
|
2021-12-25 10:04:45 +08:00
|
|
|
|
glDrawElements(UtilsGL::toGLPrimitiveType(primitiveType), count, UtilsGL::toGLIndexType(indexType),
|
|
|
|
|
(GLvoid*)offset);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
CHECK_GL_ERROR_DEBUG();
|
2022-11-10 22:03:40 +08:00
|
|
|
|
#ifndef AX_USE_GLES // glPolygonMode is only supported in Desktop OpenGL
|
2022-08-06 16:17:55 +08:00
|
|
|
|
if (wireframe) glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
|
|
|
|
|
#endif
|
2019-11-23 20:27:39 +08:00
|
|
|
|
cleanResources();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::endRenderPass()
|
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
|
AX_SAFE_RELEASE_NULL(_indexBuffer);
|
|
|
|
|
AX_SAFE_RELEASE_NULL(_vertexBuffer);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
void CommandBufferGL::endFrame() {}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
|
|
void CommandBufferGL::prepareDrawing() const
|
2021-12-25 10:04:45 +08:00
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
|
const auto& program = _renderPipeline->getProgram();
|
|
|
|
|
glUseProgram(program->getHandler());
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
bindVertexBuffer(program);
|
|
|
|
|
setUniforms(program);
|
|
|
|
|
|
|
|
|
|
// Set depth/stencil state.
|
2020-09-22 16:32:17 +08:00
|
|
|
|
if (_depthStencilStateGL->isEnabled())
|
2019-11-23 20:27:39 +08:00
|
|
|
|
_depthStencilStateGL->apply(_stencilReferenceValueFront, _stencilReferenceValueBack);
|
|
|
|
|
else
|
|
|
|
|
DepthStencilStateGL::reset();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
// Set cull mode.
|
|
|
|
|
if (CullMode::NONE == _cullMode)
|
|
|
|
|
{
|
|
|
|
|
glDisable(GL_CULL_FACE);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
glEnable(GL_CULL_FACE);
|
|
|
|
|
glCullFace(UtilsGL::toGLCullMode(_cullMode));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
void CommandBufferGL::bindVertexBuffer(ProgramGL* program) const
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
// Bind vertex buffers and set the attributes.
|
|
|
|
|
auto vertexLayout = _programState->getVertexLayout();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
if (!vertexLayout->isValid())
|
|
|
|
|
return;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, _vertexBuffer->getHandler());
|
|
|
|
|
|
|
|
|
|
const auto& attributes = vertexLayout->getAttributes();
|
|
|
|
|
for (const auto& attributeInfo : attributes)
|
|
|
|
|
{
|
|
|
|
|
const auto& attribute = attributeInfo.second;
|
|
|
|
|
glEnableVertexAttribArray(attribute.index);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
glVertexAttribPointer(attribute.index, UtilsGL::getGLAttributeSize(attribute.format),
|
|
|
|
|
UtilsGL::toGLAttributeType(attribute.format), attribute.needToBeNormallized,
|
|
|
|
|
vertexLayout->getStride(), (GLvoid*)attribute.offset);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setUniforms(ProgramGL* program) const
|
|
|
|
|
{
|
|
|
|
|
if (_programState)
|
|
|
|
|
{
|
2021-09-30 22:33:23 +08:00
|
|
|
|
auto& callbacks = _programState->getCallbackUniforms();
|
2022-07-21 19:19:08 +08:00
|
|
|
|
for (auto&& cb : callbacks)
|
2021-09-30 22:33:23 +08:00
|
|
|
|
cb.second(_programState, cb.first);
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
auto& uniformInfos = _programState->getProgram()->getAllActiveUniformInfo(ShaderStage::VERTEX);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
std::size_t bufferSize = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
char* buffer = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
_programState->getVertexUniformBuffer(&buffer, bufferSize);
|
2023-07-17 20:51:28 +08:00
|
|
|
|
std::size_t fragBufferSize = 0;
|
|
|
|
|
char* fragBuffer = nullptr;
|
|
|
|
|
_programState->getFragmentUniformBuffer(&fragBuffer, fragBufferSize);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
|
|
int i = 0;
|
2022-07-21 19:19:08 +08:00
|
|
|
|
for (auto&& iter : uniformInfos)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
auto& uniformInfo = iter.second;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (uniformInfo.size <= 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
continue;
|
|
|
|
|
|
2023-07-17 20:51:28 +08:00
|
|
|
|
if (iter.second.isUBO)
|
|
|
|
|
{
|
|
|
|
|
auto& ubo = program->getUBOHandler();
|
|
|
|
|
int offset = uniformInfo.bufferOffset;
|
|
|
|
|
if (iter.second.isFragment)
|
|
|
|
|
{
|
|
|
|
|
ubo.getFragmentBuffer()->updateSubData((void*)(fragBuffer + offset), offset, uniformInfo.size);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
program->getUBOHandler().getVertexBuffer()->updateSubData((void*)(buffer + offset), offset,
|
|
|
|
|
uniformInfo.size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
int elementCount = uniformInfo.count;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
setUniform(uniformInfo.isArray, uniformInfo.location, elementCount, uniformInfo.type,
|
|
|
|
|
(void*)(buffer + uniformInfo.bufferOffset));
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
|
const auto& textureInfo = _programState->getVertexTextureInfos();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
for (const auto& iter : textureInfo)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
2020-08-29 17:39:17 +08:00
|
|
|
|
/* About mutli textures support
|
2021-12-25 10:04:45 +08:00
|
|
|
|
* a. sampler2DArray, sampler2D[2], bind BackendTexture one by one, not use GL_TEXTURE_2D_ARRAY, not used
|
|
|
|
|
* at all engine interanl b. texture slot, one BackendTexture, multi GPU texture handlers, used by etc1,
|
|
|
|
|
* restrict: textures must have same size c. Bind multi BackendTexture to 1 Shader Program, see the
|
|
|
|
|
* ShaderTest
|
|
|
|
|
*/
|
2020-08-29 16:56:48 +08:00
|
|
|
|
auto& textures = iter.second.textures;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
auto& slots = iter.second.slots;
|
|
|
|
|
auto& indexs = iter.second.indexs;
|
|
|
|
|
auto location = iter.first;
|
2022-07-16 10:43:05 +08:00
|
|
|
|
#if AX_ENABLE_CACHE_TEXTURE_DATA
|
2019-11-23 20:27:39 +08:00
|
|
|
|
location = iter.second.location;
|
|
|
|
|
#endif
|
|
|
|
|
int i = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
for (const auto& texture : textures)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
2020-08-29 16:56:48 +08:00
|
|
|
|
applyTexture(texture, slots[i], indexs[i]);
|
|
|
|
|
++i;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
2021-09-30 08:18:01 +08:00
|
|
|
|
|
2020-08-29 16:56:48 +08:00
|
|
|
|
auto arrayCount = slots.size();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (arrayCount == 1) // Most of the time, not use sampler2DArray, should be 1
|
2020-08-29 16:56:48 +08:00
|
|
|
|
glUniform1i(location, slots[0]);
|
2020-08-29 17:39:17 +08:00
|
|
|
|
else
|
2021-09-30 08:18:44 +08:00
|
|
|
|
glUniform1iv(location, static_cast<GLsizei>(arrayCount), static_cast<const GLint*>(slots.data()));
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
2023-07-17 20:51:28 +08:00
|
|
|
|
|
|
|
|
|
auto& ubo = program->getUBOHandler();
|
|
|
|
|
glBindBufferBase(GL_UNIFORM_BUFFER, 0, ubo.getVertexBuffer()->getHandler());
|
|
|
|
|
glBindBufferBase(GL_UNIFORM_BUFFER, 1, ubo.getFragmentBuffer()->getHandler());
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
#define DEF_TO_INT(pointer, index) (*((GLint*)(pointer) + index))
|
|
|
|
|
#define DEF_TO_FLOAT(pointer, index) (*((GLfloat*)(pointer) + index))
|
2019-11-23 20:27:39 +08:00
|
|
|
|
void CommandBufferGL::setUniform(bool isArray, GLuint location, unsigned int size, GLenum uniformType, void* data) const
|
|
|
|
|
{
|
|
|
|
|
GLsizei count = size;
|
|
|
|
|
switch (uniformType)
|
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
|
case GL_INT:
|
|
|
|
|
case GL_BOOL:
|
|
|
|
|
case GL_SAMPLER_2D:
|
|
|
|
|
case GL_SAMPLER_CUBE:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform1iv(location, count, (GLint*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform1i(location, DEF_TO_INT(data, 0));
|
|
|
|
|
break;
|
|
|
|
|
case GL_INT_VEC2:
|
|
|
|
|
case GL_BOOL_VEC2:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform2iv(location, count, (GLint*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform2i(location, DEF_TO_INT(data, 0), DEF_TO_INT(data, 1));
|
|
|
|
|
break;
|
|
|
|
|
case GL_INT_VEC3:
|
|
|
|
|
case GL_BOOL_VEC3:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform3iv(location, count, (GLint*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform3i(location, DEF_TO_INT(data, 0), DEF_TO_INT(data, 1), DEF_TO_INT(data, 2));
|
|
|
|
|
break;
|
|
|
|
|
case GL_INT_VEC4:
|
|
|
|
|
case GL_BOOL_VEC4:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform4iv(location, count, (GLint*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform4i(location, DEF_TO_INT(data, 0), DEF_TO_INT(data, 1), DEF_TO_INT(data, 2), DEF_TO_INT(data, 4));
|
|
|
|
|
break;
|
|
|
|
|
case GL_FLOAT:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform1fv(location, count, (GLfloat*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform1f(location, DEF_TO_FLOAT(data, 0));
|
|
|
|
|
break;
|
|
|
|
|
case GL_FLOAT_VEC2:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform2fv(location, count, (GLfloat*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform2f(location, DEF_TO_FLOAT(data, 0), DEF_TO_FLOAT(data, 1));
|
|
|
|
|
break;
|
|
|
|
|
case GL_FLOAT_VEC3:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform3fv(location, count, (GLfloat*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform3f(location, DEF_TO_FLOAT(data, 0), DEF_TO_FLOAT(data, 1), DEF_TO_FLOAT(data, 2));
|
|
|
|
|
break;
|
|
|
|
|
case GL_FLOAT_VEC4:
|
|
|
|
|
if (isArray)
|
|
|
|
|
glUniform4fv(location, count, (GLfloat*)data);
|
|
|
|
|
else
|
|
|
|
|
glUniform4f(location, DEF_TO_FLOAT(data, 0), DEF_TO_FLOAT(data, 1), DEF_TO_FLOAT(data, 2),
|
|
|
|
|
DEF_TO_FLOAT(data, 3));
|
2019-11-23 20:27:39 +08:00
|
|
|
|
break;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
case GL_FLOAT_MAT2:
|
|
|
|
|
glUniformMatrix2fv(location, count, GL_FALSE, (GLfloat*)data);
|
|
|
|
|
break;
|
|
|
|
|
case GL_FLOAT_MAT3:
|
|
|
|
|
glUniformMatrix3fv(location, count, GL_FALSE, (GLfloat*)data);
|
|
|
|
|
break;
|
|
|
|
|
case GL_FLOAT_MAT4:
|
|
|
|
|
glUniformMatrix4fv(location, count, GL_FALSE, (GLfloat*)data);
|
|
|
|
|
break;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
2022-07-16 10:43:05 +08:00
|
|
|
|
AXASSERT(false, "invalidate Uniform data type");
|
2019-11-23 20:27:39 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::cleanResources()
|
|
|
|
|
{
|
2022-07-16 10:43:05 +08:00
|
|
|
|
AX_SAFE_RELEASE_NULL(_programState);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setLineWidth(float lineWidth)
|
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (lineWidth > 0.0f)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
glLineWidth(lineWidth);
|
|
|
|
|
else
|
|
|
|
|
glLineWidth(1.0f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CommandBufferGL::setScissorRect(bool isEnabled, float x, float y, float width, float height)
|
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (isEnabled)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
|
|
|
|
glEnable(GL_SCISSOR_TEST);
|
|
|
|
|
glScissor(x, y, width, height);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
glDisable(GL_SCISSOR_TEST);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
2020-09-21 22:10:50 +08:00
|
|
|
|
void CommandBufferGL::readPixels(RenderTarget* rt, std::function<void(const PixelBufferDescriptor&)> callback)
|
2019-11-23 20:27:39 +08:00
|
|
|
|
{
|
2020-09-11 01:19:10 +08:00
|
|
|
|
PixelBufferDescriptor pbd;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (rt->isDefaultRenderTarget())
|
|
|
|
|
{ // read pixels from screen
|
2020-09-21 22:10:50 +08:00
|
|
|
|
readPixels(rt, _viewPort.x, _viewPort.y, _viewPort.w, _viewPort.h, _viewPort.w * 4, pbd);
|
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
else
|
|
|
|
|
{
|
2020-09-21 22:10:50 +08:00
|
|
|
|
// we only readPixels from the COLOR0 attachment.
|
|
|
|
|
auto colorAttachment = rt->_color[0].texture;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (colorAttachment)
|
|
|
|
|
{
|
|
|
|
|
readPixels(rt, 0, 0, colorAttachment->getWidth(), colorAttachment->getHeight(),
|
|
|
|
|
colorAttachment->getWidth() * 4, pbd);
|
2020-09-21 22:10:50 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-11 01:19:10 +08:00
|
|
|
|
callback(pbd);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
|
void CommandBufferGL::readPixels(RenderTarget* rt,
|
|
|
|
|
int x,
|
|
|
|
|
int y,
|
|
|
|
|
uint32_t width,
|
|
|
|
|
uint32_t height,
|
|
|
|
|
uint32_t bytesPerRow,
|
|
|
|
|
PixelBufferDescriptor& pbd)
|
2020-09-21 22:10:50 +08:00
|
|
|
|
{
|
2022-06-24 14:18:48 +08:00
|
|
|
|
auto rtGL = static_cast<RenderTargetGL*>(rt);
|
|
|
|
|
rtGL->bindFrameBuffer();
|
2020-09-21 22:10:50 +08:00
|
|
|
|
|
|
|
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 1);
|
|
|
|
|
|
|
|
|
|
auto bufferSize = bytesPerRow * height;
|
2022-07-16 10:43:05 +08:00
|
|
|
|
#if (AX_TARGET_PLATFORM == AX_PLATFORM_WIN32 && defined(GL_ES_VERSION_3_0)) || \
|
|
|
|
|
(AX_TARGET_PLATFORM == AX_PLATFORM_ANDROID && defined(GL_PIXEL_PACK_BUFFER))
|
2020-09-21 22:10:50 +08:00
|
|
|
|
GLuint pbo;
|
|
|
|
|
glGenBuffers(1, &pbo);
|
|
|
|
|
glBindBuffer(GL_PIXEL_PACK_BUFFER, pbo);
|
|
|
|
|
glBufferData(GL_PIXEL_PACK_BUFFER, bufferSize, nullptr, GL_STATIC_DRAW);
|
|
|
|
|
glReadPixels(x, y, width, height, GL_RGBA, GL_UNSIGNED_BYTE, nullptr);
|
|
|
|
|
auto buffer = (uint8_t*)glMapBufferRange(GL_PIXEL_PACK_BUFFER, 0, bufferSize, GL_MAP_READ_BIT);
|
|
|
|
|
#else
|
|
|
|
|
std::unique_ptr<uint8_t[]> bufferStorage(new uint8_t[bufferSize]);
|
|
|
|
|
auto buffer = bufferStorage.get();
|
|
|
|
|
memset(buffer, 0, bufferSize);
|
|
|
|
|
glReadPixels(x, y, width, height, GL_RGBA, GL_UNSIGNED_BYTE, buffer);
|
|
|
|
|
#endif
|
|
|
|
|
uint8_t* wptr = nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
if (buffer && (wptr = pbd._data.resize(bufferSize)))
|
|
|
|
|
{
|
2020-09-21 22:10:50 +08:00
|
|
|
|
auto rptr = buffer + (height - 1) * bytesPerRow;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
for (int row = 0; row < height; ++row)
|
|
|
|
|
{
|
2020-09-21 22:10:50 +08:00
|
|
|
|
memcpy(wptr, rptr, bytesPerRow);
|
|
|
|
|
wptr += bytesPerRow;
|
|
|
|
|
rptr -= bytesPerRow;
|
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
pbd._width = width;
|
2020-09-21 22:10:50 +08:00
|
|
|
|
pbd._height = height;
|
|
|
|
|
}
|
2022-07-16 10:43:05 +08:00
|
|
|
|
#if (AX_TARGET_PLATFORM == AX_PLATFORM_WIN32 && defined(GL_ES_VERSION_3_0)) || \
|
|
|
|
|
(AX_TARGET_PLATFORM == AX_PLATFORM_ANDROID && defined(GL_PIXEL_PACK_BUFFER))
|
2020-09-21 22:10:50 +08:00
|
|
|
|
glUnmapBuffer(GL_PIXEL_PACK_BUFFER);
|
|
|
|
|
glBindBuffer(GL_PIXEL_PACK_BUFFER, 0);
|
|
|
|
|
glDeleteBuffers(1, &pbo);
|
|
|
|
|
#endif
|
|
|
|
|
|
2022-06-24 14:18:48 +08:00
|
|
|
|
if (!rtGL->isDefaultRenderTarget())
|
|
|
|
|
rtGL->unbindFrameBuffer();
|
2020-09-21 22:10:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
|
NS_AX_BACKEND_END
|