blob: dde8289b557125332961a4f06b5abfee60ace9c3 [file] [log] [blame]
#ifndef _SGLRSHADERPROGRAM_HPP
#define _SGLRSHADERPROGRAM_HPP
/*-------------------------------------------------------------------------
* drawElements Quality Program OpenGL ES Utilities
* ------------------------------------------------
*
* Copyright 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*//*!
* \file
* \brief SGLR shader program.
*//*--------------------------------------------------------------------*/
#include "tcuDefs.hpp"
#include "rrShaders.hpp"
#include "gluShaderUtil.hpp"
#include <vector>
#include <string>
namespace sglr
{
namespace rc
{
class Texture1D;
class Texture2D;
class TextureCube;
class Texture2DArray;
class Texture3D;
class TextureCubeArray;
} // rc
class ShaderProgram;
namespace pdec
{
enum VaryingFlags
{
VARYINGFLAG_NONE = 0,
VARYINGFLAG_FLATSHADE = (1 << 0),
};
struct VertexAttribute
{
VertexAttribute (const std::string& name_, rr::GenericVecType type_) : name(name_), type(type_) { }
std::string name;
rr::GenericVecType type;
};
struct VertexToFragmentVarying
{
VertexToFragmentVarying (rr::GenericVecType type_, int flags = VARYINGFLAG_NONE) : type(type_), flatshade((flags & VARYINGFLAG_FLATSHADE) != 0) { }
rr::GenericVecType type;
bool flatshade;
};
struct VertexToGeometryVarying
{
VertexToGeometryVarying (rr::GenericVecType type_, int flags = VARYINGFLAG_NONE) : type(type_), flatshade((flags & VARYINGFLAG_FLATSHADE) != 0) { }
rr::GenericVecType type;
bool flatshade;
};
struct GeometryToFragmentVarying
{
GeometryToFragmentVarying (rr::GenericVecType type_, int flags = VARYINGFLAG_NONE) : type(type_), flatshade((flags & VARYINGFLAG_FLATSHADE) != 0) { }
rr::GenericVecType type;
bool flatshade;
};
struct FragmentOutput
{
FragmentOutput (rr::GenericVecType type_) : type(type_) { }
rr::GenericVecType type;
};
struct Uniform
{
Uniform (const std::string& name_, glu::DataType type_) : name(name_), type(type_) { }
std::string name;
glu::DataType type;
};
struct VertexSource
{
VertexSource (const std::string& str) : source(str) { }
std::string source;
};
struct FragmentSource
{
FragmentSource (const std::string& str) : source(str) { }
std::string source;
};
struct GeometrySource
{
GeometrySource (const std::string& str) : source(str) { }
std::string source;
};
struct GeometryShaderDeclaration
{
GeometryShaderDeclaration (rr::GeometryShaderInputType inputType_,
rr::GeometryShaderOutputType outputType_,
size_t numOutputVertices_,
size_t numInvocations_ = 1)
: inputType (inputType_)
, outputType (outputType_)
, numOutputVertices (numOutputVertices_)
, numInvocations (numInvocations_)
{
}
rr::GeometryShaderInputType inputType;
rr::GeometryShaderOutputType outputType;
size_t numOutputVertices;
size_t numInvocations;
};
class ShaderProgramDeclaration
{
public:
ShaderProgramDeclaration (void);
ShaderProgramDeclaration& operator<< (const VertexAttribute&);
ShaderProgramDeclaration& operator<< (const VertexToFragmentVarying&);
ShaderProgramDeclaration& operator<< (const VertexToGeometryVarying&);
ShaderProgramDeclaration& operator<< (const GeometryToFragmentVarying&);
ShaderProgramDeclaration& operator<< (const FragmentOutput&);
ShaderProgramDeclaration& operator<< (const Uniform&);
ShaderProgramDeclaration& operator<< (const VertexSource&);
ShaderProgramDeclaration& operator<< (const FragmentSource&);
ShaderProgramDeclaration& operator<< (const GeometrySource&);
ShaderProgramDeclaration& operator<< (const GeometryShaderDeclaration&);
private:
inline bool hasGeometryShader (void) const { return m_geometryShaderSet; }
inline size_t getVertexInputCount (void) const { return m_vertexAttributes.size(); }
inline size_t getVertexOutputCount (void) const { return hasGeometryShader() ? m_vertexToGeometryVaryings.size() : m_vertexToFragmentVaryings.size(); }
inline size_t getFragmentInputCount (void) const { return hasGeometryShader() ? m_geometryToFragmentVaryings.size() : m_vertexToFragmentVaryings.size(); }
inline size_t getFragmentOutputCount (void) const { return m_fragmentOutputs.size(); }
inline size_t getGeometryInputCount (void) const { return hasGeometryShader() ? m_vertexToGeometryVaryings.size() : 0; }
inline size_t getGeometryOutputCount (void) const { return hasGeometryShader() ? m_geometryToFragmentVaryings.size() : 0; }
bool valid (void) const;
std::vector<VertexAttribute> m_vertexAttributes;
std::vector<VertexToFragmentVarying> m_vertexToFragmentVaryings;
std::vector<VertexToGeometryVarying> m_vertexToGeometryVaryings;
std::vector<GeometryToFragmentVarying> m_geometryToFragmentVaryings;
std::vector<FragmentOutput> m_fragmentOutputs;
std::vector<Uniform> m_uniforms;
std::string m_vertexSource;
std::string m_fragmentSource;
std::string m_geometrySource;
GeometryShaderDeclaration m_geometryDecl;
bool m_vertexShaderSet;
bool m_fragmentShaderSet;
bool m_geometryShaderSet;
friend class ::sglr::ShaderProgram;
};
} // pdec
struct UniformSlot
{
std::string name;
glu::DataType type;
union
{
deInt32 i;
deInt32 i4[4];
float f;
float f4[4];
float m3[3*3]; //!< row major, can be fed directly to tcu::Matrix constructor
float m4[4*4]; //!< row major, can be fed directly to tcu::Matrix constructor
} value;
union
{
const void* ptr;
const rc::Texture1D* tex1D;
const rc::Texture2D* tex2D;
const rc::TextureCube* texCube;
const rc::Texture2DArray* tex2DArray;
const rc::Texture3D* tex3D;
const rc::TextureCubeArray* texCubeArray;
} sampler;
inline UniformSlot (void)
: type(glu::TYPE_LAST)
{
value.i = 0;
sampler.ptr = DE_NULL;
}
};
class ShaderProgram : private rr::VertexShader, private rr::GeometryShader, private rr::FragmentShader
{
public:
ShaderProgram (const pdec::ShaderProgramDeclaration&);
virtual ~ShaderProgram (void);
const UniformSlot& getUniformByName (const char* name) const;
inline const rr::VertexShader* getVertexShader (void) const { return static_cast<const rr::VertexShader*>(this); }
inline const rr::FragmentShader* getFragmentShader (void) const { return static_cast<const rr::FragmentShader*>(this); }
inline const rr::GeometryShader* getGeometryShader (void) const { return static_cast<const rr::GeometryShader*>(this); }
private:
virtual void shadeVertices (const rr::VertexAttrib* inputs, rr::VertexPacket* const* packets, const int numPackets) const = 0;
virtual void shadeFragments (rr::FragmentPacket* packets, const int numPackets, const rr::FragmentShadingContext& context) const = 0;
virtual void shadePrimitives (rr::GeometryEmitter& output, int verticesIn, const rr::PrimitivePacket* packets, const int numPackets, int invocationID) const;
std::vector<std::string> m_attributeNames;
protected:
std::vector<UniformSlot> m_uniforms;
private:
const std::string m_vertSrc;
const std::string m_fragSrc;
const std::string m_geomSrc;
const bool m_hasGeometryShader;
friend class ReferenceContext; // for uniform access
friend class GLContext; // for source string access
} DE_WARN_UNUSED_TYPE;
} // sglr
#endif // _SGLRSHADERPROGRAM_HPP