2019-11-23 20:27:39 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
|
|
|
|
2022-07-09 22:23:34 +08:00
|
|
|
https://axis-project.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.
|
|
|
|
****************************************************************************/
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <stack>
|
|
|
|
#include <array>
|
|
|
|
#include <deque>
|
|
|
|
|
|
|
|
#include "platform/CCPlatformMacros.h"
|
|
|
|
#include "renderer/CCRenderCommand.h"
|
|
|
|
#include "renderer/backend/Types.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @addtogroup renderer
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_BEGIN
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
using CullMode = backend::CullMode;
|
2021-12-25 10:04:45 +08:00
|
|
|
using Winding = backend::Winding;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
namespace backend
|
|
|
|
{
|
2020-09-13 11:11:48 +08:00
|
|
|
class Buffer;
|
|
|
|
class CommandBuffer;
|
|
|
|
class RenderPipeline;
|
|
|
|
class RenderPass;
|
|
|
|
class TextureBackend;
|
2020-09-21 22:10:50 +08:00
|
|
|
class RenderTarget;
|
2020-09-13 11:11:48 +08:00
|
|
|
struct PixelBufferDescriptor;
|
2021-12-25 10:04:45 +08:00
|
|
|
} // namespace backend
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
class EventListenerCustom;
|
|
|
|
class TrianglesCommand;
|
|
|
|
class MeshCommand;
|
|
|
|
class GroupCommand;
|
|
|
|
class CallbackCommand;
|
|
|
|
struct PipelineDescriptor;
|
|
|
|
class Texture2D;
|
|
|
|
|
|
|
|
/** Class that knows how to sort `RenderCommand` objects.
|
|
|
|
Since the commands that have `z == 0` are "pushed back" in
|
|
|
|
the correct order, the only `RenderCommand` objects that need to be sorted,
|
|
|
|
are the ones that have `z < 0` and `z > 0`.
|
|
|
|
*/
|
|
|
|
class RenderQueue
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
RenderCommand will be divided into Queue Groups.
|
|
|
|
*/
|
|
|
|
enum QUEUE_GROUP
|
|
|
|
{
|
|
|
|
/**Objects with globalZ smaller than 0.*/
|
|
|
|
GLOBALZ_NEG = 0,
|
|
|
|
/**Opaque 3D objects with 0 globalZ.*/
|
|
|
|
OPAQUE_3D = 1,
|
|
|
|
/**Transparent 3D objects with 0 globalZ.*/
|
|
|
|
TRANSPARENT_3D = 2,
|
|
|
|
/**2D objects with 0 globalZ.*/
|
|
|
|
GLOBALZ_ZERO = 3,
|
|
|
|
/**Objects with globalZ bigger than 0.*/
|
|
|
|
GLOBALZ_POS = 4,
|
|
|
|
QUEUE_COUNT = 5,
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
/**Constructor.*/
|
|
|
|
RenderQueue();
|
|
|
|
/**Push a renderCommand into current renderqueue.*/
|
|
|
|
void push_back(RenderCommand* command);
|
|
|
|
/**Return the number of render commands.*/
|
|
|
|
ssize_t size() const;
|
|
|
|
/**Sort the render commands.*/
|
|
|
|
void sort();
|
|
|
|
/**Treat sorted commands as an array, access them one by one.*/
|
|
|
|
RenderCommand* operator[](ssize_t index) const;
|
|
|
|
/**Clear all rendered commands.*/
|
|
|
|
void clear();
|
|
|
|
/**Realloc command queues and reserve with given size. Note: this clears any existing commands.*/
|
|
|
|
void realloc(size_t reserveSize);
|
|
|
|
/**Get a sub group of the render queue.*/
|
|
|
|
std::vector<RenderCommand*>& getSubQueue(QUEUE_GROUP group) { return _commands[group]; }
|
|
|
|
/**Get the number of render commands contained in a subqueue.*/
|
|
|
|
ssize_t getSubQueueSize(QUEUE_GROUP group) const { return _commands[group].size(); }
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
protected:
|
|
|
|
/**The commands in the render queue.*/
|
|
|
|
std::vector<RenderCommand*> _commands[QUEUE_COUNT];
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
/**Cull state.*/
|
|
|
|
bool _isCullEnabled;
|
|
|
|
/**Depth test enable state.*/
|
|
|
|
bool _isDepthEnabled;
|
|
|
|
/**Depth buffer write state.*/
|
|
|
|
bool _isDepthWrite;
|
|
|
|
};
|
|
|
|
|
|
|
|
class GroupCommandManager;
|
|
|
|
|
|
|
|
/* Class responsible for the rendering in.
|
|
|
|
|
|
|
|
Whenever possible prefer to use `TrianglesCommand` objects since the renderer will automatically batch them.
|
|
|
|
*/
|
2022-07-16 10:43:05 +08:00
|
|
|
class AX_DLL Renderer
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
/**The max number of vertices in a vertex buffer object.*/
|
|
|
|
static const int VBO_SIZE = 65536;
|
|
|
|
/**The max number of indices in a index buffer.*/
|
|
|
|
static const int INDEX_VBO_SIZE = VBO_SIZE * 6 / 4;
|
|
|
|
/**The rendercommands which can be batched will be saved into a list, this is the reserved size of this list.*/
|
|
|
|
static const int BATCH_TRIAGCOMMAND_RESERVED_SIZE = 64;
|
|
|
|
/**Reserved for material id, which means that the command could not be batched.*/
|
|
|
|
static const int MATERIAL_ID_DO_NOT_BATCH = 0;
|
|
|
|
/**Constructor.*/
|
|
|
|
Renderer();
|
|
|
|
/**Destructor.*/
|
|
|
|
~Renderer();
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
// TODO: manage GLView inside Render itself
|
2019-11-23 20:27:39 +08:00
|
|
|
void init();
|
|
|
|
|
2022-06-24 14:18:48 +08:00
|
|
|
void addCallbackCommand(std::function<void()> func, float globalZOrder = 0.0f);
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
/** Adds a `RenderComamnd` into the renderer */
|
|
|
|
void addCommand(RenderCommand* command);
|
|
|
|
|
|
|
|
/** Adds a `RenderComamnd` into the renderer specifying a particular render queue ID */
|
|
|
|
void addCommand(RenderCommand* command, int renderQueueID);
|
|
|
|
|
|
|
|
/** Pushes a group into the render queue */
|
|
|
|
void pushGroup(int renderQueueID);
|
|
|
|
|
|
|
|
/** Pops a group from the render queue */
|
|
|
|
void popGroup();
|
|
|
|
|
|
|
|
/** Creates a render queue and returns its Id */
|
|
|
|
int createRenderQueue();
|
|
|
|
|
|
|
|
/** Renders into the GLView all the queued `RenderCommand` objects */
|
|
|
|
void render();
|
|
|
|
|
|
|
|
/** Cleans all `RenderCommand`s in the queue */
|
|
|
|
void clean();
|
|
|
|
|
|
|
|
/* returns the number of drawn batches in the last frame */
|
|
|
|
ssize_t getDrawnBatches() const { return _drawnBatches; }
|
|
|
|
/* RenderCommands (except) TrianglesCommand should update this value */
|
|
|
|
void addDrawnBatches(ssize_t number) { _drawnBatches += number; };
|
|
|
|
/* returns the number of drawn triangles in the last frame */
|
|
|
|
ssize_t getDrawnVertices() const { return _drawnVertices; }
|
|
|
|
/* RenderCommands (except) TrianglesCommand should update this value */
|
|
|
|
void addDrawnVertices(ssize_t number) { _drawnVertices += number; };
|
|
|
|
/* clear draw stats */
|
2022-07-01 23:15:20 +08:00
|
|
|
void clearDrawStats() { _drawnBatches = _drawnVertices = 0; }
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
Set render targets. If not set, will use default render targets. It will effect all commands.
|
|
|
|
@flags Flags to indicate which attachment to be replaced.
|
|
|
|
@colorAttachment The value to replace color attachment, only one color attachment supported now.
|
|
|
|
@depthAttachment The value to repalce depth attachment.
|
|
|
|
@stencilAttachment The value to replace stencil attachment. Depth attachment and stencil attachment
|
|
|
|
can be the same value.
|
|
|
|
*/
|
2020-09-21 22:10:50 +08:00
|
|
|
backend::RenderTarget* getRenderTarget() const { return _currentRT; }
|
|
|
|
void setRenderTarget(backend::RenderTarget* rt) { _currentRT = rt; };
|
|
|
|
|
|
|
|
backend::RenderTarget* getDefaultRenderTarget() const { return _defaultRT; }
|
|
|
|
|
2022-06-24 14:18:48 +08:00
|
|
|
/* The offscreen render target for RenderTexture to share it */
|
|
|
|
backend::RenderTarget* getOffscreenRenderTarget();
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
/**
|
|
|
|
Set clear values for each attachment.
|
|
|
|
@flags Flags to indicate which attachment clear value to be modified.
|
|
|
|
@color The clear color value.
|
|
|
|
@depth The clear depth value.
|
|
|
|
@stencil The clear stencil value.
|
|
|
|
*/
|
|
|
|
void clear(ClearFlag flags, const Color4F& color, float depth, unsigned int stencil, float globalOrder);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get color clear value.
|
|
|
|
* @return Color clear value.
|
|
|
|
*/
|
|
|
|
const Color4F& getClearColor() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get depth clear value.
|
2021-12-25 10:04:45 +08:00
|
|
|
* @return Depth clear value.
|
2019-11-23 20:27:39 +08:00
|
|
|
*/
|
|
|
|
float getClearDepth() const;
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Get the index when the stencil buffer is cleared.
|
|
|
|
* @return The index used when the stencil buffer is cleared.
|
2019-11-23 20:27:39 +08:00
|
|
|
*/
|
|
|
|
unsigned int getClearStencil() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the clear flag.
|
|
|
|
* @return The clear flag.
|
|
|
|
*/
|
|
|
|
ClearFlag getClearFlag() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the render target flag.
|
|
|
|
* @return The render target flag.
|
|
|
|
*/
|
|
|
|
RenderTargetFlag getRenderTargetFlag() const;
|
|
|
|
|
|
|
|
// depth/stencil state.
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Enable/disable depth test.
|
2019-11-23 20:27:39 +08:00
|
|
|
* @param value true means enable depth test, otherwise false.
|
|
|
|
*/
|
|
|
|
void setDepthTest(bool value);
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Enable/disable to update depth buffer.
|
2019-11-23 20:27:39 +08:00
|
|
|
* @param value true means enable writing into the depth buffer, otherwise false.
|
|
|
|
*/
|
|
|
|
void setDepthWrite(bool value);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set depth compare function.
|
|
|
|
* @param func Specifies the value used for depth buffer comparisons.
|
|
|
|
*/
|
|
|
|
void setDepthCompareFunction(backend::CompareFunction func);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get whether depth test state is enabled or disabled.
|
|
|
|
* @return true if depth test is enabled, otherwise false.
|
|
|
|
*/
|
|
|
|
bool getDepthTest() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get whether writing to depth buffer is enabled or not.
|
|
|
|
* @return true if enable writing into the depth buffer, false otherwise.
|
|
|
|
*/
|
|
|
|
bool getDepthWrite() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get depth compare function.
|
|
|
|
* @return Depth compare function.
|
|
|
|
*/
|
|
|
|
backend::CompareFunction getDepthCompareFunction() const;
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Enable/disable stencil test.
|
2019-11-23 20:27:39 +08:00
|
|
|
* @param value true means enable stencil test, otherwise false.
|
|
|
|
*/
|
|
|
|
void setStencilTest(bool value);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set front and back function and reference value for stencil testing.
|
|
|
|
* @param func Specifies the stencil test function.
|
2021-12-25 10:04:45 +08:00
|
|
|
* @param ref Specifies the reference value for the stencil test.
|
|
|
|
* @readMask Specifies a mask that is ANDed with both the reference value and the stored stencil value when the test
|
|
|
|
* is done.
|
2019-11-23 20:27:39 +08:00
|
|
|
*/
|
|
|
|
void setStencilCompareFunction(backend::CompareFunction func, unsigned int ref, unsigned int readMask);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set front and back stencil test actions.
|
2021-12-25 10:04:45 +08:00
|
|
|
* @param stencilFailureOp Specifies the action to take when the stencil test fails.
|
|
|
|
* @param depthFailureOp Specifies the stencil action when the stencil test passes, but the depth test fails.
|
|
|
|
* @param stencilDepthPassOp Specifies the stencil action when both the stencil test and the depth test pass, or
|
|
|
|
* when the stencil test passes and either there is no depth buffer or depth testing is not enabled.
|
2019-11-23 20:27:39 +08:00
|
|
|
*/
|
|
|
|
void setStencilOperation(backend::StencilOperation stencilFailureOp,
|
|
|
|
backend::StencilOperation depthFailureOp,
|
|
|
|
backend::StencilOperation stencilDepthPassOp);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Control the front and back writing of individual bits in the stencil planes.
|
|
|
|
* @param mask Specifies a bit mask to enable and disable writing of individual bits in the stencil planes.
|
2021-12-25 10:04:45 +08:00
|
|
|
*/
|
2019-11-23 20:27:39 +08:00
|
|
|
void setStencilWriteMask(unsigned int mask);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get whether stencil test is enabled or not.
|
|
|
|
* @return true if stencil test is enabled, false otherwise.
|
|
|
|
*/
|
|
|
|
bool getStencilTest() const;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
/// Get the action to take when the stencil test fails.
|
2019-11-23 20:27:39 +08:00
|
|
|
backend::StencilOperation getStencilFailureOperation() const;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
/// Get the stencil action when the stencil test passes, but the depth test fails.
|
2019-11-23 20:27:39 +08:00
|
|
|
backend::StencilOperation getStencilPassDepthFailureOperation() const;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
/// Get the stencil action when both the stencil test and the depth test pass, or when the stencil test passes and
|
|
|
|
/// either there is no depth buffer or depth testing is not enabled.
|
2019-11-23 20:27:39 +08:00
|
|
|
backend::StencilOperation getStencilDepthPassOperation() const;
|
|
|
|
|
|
|
|
/// Get the stencil test function.
|
|
|
|
backend::CompareFunction getStencilCompareFunction() const;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
/**
|
|
|
|
* Get the stencil readMask.
|
|
|
|
* @return Stencil read mask.
|
|
|
|
* @see `setStencilCompareFunction(backend::CompareFunction func, unsigned int ref, unsigned int readMask)`
|
2021-12-25 10:04:45 +08:00
|
|
|
*/
|
2019-11-23 20:27:39 +08:00
|
|
|
unsigned int getStencilReadMask() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the stencil write mask.
|
|
|
|
* @return Stencil write mask.
|
|
|
|
* @see `setStencilWriteMask(unsigned int mask)`
|
|
|
|
*/
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned int getStencilWriteMask() const;
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Get stencil reference value set by `setStencilCompareFunction`.
|
2019-11-23 20:27:39 +08:00
|
|
|
* @return Stencil reference value.
|
|
|
|
* @see `setStencilCompareFunction(backend::CompareFunction func, unsigned int ref, unsigned int readMask)`
|
|
|
|
*/
|
|
|
|
unsigned int getStencilReferenceValue() const;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2021-04-22 22:01:47 +08:00
|
|
|
/**
|
|
|
|
* Sets depth stencil descriptor
|
|
|
|
*/
|
|
|
|
void setDepthStencilDesc(const backend::DepthStencilDescriptor& dsDesc);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2021-04-22 22:01:47 +08:00
|
|
|
/**
|
|
|
|
* Gets depth stencil descriptor
|
|
|
|
*/
|
|
|
|
const backend::DepthStencilDescriptor& getDepthStencilDesc() const;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Fixed-function state
|
|
|
|
* @param mode Controls if primitives are culled when front facing, back facing, or not culled at all.
|
|
|
|
*/
|
|
|
|
void setCullMode(CullMode mode) { _cullMode = mode; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get cull mode.
|
|
|
|
* @return The cull mode.
|
|
|
|
*/
|
|
|
|
CullMode getCullMode() const { return _cullMode; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fixed-function state
|
|
|
|
* @param winding The winding order of front-facing primitives.
|
|
|
|
*/
|
|
|
|
void setWinding(Winding winding) { _winding = winding; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get winding mode.
|
|
|
|
* @return The winding mode.
|
|
|
|
*/
|
|
|
|
Winding getWinding() const { return _winding; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fixed-function state
|
|
|
|
* @param x The x coordinate of the upper-left corner of the viewport.
|
|
|
|
* @param y The y coordinate of the upper-left corner of the viewport.
|
|
|
|
* @param w The width of the viewport, in pixels.
|
|
|
|
* @param h The height of the viewport, in pixels.
|
|
|
|
*/
|
|
|
|
void setViewPort(int x, int y, unsigned int w, unsigned int h);
|
|
|
|
|
|
|
|
/// Get viewport.
|
|
|
|
const Viewport& getViewport() const { return _viewport; }
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Enable/disable scissor test.
|
2019-11-23 20:27:39 +08:00
|
|
|
* @param enabled true if enable scissor test, false otherwise.
|
|
|
|
*/
|
|
|
|
void setScissorTest(bool enabled);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fixed-function state
|
|
|
|
* @param x, y Specifies the lower left corner of the scissor box
|
|
|
|
* @param wdith Specifies the width of the scissor box
|
|
|
|
* @param height Specifies the height of the scissor box
|
|
|
|
*/
|
|
|
|
void setScissorRect(float x, float y, float width, float height);
|
2021-12-25 10:04:45 +08:00
|
|
|
bool getScissorTest() const; ///< Get whether scissor test is enabled or not.
|
|
|
|
const ScissorRect& getScissorRect() const; ///< Get scissor rectangle.
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
/** returns whether or not a rectangle is visible or not */
|
2021-10-23 23:27:14 +08:00
|
|
|
bool checkVisibility(const Mat4& transform, const Vec2& size);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2020-09-21 22:10:50 +08:00
|
|
|
/** read pixels from RenderTarget or screen framebuffer */
|
|
|
|
void readPixels(backend::RenderTarget* rt, std::function<void(const backend::PixelBufferDescriptor&)> callback);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
void beginRenderPass(); /// Begin a render pass.
|
2021-06-17 22:22:41 +08:00
|
|
|
void endRenderPass();
|
|
|
|
|
2022-06-24 14:18:48 +08:00
|
|
|
CallbackCommand* nextCallbackCommand();
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
protected:
|
|
|
|
friend class Director;
|
|
|
|
friend class GroupCommand;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create and reuse vertex and index buffer for triangleCommand.
|
|
|
|
* When queued vertex or index count exceed the limited value, a new vertex or index buffer will be created.
|
|
|
|
*/
|
|
|
|
class TriangleCommandBufferManager
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
~TriangleCommandBufferManager();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new vertex buffer and a index buffer and push it to cache.
|
|
|
|
* @note Should invoke firstly.
|
|
|
|
*/
|
|
|
|
void init();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reset avalable buffer index to zero.
|
2021-12-25 10:04:45 +08:00
|
|
|
* That means when get vertex buffer or index buffer, the earliest created buffer object in the cache will be
|
|
|
|
* returned.
|
2019-11-23 20:27:39 +08:00
|
|
|
*/
|
|
|
|
void putbackAllBuffers();
|
|
|
|
|
|
|
|
/**
|
2021-12-25 10:04:45 +08:00
|
|
|
* Buffer will be created If next buffer unavailable in the cache, otherwise set the buffer index in order to
|
|
|
|
* get the next available buffer.
|
2019-11-23 20:27:39 +08:00
|
|
|
*/
|
|
|
|
void prepareNextBuffer();
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
backend::Buffer* getVertexBuffer() const; ///< Get the vertex buffer.
|
|
|
|
backend::Buffer* getIndexBuffer() const; ///< Get the index buffer.
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
void createBuffer();
|
|
|
|
|
|
|
|
int _currentBufferIndex = 0;
|
|
|
|
std::vector<backend::Buffer*> _vertexBufferPool;
|
|
|
|
std::vector<backend::Buffer*> _indexBufferPool;
|
|
|
|
};
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
inline GroupCommandManager* getGroupCommandManager() const { return _groupCommandManager; }
|
2019-11-23 20:27:39 +08:00
|
|
|
void drawBatchedTriangles();
|
|
|
|
void drawCustomCommand(RenderCommand* command);
|
|
|
|
void drawMeshCommand(RenderCommand* command);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
bool beginFrame(); /// Indicate the begining of a frame
|
|
|
|
void endFrame(); /// Finish a frame.
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
/// Draw the previews queued triangles and flush previous context
|
2019-11-23 20:27:39 +08:00
|
|
|
void flush();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
void flush2D();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
void flush3D();
|
|
|
|
|
|
|
|
void flushTriangles();
|
|
|
|
|
|
|
|
void processRenderCommand(RenderCommand* command);
|
|
|
|
void processGroupCommand(GroupCommand*);
|
|
|
|
void visitRenderQueue(RenderQueue& queue);
|
|
|
|
void doVisitRenderQueue(const std::vector<RenderCommand*>&);
|
|
|
|
|
|
|
|
void fillVerticesAndIndices(const TrianglesCommand* cmd, unsigned int vertexBufferOffset);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
void pushStateBlock();
|
|
|
|
|
|
|
|
void popStateBlock();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
backend::RenderPipeline* _renderPipeline = nullptr;
|
|
|
|
|
|
|
|
Viewport _viewport;
|
2021-12-25 10:04:45 +08:00
|
|
|
CullMode _cullMode = CullMode::NONE;
|
|
|
|
Winding _winding = Winding::COUNTER_CLOCK_WISE; // default front face is CCW in GL
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
std::stack<int> _commandGroupStack;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
std::vector<RenderQueue> _renderGroups;
|
|
|
|
|
|
|
|
std::vector<TrianglesCommand*> _queuedTriangleCommands;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2022-06-24 14:18:48 +08:00
|
|
|
// the pool for callback commands
|
|
|
|
std::vector<CallbackCommand*> _callbackCommandsPool;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
// for TrianglesCommand
|
2019-11-23 20:27:39 +08:00
|
|
|
V3F_C4B_T2F _verts[VBO_SIZE];
|
|
|
|
unsigned short _indices[INDEX_VBO_SIZE];
|
|
|
|
backend::Buffer* _vertexBuffer = nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
backend::Buffer* _indexBuffer = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
TriangleCommandBufferManager _triangleCommandBufferManager;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
backend::CommandBuffer* _commandBuffer = nullptr;
|
2021-04-22 22:01:47 +08:00
|
|
|
backend::RenderPassDescriptor _renderPassDesc;
|
2020-09-22 16:32:17 +08:00
|
|
|
|
|
|
|
backend::DepthStencilState* _depthStencilState = nullptr;
|
2021-04-22 22:01:47 +08:00
|
|
|
backend::DepthStencilDescriptor _dsDesc;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
// Internal structure that has the information for the batches
|
|
|
|
struct TriBatchToDraw
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
TrianglesCommand* cmd = nullptr; // needed for the Material
|
2019-11-23 20:27:39 +08:00
|
|
|
unsigned int indicesToDraw = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned int offset = 0;
|
2019-11-23 20:27:39 +08:00
|
|
|
};
|
|
|
|
// capacity of the array of TriBatches
|
|
|
|
int _triBatchesToDrawCapacity = 500;
|
|
|
|
// the TriBatches
|
|
|
|
TriBatchToDraw* _triBatchesToDraw = nullptr;
|
|
|
|
|
|
|
|
unsigned int _queuedTotalVertexCount = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned int _queuedTotalIndexCount = 0;
|
|
|
|
unsigned int _queuedVertexCount = 0;
|
|
|
|
unsigned int _queuedIndexCount = 0;
|
|
|
|
unsigned int _filledIndex = 0;
|
|
|
|
unsigned int _filledVertex = 0;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
// stats
|
2021-12-25 10:04:45 +08:00
|
|
|
size_t _drawnBatches = 0;
|
2021-08-27 12:34:24 +08:00
|
|
|
size_t _drawnVertices = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
// the flag for checking whether renderer is rendering
|
|
|
|
bool _isRendering = false;
|
2019-11-23 20:27:39 +08:00
|
|
|
bool _isDepthTestFor2D = false;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
GroupCommandManager* _groupCommandManager = nullptr;
|
|
|
|
|
|
|
|
unsigned int _stencilRef = 0;
|
|
|
|
|
2020-09-21 22:10:50 +08:00
|
|
|
backend::RenderTarget* _defaultRT = nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
backend::RenderTarget* _currentRT = nullptr; // weak ref
|
2020-09-21 22:10:50 +08:00
|
|
|
|
2022-06-24 14:18:48 +08:00
|
|
|
backend::RenderTarget* _offscreenRT = nullptr;
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
Color4F _clearColor = Color4F::BLACK;
|
|
|
|
ClearFlag _clearFlag;
|
|
|
|
|
|
|
|
struct ScissorState
|
|
|
|
{
|
|
|
|
ScissorRect rect;
|
|
|
|
bool isEnabled = false;
|
|
|
|
};
|
|
|
|
ScissorState _scissorState;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
struct StateBlock
|
|
|
|
{
|
|
|
|
bool depthTest = false;
|
|
|
|
bool depthWrite = false;
|
|
|
|
backend::CullMode cullMode = backend::CullMode::NONE;
|
2019-11-23 20:27:39 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
std::deque<StateBlock> _stateBlockStack;
|
|
|
|
};
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_END
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
end of support group
|
|
|
|
@}
|
|
|
|
*/
|