qxLib
shader_program.inl
Go to the documentation of this file.
1 /**
2 
3  @file shader_program.inl
4  @author Khrapov
5  @date 16.01.2020
6  @copyright © Nick Khrapov, 2021. All right reserved.
7 
8 **/
9 
10 namespace qx
11 {
12 
13 inline base_shader_program::base_shader_program(base_shader_program&& baseShaderProgram) noexcept
14 {
15  std::swap(m_nProgram, baseShaderProgram.m_nProgram);
16 }
17 
18 inline base_shader_program::~base_shader_program() noexcept
19 {
20  if (m_nProgram != std::numeric_limits<GLuint>::max())
21  glDeleteProgram(m_nProgram);
22 }
23 
24 inline void base_shader_program::Init() noexcept
25 {
26  m_nProgram = glCreateProgram();
27 }
28 
29 template<GLenum ShaderType>
31 {
32  glAttachShader(m_nProgram, pShader->GetID());
33 }
34 
35 inline string base_shader_program::Link() noexcept
36 {
37  cstring sError;
38 
39  glLinkProgram(m_nProgram);
40  const GLint bSuccess = GetParameter(GL_LINK_STATUS);
41 
42  if (bSuccess != GL_TRUE)
43  {
44  const GLsizei nLogLength = GetParameter(GL_INFO_LOG_LENGTH);
45  sError = cstring(nLogLength, '\0');
46  glGetProgramInfoLog(m_nProgram, nLogLength, nullptr, sError.data());
47  }
48 
49  return qx::to_string(sError);
50 }
51 
52 inline void base_shader_program::Use() const noexcept
53 {
54  glUseProgram(m_nProgram);
55 }
56 
57 inline void base_shader_program::Unuse() const noexcept
58 {
59  glUseProgram(0);
60 }
61 
62 inline GLint base_shader_program::GetParameter(GLenum eParameter) const noexcept
63 {
64  GLint nRet = -1;
65  glGetProgramiv(m_nProgram, eParameter, &nRet);
66  return nRet;
67 }
68 
69 inline GLuint base_shader_program::GetBufferName() const noexcept
70 {
71  return m_nProgram;
72 }
73 
74 template<class T>
75 inline void base_shader_program::SetUniform(GLint nUniformLocation, const T* pValue, GLsizei nCount) noexcept
76 {
77  if constexpr (std::is_same_v<T, GLfloat>)
78  {
79  glUniform1fv(nUniformLocation, nCount, pValue);
80  }
81  else if constexpr (std::is_same_v<T, glm::vec2>)
82  {
83  glUniform2fv(nUniformLocation, nCount, glm::value_ptr(*pValue));
84  }
85  else if constexpr (std::is_same_v<T, glm::vec3>)
86  {
87  glUniform3fv(nUniformLocation, nCount, glm::value_ptr(*pValue));
88  }
89  else if constexpr (std::is_same_v<T, glm::vec4>)
90  {
91  glUniform4fv(nUniformLocation, nCount, glm::value_ptr(*pValue));
92  }
93  else if constexpr (std::is_same_v<T, GLint>)
94  {
95  glUniform1iv(nUniformLocation, nCount, pValue);
96  }
97  else if constexpr (std::is_same_v<T, glm::ivec2>)
98  {
99  glUniform2iv(nUniformLocation, nCount, glm::value_ptr(*pValue));
100  }
101  else if constexpr (std::is_same_v<T, glm::ivec3>)
102  {
103  glUniform3iv(nUniformLocation, nCount, glm::value_ptr(*pValue));
104  }
105  else if constexpr (std::is_same_v<T, glm::ivec4>)
106  {
107  glUniform4iv(nUniformLocation, nCount, glm::value_ptr(*pValue));
108  }
109  else if constexpr (std::is_same_v<T, GLuint>)
110  {
111  glUniform1uiv(nUniformLocation, nCount, pValue);
112  }
113  else if constexpr (std::is_same_v<T, glm::uvec2>)
114  {
115  glUniform2uiv(nUniformLocation, nCount, glm::value_ptr(*pValue));
116  }
117  else if constexpr (std::is_same_v<T, glm::uvec3>)
118  {
119  glUniform3uiv(nUniformLocation, nCount, glm::value_ptr(*pValue));
120  }
121  else if constexpr (std::is_same_v<T, glm::uvec4>)
122  {
123  glUniform4uiv(nUniformLocation, nCount, glm::value_ptr(*pValue));
124  }
125  else if constexpr (std::is_same_v<T, glm::mat2>)
126  {
127  glUniformMatrix2fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
128  }
129  else if constexpr (std::is_same_v<T, glm::mat3>)
130  {
131  glUniformMatrix3fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
132  }
133  else if constexpr (std::is_same_v<T, glm::mat4>)
134  {
135  glUniformMatrix4fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
136  }
137  else if constexpr (std::is_same_v<T, glm::mat2x3>)
138  {
139  glUniformMatrix2x3fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
140  }
141  else if constexpr (std::is_same_v<T, glm::mat3x2>)
142  {
143  glUniformMatrix3x2fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
144  }
145  else if constexpr (std::is_same_v<T, glm::mat2x4>)
146  {
147  glUniformMatrix2x4fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
148  }
149  else if constexpr (std::is_same_v<T, glm::mat4x2>)
150  {
151  glUniformMatrix4x2fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
152  }
153  else if constexpr (std::is_same_v<T, glm::mat3x4>)
154  {
155  glUniformMatrix3x4fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
156  }
157  else if constexpr (std::is_same_v<T, glm::mat4x3>)
158  {
159  glUniformMatrix4x3fv(nUniformLocation, nCount, GL_FALSE, glm::value_ptr(*pValue));
160  }
161  else
162  {
163  []<bool flag = false>()
164  {
165  static_assert(flag, "Uniform type is not supported");
166  }
167  ();
168  }
169 }
170 
171 template<class T>
172 inline void base_shader_program::SetUniform(const GLchar* pszName, const T* pValue, GLsizei nCount) noexcept
173 {
174  SetUniform(GetUniformLocation(pszName), pValue, nCount);
175 }
176 
177 template<class T>
178 inline void base_shader_program::SetUniform(GLint nUniformLocation, const T& value) noexcept
179 {
180  using type = std::remove_cvref_t<T>;
181  if constexpr (std::is_same_v<T, GLfloat>)
182  glUniform1f(nUniformLocation, value);
183  else if constexpr (std::is_same_v<T, glm::vec2>)
184  glUniform2f(nUniformLocation, value.x, value.y);
185  else if constexpr (std::is_same_v<T, glm::vec3>)
186  glUniform3f(nUniformLocation, value.x, value.y, value.z);
187  else if constexpr (std::is_same_v<T, glm::vec4>)
188  glUniform4f(nUniformLocation, value.x, value.y, value.z, value.w);
189  else if constexpr (std::is_same_v<T, GLint>)
190  glUniform1i(nUniformLocation, value);
191  else if constexpr (std::is_same_v<T, glm::ivec2>)
192  glUniform2i(nUniformLocation, value.x, value.y);
193  else if constexpr (std::is_same_v<T, glm::ivec3>)
194  glUniform3i(nUniformLocation, value.x, value.y, value.z);
195  else if constexpr (std::is_same_v<T, glm::ivec4>)
196  glUniform4i(nUniformLocation, value.x, value.y, value.z, value.w);
197  else if constexpr (std::is_same_v<T, bool>)
198  glUniform1i(nUniformLocation, value ? GL_TRUE : GL_FALSE);
199  else
200  SetUniform(nUniformLocation, &value, 1);
201 }
202 
203 template<class T>
204 inline void base_shader_program::SetUniform(const GLchar* pszName, const T& value) noexcept
205 {
206  SetUniform(GetUniformLocation(pszName), value);
207 }
208 
209 inline GLint base_shader_program::GetUniformLocation(const GLchar* pszName) const noexcept
210 {
211  return glGetUniformLocation(m_nProgram, pszName);
212 }
213 
215  const char* pszName,
216  GLint nNameLength,
217  const char* pszText,
218  GLint nTextLength) noexcept
219 {
220  const bool bGlslIncludeSupported = GLEW_ARB_shading_language_include != 0;
221  if (bGlslIncludeSupported)
222  {
223  glNamedStringARB(GL_SHADER_INCLUDE_ARB, nNameLength, pszName, nTextLength, pszText);
224  }
225 
226  return bGlslIncludeSupported;
227 }
228 
229 inline void base_shader_program::DispatchCompute(GLuint nGroupsX, GLuint nGroupsY, GLuint nGroupsZ) noexcept
230 {
231  glDispatchCompute(nGroupsX, nGroupsY, nGroupsZ);
232 }
233 
234 inline bool base_shader_program::operator==(const base_shader_program& other) const noexcept
235 {
236  return m_nProgram == other.m_nProgram;
237 }
238 
239 inline base_shader_program& base_shader_program::operator=(base_shader_program&& baseShaderProgram) noexcept
240 {
241  std::swap(m_nProgram, baseShaderProgram.m_nProgram);
242  return *this;
243 }
244 
245 } // namespace qx
Shader program class.
void Init() noexcept
Init shader program.
static bool AddInclude(const char *pszName, GLint nNameLength, const char *pszText, GLint nTextLength) noexcept
Add include string.
void SetUniform(GLint nUniformLocation, const T *pValue, GLsizei nCount) noexcept
Specify the value of a uniform variable.
static void DispatchCompute(GLuint nGroupsX, GLuint nGroupsY, GLuint nGroupsZ) noexcept
Dispatch program compute.
void Unuse() const noexcept
Drop current shader.
GLuint GetBufferName() const noexcept
Get shader buffer name.
void AttachShader(shader_base< ShaderType > *pShader) noexcept
Attach shader to the program.
GLint GetParameter(GLenum eParameter) const noexcept
Get shader program parameter.
void Use() const noexcept
Use shader program.
string Link() noexcept
Link attached shaders.
GLint GetUniformLocation(const GLchar *pszName) const noexcept
Get uniform location based on it's name.
String class.
Definition: string.h:64
OpenGL base shader.
Definition: shaders.h:43
string to_string(cstring_view stringView, const std::locale &locale=std::locale())
Convert a char string to common string type.