blob: cac198540ed2ef497cf7869520ae8464e3e73da8 [file] [log] [blame]
Geoff Langf9a6f082015-01-22 13:32:49 -05001//
2// Copyright 2015 The ANGLE Project Authors. All rights reserved.
3// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
7// ProgramGL.cpp: Implements the class methods for ProgramGL.
8
9#include "libANGLE/renderer/gl/ProgramGL.h"
10
11#include "common/debug.h"
Geoff Lang5ed74cf2015-04-14 13:57:07 -040012#include "common/utilities.h"
Geoff Langb1f435e2015-02-20 10:01:01 -050013#include "libANGLE/renderer/gl/FunctionsGL.h"
14#include "libANGLE/renderer/gl/ShaderGL.h"
15#include "libANGLE/renderer/gl/StateManagerGL.h"
unknownb4a3af22015-11-25 15:02:51 -050016#include "platform/Platform.h"
Geoff Langf9a6f082015-01-22 13:32:49 -050017
18namespace rx
19{
20
Jamie Madill5c6b7bf2015-08-17 12:53:35 -040021ProgramGL::ProgramGL(const gl::Program::Data &data,
22 const FunctionsGL *functions,
23 StateManagerGL *stateManager)
24 : ProgramImpl(data), mFunctions(functions), mStateManager(stateManager), mProgramID(0)
Geoff Langb1f435e2015-02-20 10:01:01 -050025{
26 ASSERT(mFunctions);
27 ASSERT(mStateManager);
Geoff Lang0ca53782015-05-07 13:49:39 -040028
29 mProgramID = mFunctions->createProgram();
Geoff Langb1f435e2015-02-20 10:01:01 -050030}
Geoff Langf9a6f082015-01-22 13:32:49 -050031
32ProgramGL::~ProgramGL()
Geoff Langb1f435e2015-02-20 10:01:01 -050033{
Geoff Lang0ca53782015-05-07 13:49:39 -040034 mFunctions->deleteProgram(mProgramID);
35 mProgramID = 0;
Geoff Langb1f435e2015-02-20 10:01:01 -050036}
Geoff Langf9a6f082015-01-22 13:32:49 -050037
Geoff Langf9a6f082015-01-22 13:32:49 -050038LinkResult ProgramGL::load(gl::InfoLog &infoLog, gl::BinaryInputStream *stream)
39{
Geoff Lang01306fc2015-10-05 16:53:10 +000040 UNIMPLEMENTED();
41 return LinkResult(false, gl::Error(GL_INVALID_OPERATION));
Geoff Langf9a6f082015-01-22 13:32:49 -050042}
43
44gl::Error ProgramGL::save(gl::BinaryOutputStream *stream)
45{
Geoff Lang01306fc2015-10-05 16:53:10 +000046 UNIMPLEMENTED();
47 return gl::Error(GL_INVALID_OPERATION);
Geoff Langf9a6f082015-01-22 13:32:49 -050048}
49
Geoff Langc5629752015-12-07 16:29:04 -050050void ProgramGL::setBinaryRetrievableHint(bool retrievable)
51{
52 UNIMPLEMENTED();
53}
54
Jamie Madillf5f4ad22015-09-02 18:32:38 +000055LinkResult ProgramGL::link(const gl::Data &data, gl::InfoLog &infoLog)
Geoff Langf9a6f082015-01-22 13:32:49 -050056{
Geoff Lang01306fc2015-10-05 16:53:10 +000057 // Reset the program state, delete the current program if one exists
58 reset();
Geoff Langb1f435e2015-02-20 10:01:01 -050059
Geoff Lang1a683462015-09-29 15:09:59 -040060 // Set the transform feedback state
61 std::vector<const GLchar *> transformFeedbackVaryings;
62 for (const auto &tfVarying : mData.getTransformFeedbackVaryingNames())
63 {
64 transformFeedbackVaryings.push_back(tfVarying.c_str());
65 }
66
67 if (transformFeedbackVaryings.empty())
68 {
69 if (mFunctions->transformFeedbackVaryings)
70 {
71 mFunctions->transformFeedbackVaryings(mProgramID, 0, nullptr,
72 mData.getTransformFeedbackBufferMode());
73 }
74 }
75 else
76 {
77 ASSERT(mFunctions->transformFeedbackVaryings);
78 mFunctions->transformFeedbackVaryings(
79 mProgramID, static_cast<GLsizei>(transformFeedbackVaryings.size()),
80 &transformFeedbackVaryings[0], mData.getTransformFeedbackBufferMode());
81 }
82
Geoff Lang01306fc2015-10-05 16:53:10 +000083 const gl::Shader *vertexShader = mData.getAttachedVertexShader();
84 const gl::Shader *fragmentShader = mData.getAttachedFragmentShader();
85
86 const ShaderGL *vertexShaderGL = GetImplAs<ShaderGL>(vertexShader);
87 const ShaderGL *fragmentShaderGL = GetImplAs<ShaderGL>(fragmentShader);
Geoff Langb1f435e2015-02-20 10:01:01 -050088
Geoff Langb1f435e2015-02-20 10:01:01 -050089 // Attach the shaders
90 mFunctions->attachShader(mProgramID, vertexShaderGL->getShaderID());
91 mFunctions->attachShader(mProgramID, fragmentShaderGL->getShaderID());
92
Geoff Lang1528e562015-08-24 15:10:58 -040093 // Bind attribute locations to match the GL layer.
94 for (const sh::Attribute &attribute : mData.getAttributes())
95 {
96 if (!attribute.staticUse)
97 {
98 continue;
99 }
100
101 mFunctions->bindAttribLocation(mProgramID, attribute.location, attribute.name.c_str());
Geoff Lang1528e562015-08-24 15:10:58 -0400102 }
103
Geoff Langb1f435e2015-02-20 10:01:01 -0500104 // Link and verify
105 mFunctions->linkProgram(mProgramID);
106
Geoff Lang0ca53782015-05-07 13:49:39 -0400107 // Detach the shaders
108 mFunctions->detachShader(mProgramID, vertexShaderGL->getShaderID());
109 mFunctions->detachShader(mProgramID, fragmentShaderGL->getShaderID());
110
111 // Verify the link
Geoff Lang01306fc2015-10-05 16:53:10 +0000112 GLint linkStatus = GL_FALSE;
113 mFunctions->getProgramiv(mProgramID, GL_LINK_STATUS, &linkStatus);
Geoff Lang01306fc2015-10-05 16:53:10 +0000114 if (linkStatus == GL_FALSE)
Geoff Langb1f435e2015-02-20 10:01:01 -0500115 {
Geoff Lang01306fc2015-10-05 16:53:10 +0000116 // Linking failed, put the error into the info log
117 GLint infoLogLength = 0;
118 mFunctions->getProgramiv(mProgramID, GL_INFO_LOG_LENGTH, &infoLogLength);
119
Corentin Wallez01ad6442016-03-03 13:53:45 -0500120 std::string warning;
Olli Etuaho67946762016-03-08 15:43:55 +0200121
122 // Info log length includes the null terminator, so 1 means that the info log is an empty
123 // string.
124 if (infoLogLength > 1)
Corentin Wallez01ad6442016-03-03 13:53:45 -0500125 {
126 std::vector<char> buf(infoLogLength);
127 mFunctions->getProgramInfoLog(mProgramID, infoLogLength, nullptr, &buf[0]);
Geoff Lang01306fc2015-10-05 16:53:10 +0000128
Corentin Wallez01ad6442016-03-03 13:53:45 -0500129 mFunctions->deleteProgram(mProgramID);
130 mProgramID = 0;
Geoff Lang01306fc2015-10-05 16:53:10 +0000131
Corentin Wallez01ad6442016-03-03 13:53:45 -0500132 infoLog << buf.data();
unknownb4a3af22015-11-25 15:02:51 -0500133
Corentin Wallez01ad6442016-03-03 13:53:45 -0500134 warning = FormatString("Program link failed unexpectedly: %s", buf.data());
135 }
136 else
137 {
138 warning = "Program link failed unexpectedly with no info log.";
139 }
unknownb4a3af22015-11-25 15:02:51 -0500140 ANGLEPlatformCurrent()->logWarning(warning.c_str());
141 TRACE("\n%s", warning.c_str());
Geoff Lang01306fc2015-10-05 16:53:10 +0000142
143 // TODO, return GL_OUT_OF_MEMORY or just fail the link? This is an unexpected case
Geoff Langb1f435e2015-02-20 10:01:01 -0500144 return LinkResult(false, gl::Error(GL_NO_ERROR));
145 }
146
Geoff Lang01306fc2015-10-05 16:53:10 +0000147 // Query the uniform information
148 ASSERT(mUniformRealLocationMap.empty());
149 const auto &uniforms = mData.getUniforms();
150 for (const gl::VariableLocation &entry : mData.getUniformLocations())
151 {
152 // From the spec:
153 // "Locations for sequential array indices are not required to be sequential."
154 const gl::LinkedUniform &uniform = uniforms[entry.index];
155 std::stringstream fullNameStr;
156 fullNameStr << uniform.name;
157 if (uniform.isArray())
158 {
159 fullNameStr << "[" << entry.element << "]";
160 }
161 const std::string &fullName = fullNameStr.str();
162
163 GLint realLocation = mFunctions->getUniformLocation(mProgramID, fullName.c_str());
164 mUniformRealLocationMap.push_back(realLocation);
165 }
166
167 mUniformIndexToSamplerIndex.resize(mData.getUniforms().size(), GL_INVALID_INDEX);
168
169 for (size_t uniformId = 0; uniformId < uniforms.size(); ++uniformId)
170 {
171 const gl::LinkedUniform &linkedUniform = uniforms[uniformId];
172
173 if (!linkedUniform.isSampler() || !linkedUniform.staticUse)
174 continue;
175
176 mUniformIndexToSamplerIndex[uniformId] = mSamplerBindings.size();
177
178 // If uniform is a sampler type, insert it into the mSamplerBindings array
179 SamplerBindingGL samplerBinding;
180 samplerBinding.textureType = gl::SamplerTypeToTextureType(linkedUniform.type);
181 samplerBinding.boundTextureUnits.resize(linkedUniform.elementCount(), 0);
182 mSamplerBindings.push_back(samplerBinding);
183 }
Geoff Langb1f435e2015-02-20 10:01:01 -0500184
Geoff Langb1f435e2015-02-20 10:01:01 -0500185 return LinkResult(true, gl::Error(GL_NO_ERROR));
Geoff Langf9a6f082015-01-22 13:32:49 -0500186}
187
Jamie Madill36cfd6a2015-08-18 10:46:20 -0400188GLboolean ProgramGL::validate(const gl::Caps & /*caps*/, gl::InfoLog * /*infoLog*/)
189{
190 // TODO(jmadill): implement validate
191 return true;
192}
193
Geoff Langf9a6f082015-01-22 13:32:49 -0500194void ProgramGL::setUniform1fv(GLint location, GLsizei count, const GLfloat *v)
195{
Geoff Lang63cbace2015-02-26 10:03:12 -0500196 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400197 mFunctions->uniform1fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500198}
199
200void ProgramGL::setUniform2fv(GLint location, GLsizei count, const GLfloat *v)
201{
Geoff Lang63cbace2015-02-26 10:03:12 -0500202 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400203 mFunctions->uniform2fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500204}
205
206void ProgramGL::setUniform3fv(GLint location, GLsizei count, const GLfloat *v)
207{
Geoff Lang63cbace2015-02-26 10:03:12 -0500208 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400209 mFunctions->uniform3fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500210}
211
212void ProgramGL::setUniform4fv(GLint location, GLsizei count, const GLfloat *v)
213{
Geoff Lang63cbace2015-02-26 10:03:12 -0500214 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400215 mFunctions->uniform4fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500216}
217
218void ProgramGL::setUniform1iv(GLint location, GLsizei count, const GLint *v)
219{
Geoff Lang63cbace2015-02-26 10:03:12 -0500220 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400221 mFunctions->uniform1iv(uniLoc(location), count, v);
Geoff Langf51bc792015-05-04 14:57:03 -0400222
Jamie Madill62d31cb2015-09-11 13:25:51 -0400223 const gl::VariableLocation &locationEntry = mData.getUniformLocations()[location];
224
225 size_t samplerIndex = mUniformIndexToSamplerIndex[locationEntry.index];
226 if (samplerIndex != GL_INVALID_INDEX)
Geoff Langf51bc792015-05-04 14:57:03 -0400227 {
Jamie Madill62d31cb2015-09-11 13:25:51 -0400228 std::vector<GLuint> &boundTextureUnits = mSamplerBindings[samplerIndex].boundTextureUnits;
Geoff Langf51bc792015-05-04 14:57:03 -0400229
Jamie Madill62d31cb2015-09-11 13:25:51 -0400230 size_t copyCount =
231 std::max<size_t>(count, boundTextureUnits.size() - locationEntry.element);
232 std::copy(v, v + copyCount, boundTextureUnits.begin() + locationEntry.element);
Geoff Langf51bc792015-05-04 14:57:03 -0400233 }
Geoff Langf9a6f082015-01-22 13:32:49 -0500234}
235
236void ProgramGL::setUniform2iv(GLint location, GLsizei count, const GLint *v)
237{
Geoff Lang63cbace2015-02-26 10:03:12 -0500238 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400239 mFunctions->uniform2iv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500240}
241
242void ProgramGL::setUniform3iv(GLint location, GLsizei count, const GLint *v)
243{
Geoff Lang63cbace2015-02-26 10:03:12 -0500244 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400245 mFunctions->uniform3iv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500246}
247
248void ProgramGL::setUniform4iv(GLint location, GLsizei count, const GLint *v)
249{
Geoff Lang63cbace2015-02-26 10:03:12 -0500250 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400251 mFunctions->uniform4iv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500252}
253
254void ProgramGL::setUniform1uiv(GLint location, GLsizei count, const GLuint *v)
255{
Geoff Lang63cbace2015-02-26 10:03:12 -0500256 mStateManager->useProgram(mProgramID);
Geoff Langb1f435e2015-02-20 10:01:01 -0500257 mFunctions->uniform1uiv(location, count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500258}
259
260void ProgramGL::setUniform2uiv(GLint location, GLsizei count, const GLuint *v)
261{
Geoff Lang63cbace2015-02-26 10:03:12 -0500262 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400263 mFunctions->uniform2uiv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500264}
265
266void ProgramGL::setUniform3uiv(GLint location, GLsizei count, const GLuint *v)
267{
Geoff Lang63cbace2015-02-26 10:03:12 -0500268 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400269 mFunctions->uniform3uiv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500270}
271
272void ProgramGL::setUniform4uiv(GLint location, GLsizei count, const GLuint *v)
273{
Geoff Lang63cbace2015-02-26 10:03:12 -0500274 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400275 mFunctions->uniform4uiv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500276}
277
278void ProgramGL::setUniformMatrix2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
279{
Geoff Lang63cbace2015-02-26 10:03:12 -0500280 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400281 mFunctions->uniformMatrix2fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500282}
283
284void ProgramGL::setUniformMatrix3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
285{
Geoff Lang63cbace2015-02-26 10:03:12 -0500286 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400287 mFunctions->uniformMatrix3fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500288}
289
290void ProgramGL::setUniformMatrix4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
291{
Geoff Lang63cbace2015-02-26 10:03:12 -0500292 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400293 mFunctions->uniformMatrix4fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500294}
295
296void ProgramGL::setUniformMatrix2x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
297{
Geoff Lang63cbace2015-02-26 10:03:12 -0500298 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400299 mFunctions->uniformMatrix2x3fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500300}
301
302void ProgramGL::setUniformMatrix3x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
303{
Geoff Lang63cbace2015-02-26 10:03:12 -0500304 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400305 mFunctions->uniformMatrix3x2fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500306}
307
308void ProgramGL::setUniformMatrix2x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
309{
Geoff Lang63cbace2015-02-26 10:03:12 -0500310 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400311 mFunctions->uniformMatrix2x4fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500312}
313
314void ProgramGL::setUniformMatrix4x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
315{
Geoff Lang63cbace2015-02-26 10:03:12 -0500316 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400317 mFunctions->uniformMatrix4x2fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500318}
319
320void ProgramGL::setUniformMatrix3x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
321{
Geoff Lang63cbace2015-02-26 10:03:12 -0500322 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400323 mFunctions->uniformMatrix3x4fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500324}
325
326void ProgramGL::setUniformMatrix4x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
327{
Geoff Lang63cbace2015-02-26 10:03:12 -0500328 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400329 mFunctions->uniformMatrix4x3fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500330}
331
Geoff Lang5d124a62015-09-15 13:03:27 -0400332void ProgramGL::setUniformBlockBinding(GLuint uniformBlockIndex, GLuint uniformBlockBinding)
333{
Jamie Madill4a3c2342015-10-08 12:58:45 -0400334 // Lazy init
335 if (mUniformBlockRealLocationMap.empty())
336 {
337 mUniformBlockRealLocationMap.reserve(mData.getUniformBlocks().size());
338 for (const gl::UniformBlock &uniformBlock : mData.getUniformBlocks())
339 {
340 const std::string &nameWithIndex = uniformBlock.nameWithArrayIndex();
341 GLuint blockIndex = mFunctions->getUniformBlockIndex(mProgramID, nameWithIndex.c_str());
342 mUniformBlockRealLocationMap.push_back(blockIndex);
343 }
344 }
345
346 GLuint realBlockIndex = mUniformBlockRealLocationMap[uniformBlockIndex];
347 if (realBlockIndex != GL_INVALID_INDEX)
348 {
349 mFunctions->uniformBlockBinding(mProgramID, realBlockIndex, uniformBlockBinding);
350 }
Geoff Lang5d124a62015-09-15 13:03:27 -0400351}
352
Geoff Lang01306fc2015-10-05 16:53:10 +0000353void ProgramGL::reset()
354{
355 mUniformRealLocationMap.clear();
Jamie Madill4a3c2342015-10-08 12:58:45 -0400356 mUniformBlockRealLocationMap.clear();
Geoff Lang01306fc2015-10-05 16:53:10 +0000357 mSamplerBindings.clear();
358 mUniformIndexToSamplerIndex.clear();
359}
360
Geoff Langb1f435e2015-02-20 10:01:01 -0500361GLuint ProgramGL::getProgramID() const
362{
363 return mProgramID;
364}
365
Geoff Langf51bc792015-05-04 14:57:03 -0400366const std::vector<SamplerBindingGL> &ProgramGL::getAppliedSamplerUniforms() const
367{
368 return mSamplerBindings;
369}
370
Jamie Madill4a3c2342015-10-08 12:58:45 -0400371bool ProgramGL::getUniformBlockSize(const std::string &blockName, size_t *sizeOut) const
Jamie Madill62d31cb2015-09-11 13:25:51 -0400372{
Jamie Madill4a3c2342015-10-08 12:58:45 -0400373 ASSERT(mProgramID != 0u);
Geoff Lang5d124a62015-09-15 13:03:27 -0400374
Jamie Madill4a3c2342015-10-08 12:58:45 -0400375 GLuint blockIndex = mFunctions->getUniformBlockIndex(mProgramID, blockName.c_str());
376 if (blockIndex == GL_INVALID_INDEX)
Geoff Lang5d124a62015-09-15 13:03:27 -0400377 {
Jamie Madill4a3c2342015-10-08 12:58:45 -0400378 *sizeOut = 0;
379 return false;
Geoff Lang5d124a62015-09-15 13:03:27 -0400380 }
381
Jamie Madill4a3c2342015-10-08 12:58:45 -0400382 GLint dataSize = 0;
383 mFunctions->getActiveUniformBlockiv(mProgramID, blockIndex, GL_UNIFORM_BLOCK_DATA_SIZE,
384 &dataSize);
385 *sizeOut = static_cast<size_t>(dataSize);
386 return true;
387}
388
389bool ProgramGL::getUniformBlockMemberInfo(const std::string &memberUniformName,
390 sh::BlockMemberInfo *memberInfoOut) const
391{
392 GLuint uniformIndex;
393 const GLchar *memberNameGLStr = memberUniformName.c_str();
394 mFunctions->getUniformIndices(mProgramID, 1, &memberNameGLStr, &uniformIndex);
395
396 if (uniformIndex == GL_INVALID_INDEX)
Geoff Lang5d124a62015-09-15 13:03:27 -0400397 {
Jamie Madill4a3c2342015-10-08 12:58:45 -0400398 *memberInfoOut = sh::BlockMemberInfo::getDefaultBlockInfo();
399 return false;
Geoff Lang5d124a62015-09-15 13:03:27 -0400400 }
Jamie Madill4a3c2342015-10-08 12:58:45 -0400401
402 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_OFFSET,
403 &memberInfoOut->offset);
404 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_ARRAY_STRIDE,
405 &memberInfoOut->arrayStride);
406 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_MATRIX_STRIDE,
407 &memberInfoOut->matrixStride);
408
409 // TODO(jmadill): possibly determine this at the gl::Program level.
410 GLint isRowMajorMatrix = 0;
411 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_IS_ROW_MAJOR,
412 &isRowMajorMatrix);
413 memberInfoOut->isRowMajorMatrix = isRowMajorMatrix != GL_FALSE;
414 return true;
Jamie Madill62d31cb2015-09-11 13:25:51 -0400415}
Jamie Madill4a3c2342015-10-08 12:58:45 -0400416
417} // namespace rx