blob: 80d8c92be6395cb3fbdfe95016176c2c37b8fd2a [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"
Geoff Langf9a6f082015-01-22 13:32:49 -050016
17namespace rx
18{
19
Jamie Madill5c6b7bf2015-08-17 12:53:35 -040020ProgramGL::ProgramGL(const gl::Program::Data &data,
21 const FunctionsGL *functions,
22 StateManagerGL *stateManager)
23 : ProgramImpl(data), mFunctions(functions), mStateManager(stateManager), mProgramID(0)
Geoff Langb1f435e2015-02-20 10:01:01 -050024{
25 ASSERT(mFunctions);
26 ASSERT(mStateManager);
Geoff Lang0ca53782015-05-07 13:49:39 -040027
28 mProgramID = mFunctions->createProgram();
Geoff Langb1f435e2015-02-20 10:01:01 -050029}
Geoff Langf9a6f082015-01-22 13:32:49 -050030
31ProgramGL::~ProgramGL()
Geoff Langb1f435e2015-02-20 10:01:01 -050032{
Geoff Lang0ca53782015-05-07 13:49:39 -040033 mFunctions->deleteProgram(mProgramID);
34 mProgramID = 0;
Geoff Langb1f435e2015-02-20 10:01:01 -050035}
Geoff Langf9a6f082015-01-22 13:32:49 -050036
Geoff Langf9a6f082015-01-22 13:32:49 -050037LinkResult ProgramGL::load(gl::InfoLog &infoLog, gl::BinaryInputStream *stream)
38{
Geoff Lang01306fc2015-10-05 16:53:10 +000039 UNIMPLEMENTED();
40 return LinkResult(false, gl::Error(GL_INVALID_OPERATION));
Geoff Langf9a6f082015-01-22 13:32:49 -050041}
42
43gl::Error ProgramGL::save(gl::BinaryOutputStream *stream)
44{
Geoff Lang01306fc2015-10-05 16:53:10 +000045 UNIMPLEMENTED();
46 return gl::Error(GL_INVALID_OPERATION);
Geoff Langf9a6f082015-01-22 13:32:49 -050047}
48
Jamie Madillf5f4ad22015-09-02 18:32:38 +000049LinkResult ProgramGL::link(const gl::Data &data, gl::InfoLog &infoLog)
Geoff Langf9a6f082015-01-22 13:32:49 -050050{
Geoff Lang01306fc2015-10-05 16:53:10 +000051 // Reset the program state, delete the current program if one exists
52 reset();
Geoff Langb1f435e2015-02-20 10:01:01 -050053
Geoff Lang1a683462015-09-29 15:09:59 -040054 // Set the transform feedback state
55 std::vector<const GLchar *> transformFeedbackVaryings;
56 for (const auto &tfVarying : mData.getTransformFeedbackVaryingNames())
57 {
58 transformFeedbackVaryings.push_back(tfVarying.c_str());
59 }
60
61 if (transformFeedbackVaryings.empty())
62 {
63 if (mFunctions->transformFeedbackVaryings)
64 {
65 mFunctions->transformFeedbackVaryings(mProgramID, 0, nullptr,
66 mData.getTransformFeedbackBufferMode());
67 }
68 }
69 else
70 {
71 ASSERT(mFunctions->transformFeedbackVaryings);
72 mFunctions->transformFeedbackVaryings(
73 mProgramID, static_cast<GLsizei>(transformFeedbackVaryings.size()),
74 &transformFeedbackVaryings[0], mData.getTransformFeedbackBufferMode());
75 }
76
Geoff Lang01306fc2015-10-05 16:53:10 +000077 const gl::Shader *vertexShader = mData.getAttachedVertexShader();
78 const gl::Shader *fragmentShader = mData.getAttachedFragmentShader();
79
80 const ShaderGL *vertexShaderGL = GetImplAs<ShaderGL>(vertexShader);
81 const ShaderGL *fragmentShaderGL = GetImplAs<ShaderGL>(fragmentShader);
Geoff Langb1f435e2015-02-20 10:01:01 -050082
Geoff Langb1f435e2015-02-20 10:01:01 -050083 // Attach the shaders
84 mFunctions->attachShader(mProgramID, vertexShaderGL->getShaderID());
85 mFunctions->attachShader(mProgramID, fragmentShaderGL->getShaderID());
86
Geoff Lang1528e562015-08-24 15:10:58 -040087 // Bind attribute locations to match the GL layer.
88 for (const sh::Attribute &attribute : mData.getAttributes())
89 {
90 if (!attribute.staticUse)
91 {
92 continue;
93 }
94
95 mFunctions->bindAttribLocation(mProgramID, attribute.location, attribute.name.c_str());
Geoff Lang1528e562015-08-24 15:10:58 -040096 }
97
Geoff Langb1f435e2015-02-20 10:01:01 -050098 // Link and verify
99 mFunctions->linkProgram(mProgramID);
100
Geoff Lang0ca53782015-05-07 13:49:39 -0400101 // Detach the shaders
102 mFunctions->detachShader(mProgramID, vertexShaderGL->getShaderID());
103 mFunctions->detachShader(mProgramID, fragmentShaderGL->getShaderID());
104
105 // Verify the link
Geoff Lang01306fc2015-10-05 16:53:10 +0000106 GLint linkStatus = GL_FALSE;
107 mFunctions->getProgramiv(mProgramID, GL_LINK_STATUS, &linkStatus);
108 ASSERT(linkStatus == GL_TRUE);
109 if (linkStatus == GL_FALSE)
Geoff Langb1f435e2015-02-20 10:01:01 -0500110 {
Geoff Lang01306fc2015-10-05 16:53:10 +0000111 // Linking failed, put the error into the info log
112 GLint infoLogLength = 0;
113 mFunctions->getProgramiv(mProgramID, GL_INFO_LOG_LENGTH, &infoLogLength);
114
115 std::vector<char> buf(infoLogLength);
116 mFunctions->getProgramInfoLog(mProgramID, infoLogLength, nullptr, &buf[0]);
117
118 mFunctions->deleteProgram(mProgramID);
119 mProgramID = 0;
120
121 infoLog << &buf[0];
122 TRACE("\n%s", &buf[0]);
123
124 // TODO, return GL_OUT_OF_MEMORY or just fail the link? This is an unexpected case
Geoff Langb1f435e2015-02-20 10:01:01 -0500125 return LinkResult(false, gl::Error(GL_NO_ERROR));
126 }
127
Geoff Lang01306fc2015-10-05 16:53:10 +0000128 // Query the uniform information
129 ASSERT(mUniformRealLocationMap.empty());
130 const auto &uniforms = mData.getUniforms();
131 for (const gl::VariableLocation &entry : mData.getUniformLocations())
132 {
133 // From the spec:
134 // "Locations for sequential array indices are not required to be sequential."
135 const gl::LinkedUniform &uniform = uniforms[entry.index];
136 std::stringstream fullNameStr;
137 fullNameStr << uniform.name;
138 if (uniform.isArray())
139 {
140 fullNameStr << "[" << entry.element << "]";
141 }
142 const std::string &fullName = fullNameStr.str();
143
144 GLint realLocation = mFunctions->getUniformLocation(mProgramID, fullName.c_str());
145 mUniformRealLocationMap.push_back(realLocation);
146 }
147
148 mUniformIndexToSamplerIndex.resize(mData.getUniforms().size(), GL_INVALID_INDEX);
149
150 for (size_t uniformId = 0; uniformId < uniforms.size(); ++uniformId)
151 {
152 const gl::LinkedUniform &linkedUniform = uniforms[uniformId];
153
154 if (!linkedUniform.isSampler() || !linkedUniform.staticUse)
155 continue;
156
157 mUniformIndexToSamplerIndex[uniformId] = mSamplerBindings.size();
158
159 // If uniform is a sampler type, insert it into the mSamplerBindings array
160 SamplerBindingGL samplerBinding;
161 samplerBinding.textureType = gl::SamplerTypeToTextureType(linkedUniform.type);
162 samplerBinding.boundTextureUnits.resize(linkedUniform.elementCount(), 0);
163 mSamplerBindings.push_back(samplerBinding);
164 }
Geoff Langb1f435e2015-02-20 10:01:01 -0500165
Geoff Langb1f435e2015-02-20 10:01:01 -0500166 return LinkResult(true, gl::Error(GL_NO_ERROR));
Geoff Langf9a6f082015-01-22 13:32:49 -0500167}
168
Jamie Madill36cfd6a2015-08-18 10:46:20 -0400169GLboolean ProgramGL::validate(const gl::Caps & /*caps*/, gl::InfoLog * /*infoLog*/)
170{
171 // TODO(jmadill): implement validate
172 return true;
173}
174
Geoff Langf9a6f082015-01-22 13:32:49 -0500175void ProgramGL::setUniform1fv(GLint location, GLsizei count, const GLfloat *v)
176{
Geoff Lang63cbace2015-02-26 10:03:12 -0500177 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400178 mFunctions->uniform1fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500179}
180
181void ProgramGL::setUniform2fv(GLint location, GLsizei count, const GLfloat *v)
182{
Geoff Lang63cbace2015-02-26 10:03:12 -0500183 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400184 mFunctions->uniform2fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500185}
186
187void ProgramGL::setUniform3fv(GLint location, GLsizei count, const GLfloat *v)
188{
Geoff Lang63cbace2015-02-26 10:03:12 -0500189 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400190 mFunctions->uniform3fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500191}
192
193void ProgramGL::setUniform4fv(GLint location, GLsizei count, const GLfloat *v)
194{
Geoff Lang63cbace2015-02-26 10:03:12 -0500195 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400196 mFunctions->uniform4fv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500197}
198
199void ProgramGL::setUniform1iv(GLint location, GLsizei count, const GLint *v)
200{
Geoff Lang63cbace2015-02-26 10:03:12 -0500201 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400202 mFunctions->uniform1iv(uniLoc(location), count, v);
Geoff Langf51bc792015-05-04 14:57:03 -0400203
Jamie Madill62d31cb2015-09-11 13:25:51 -0400204 const gl::VariableLocation &locationEntry = mData.getUniformLocations()[location];
205
206 size_t samplerIndex = mUniformIndexToSamplerIndex[locationEntry.index];
207 if (samplerIndex != GL_INVALID_INDEX)
Geoff Langf51bc792015-05-04 14:57:03 -0400208 {
Jamie Madill62d31cb2015-09-11 13:25:51 -0400209 std::vector<GLuint> &boundTextureUnits = mSamplerBindings[samplerIndex].boundTextureUnits;
Geoff Langf51bc792015-05-04 14:57:03 -0400210
Jamie Madill62d31cb2015-09-11 13:25:51 -0400211 size_t copyCount =
212 std::max<size_t>(count, boundTextureUnits.size() - locationEntry.element);
213 std::copy(v, v + copyCount, boundTextureUnits.begin() + locationEntry.element);
Geoff Langf51bc792015-05-04 14:57:03 -0400214 }
Geoff Langf9a6f082015-01-22 13:32:49 -0500215}
216
217void ProgramGL::setUniform2iv(GLint location, GLsizei count, const GLint *v)
218{
Geoff Lang63cbace2015-02-26 10:03:12 -0500219 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400220 mFunctions->uniform2iv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500221}
222
223void ProgramGL::setUniform3iv(GLint location, GLsizei count, const GLint *v)
224{
Geoff Lang63cbace2015-02-26 10:03:12 -0500225 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400226 mFunctions->uniform3iv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500227}
228
229void ProgramGL::setUniform4iv(GLint location, GLsizei count, const GLint *v)
230{
Geoff Lang63cbace2015-02-26 10:03:12 -0500231 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400232 mFunctions->uniform4iv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500233}
234
235void ProgramGL::setUniform1uiv(GLint location, GLsizei count, const GLuint *v)
236{
Geoff Lang63cbace2015-02-26 10:03:12 -0500237 mStateManager->useProgram(mProgramID);
Geoff Langb1f435e2015-02-20 10:01:01 -0500238 mFunctions->uniform1uiv(location, count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500239}
240
241void ProgramGL::setUniform2uiv(GLint location, GLsizei count, const GLuint *v)
242{
Geoff Lang63cbace2015-02-26 10:03:12 -0500243 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400244 mFunctions->uniform2uiv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500245}
246
247void ProgramGL::setUniform3uiv(GLint location, GLsizei count, const GLuint *v)
248{
Geoff Lang63cbace2015-02-26 10:03:12 -0500249 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400250 mFunctions->uniform3uiv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500251}
252
253void ProgramGL::setUniform4uiv(GLint location, GLsizei count, const GLuint *v)
254{
Geoff Lang63cbace2015-02-26 10:03:12 -0500255 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400256 mFunctions->uniform4uiv(uniLoc(location), count, v);
Geoff Langf9a6f082015-01-22 13:32:49 -0500257}
258
259void ProgramGL::setUniformMatrix2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
260{
Geoff Lang63cbace2015-02-26 10:03:12 -0500261 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400262 mFunctions->uniformMatrix2fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500263}
264
265void ProgramGL::setUniformMatrix3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
266{
Geoff Lang63cbace2015-02-26 10:03:12 -0500267 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400268 mFunctions->uniformMatrix3fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500269}
270
271void ProgramGL::setUniformMatrix4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
272{
Geoff Lang63cbace2015-02-26 10:03:12 -0500273 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400274 mFunctions->uniformMatrix4fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500275}
276
277void ProgramGL::setUniformMatrix2x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
278{
Geoff Lang63cbace2015-02-26 10:03:12 -0500279 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400280 mFunctions->uniformMatrix2x3fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500281}
282
283void ProgramGL::setUniformMatrix3x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
284{
Geoff Lang63cbace2015-02-26 10:03:12 -0500285 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400286 mFunctions->uniformMatrix3x2fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500287}
288
289void ProgramGL::setUniformMatrix2x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
290{
Geoff Lang63cbace2015-02-26 10:03:12 -0500291 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400292 mFunctions->uniformMatrix2x4fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500293}
294
295void ProgramGL::setUniformMatrix4x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
296{
Geoff Lang63cbace2015-02-26 10:03:12 -0500297 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400298 mFunctions->uniformMatrix4x2fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500299}
300
301void ProgramGL::setUniformMatrix3x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
302{
Geoff Lang63cbace2015-02-26 10:03:12 -0500303 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400304 mFunctions->uniformMatrix3x4fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500305}
306
307void ProgramGL::setUniformMatrix4x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value)
308{
Geoff Lang63cbace2015-02-26 10:03:12 -0500309 mStateManager->useProgram(mProgramID);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400310 mFunctions->uniformMatrix4x3fv(uniLoc(location), count, transpose, value);
Geoff Langf9a6f082015-01-22 13:32:49 -0500311}
312
Geoff Lang5d124a62015-09-15 13:03:27 -0400313void ProgramGL::setUniformBlockBinding(GLuint uniformBlockIndex, GLuint uniformBlockBinding)
314{
Jamie Madill4a3c2342015-10-08 12:58:45 -0400315 // Lazy init
316 if (mUniformBlockRealLocationMap.empty())
317 {
318 mUniformBlockRealLocationMap.reserve(mData.getUniformBlocks().size());
319 for (const gl::UniformBlock &uniformBlock : mData.getUniformBlocks())
320 {
321 const std::string &nameWithIndex = uniformBlock.nameWithArrayIndex();
322 GLuint blockIndex = mFunctions->getUniformBlockIndex(mProgramID, nameWithIndex.c_str());
323 mUniformBlockRealLocationMap.push_back(blockIndex);
324 }
325 }
326
327 GLuint realBlockIndex = mUniformBlockRealLocationMap[uniformBlockIndex];
328 if (realBlockIndex != GL_INVALID_INDEX)
329 {
330 mFunctions->uniformBlockBinding(mProgramID, realBlockIndex, uniformBlockBinding);
331 }
Geoff Lang5d124a62015-09-15 13:03:27 -0400332}
333
Geoff Lang01306fc2015-10-05 16:53:10 +0000334void ProgramGL::reset()
335{
336 mUniformRealLocationMap.clear();
Jamie Madill4a3c2342015-10-08 12:58:45 -0400337 mUniformBlockRealLocationMap.clear();
Geoff Lang01306fc2015-10-05 16:53:10 +0000338 mSamplerBindings.clear();
339 mUniformIndexToSamplerIndex.clear();
340}
341
Geoff Langb1f435e2015-02-20 10:01:01 -0500342GLuint ProgramGL::getProgramID() const
343{
344 return mProgramID;
345}
346
Geoff Langf51bc792015-05-04 14:57:03 -0400347const std::vector<SamplerBindingGL> &ProgramGL::getAppliedSamplerUniforms() const
348{
349 return mSamplerBindings;
350}
351
Jamie Madill4a3c2342015-10-08 12:58:45 -0400352bool ProgramGL::getUniformBlockSize(const std::string &blockName, size_t *sizeOut) const
Jamie Madill62d31cb2015-09-11 13:25:51 -0400353{
Jamie Madill4a3c2342015-10-08 12:58:45 -0400354 ASSERT(mProgramID != 0u);
Geoff Lang5d124a62015-09-15 13:03:27 -0400355
Jamie Madill4a3c2342015-10-08 12:58:45 -0400356 GLuint blockIndex = mFunctions->getUniformBlockIndex(mProgramID, blockName.c_str());
357 if (blockIndex == GL_INVALID_INDEX)
Geoff Lang5d124a62015-09-15 13:03:27 -0400358 {
Jamie Madill4a3c2342015-10-08 12:58:45 -0400359 *sizeOut = 0;
360 return false;
Geoff Lang5d124a62015-09-15 13:03:27 -0400361 }
362
Jamie Madill4a3c2342015-10-08 12:58:45 -0400363 GLint dataSize = 0;
364 mFunctions->getActiveUniformBlockiv(mProgramID, blockIndex, GL_UNIFORM_BLOCK_DATA_SIZE,
365 &dataSize);
366 *sizeOut = static_cast<size_t>(dataSize);
367 return true;
368}
369
370bool ProgramGL::getUniformBlockMemberInfo(const std::string &memberUniformName,
371 sh::BlockMemberInfo *memberInfoOut) const
372{
373 GLuint uniformIndex;
374 const GLchar *memberNameGLStr = memberUniformName.c_str();
375 mFunctions->getUniformIndices(mProgramID, 1, &memberNameGLStr, &uniformIndex);
376
377 if (uniformIndex == GL_INVALID_INDEX)
Geoff Lang5d124a62015-09-15 13:03:27 -0400378 {
Jamie Madill4a3c2342015-10-08 12:58:45 -0400379 *memberInfoOut = sh::BlockMemberInfo::getDefaultBlockInfo();
380 return false;
Geoff Lang5d124a62015-09-15 13:03:27 -0400381 }
Jamie Madill4a3c2342015-10-08 12:58:45 -0400382
383 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_OFFSET,
384 &memberInfoOut->offset);
385 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_ARRAY_STRIDE,
386 &memberInfoOut->arrayStride);
387 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_MATRIX_STRIDE,
388 &memberInfoOut->matrixStride);
389
390 // TODO(jmadill): possibly determine this at the gl::Program level.
391 GLint isRowMajorMatrix = 0;
392 mFunctions->getActiveUniformsiv(mProgramID, 1, &uniformIndex, GL_UNIFORM_IS_ROW_MAJOR,
393 &isRowMajorMatrix);
394 memberInfoOut->isRowMajorMatrix = isRowMajorMatrix != GL_FALSE;
395 return true;
Jamie Madill62d31cb2015-09-11 13:25:51 -0400396}
Jamie Madill4a3c2342015-10-08 12:58:45 -0400397
398} // namespace rx