blob: de94bdb414763ebc60e5e1795d035e56d89492a4 [file] [log] [blame]
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00001//
Geoff Lang1b6edcb2014-02-03 14:27:56 -05002// Copyright (c) 2002-2014 The ANGLE Project Authors. All rights reserved.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
7// Program.h: Defines the gl::Program class. Implements GL program objects
8// and related functionality. [OpenGL ES 2.0.24] section 2.10.3 page 28.
9
Geoff Lang0a73dd82014-11-19 16:18:08 -050010#ifndef LIBANGLE_PROGRAM_H_
11#define LIBANGLE_PROGRAM_H_
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000012
Geoff Lang7dd2e102014-11-10 15:19:26 -050013#include "libANGLE/angletypes.h"
Geoff Lang21329412014-12-02 20:50:30 +000014#include "libANGLE/Constants.h"
Geoff Lang7dd2e102014-11-10 15:19:26 -050015#include "libANGLE/Error.h"
Geoff Lang7dd2e102014-11-10 15:19:26 -050016#include "libANGLE/RefCountObject.h"
17
18#include "common/angleutils.h"
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000019
Geoff Lang0b7eef72014-06-12 14:10:47 -040020#include <GLES2/gl2.h>
Geoff Lang7dd2e102014-11-10 15:19:26 -050021#include <GLSLANG/ShaderLang.h>
Geoff Lang0b7eef72014-06-12 14:10:47 -040022
Jamie Madill63805b42015-08-25 13:17:39 -040023#include <set>
Jamie Madill71c3b2c2015-05-07 11:49:20 -040024#include <sstream>
Geoff Lang0b7eef72014-06-12 14:10:47 -040025#include <string>
Jamie Madill63805b42015-08-25 13:17:39 -040026#include <vector>
Geoff Lang0b7eef72014-06-12 14:10:47 -040027
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000028namespace rx
29{
Jamie Madill5c6b7bf2015-08-17 12:53:35 -040030class ImplFactory;
Geoff Lang7dd2e102014-11-10 15:19:26 -050031class ProgramImpl;
Jamie Madill5c6b7bf2015-08-17 12:53:35 -040032struct TranslatedAttribute;
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000033}
daniel@transgaming.come6842292010-04-20 18:52:50 +000034
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000035namespace gl
36{
Brandon Jones43a53e22014-08-28 16:23:22 -070037struct Caps;
Jamie Madillde8892b2014-11-11 13:00:22 -050038struct Data;
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +000039class ResourceManager;
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000040class Shader;
Geoff Lang7dd2e102014-11-10 15:19:26 -050041class InfoLog;
42class AttributeBindings;
43class Buffer;
44class Framebuffer;
45struct UniformBlock;
46struct LinkedUniform;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000047
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000048extern const char * const g_fakepath;
49
apatrick@chromium.org9a30b092012-06-06 20:21:55 +000050class AttributeBindings
51{
52 public:
53 AttributeBindings();
54 ~AttributeBindings();
55
56 void bindAttributeLocation(GLuint index, const char *name);
57 int getAttributeBinding(const std::string &name) const;
58
59 private:
60 std::set<std::string> mAttributeBinding[MAX_VERTEX_ATTRIBS];
61};
62
Jamie Madillf0d10f82015-03-31 12:56:52 -040063class InfoLog : angle::NonCopyable
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000064{
65 public:
66 InfoLog();
67 ~InfoLog();
68
Jamie Madill71c3b2c2015-05-07 11:49:20 -040069 size_t getLength() const;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000070 void getLog(GLsizei bufSize, GLsizei *length, char *infoLog);
71
72 void appendSanitized(const char *message);
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000073 void reset();
Jamie Madillf6113162015-05-07 11:49:21 -040074
75 // This helper class ensures we append a newline after writing a line.
76 class StreamHelper : angle::NonCopyable
77 {
78 public:
79 StreamHelper(StreamHelper &&rhs)
80 : mStream(rhs.mStream)
81 {
82 rhs.mStream = nullptr;
83 }
84
85 StreamHelper &operator=(StreamHelper &&rhs)
86 {
87 std::swap(mStream, rhs.mStream);
88 return *this;
89 }
90
91 ~StreamHelper()
92 {
93 // Write newline when destroyed on the stack
94 if (mStream)
95 {
96 (*mStream) << std::endl;
97 }
98 }
99
100 template <typename T>
101 StreamHelper &operator<<(const T &value)
102 {
103 (*mStream) << value;
104 return *this;
105 }
106
107 private:
108 friend class InfoLog;
109
110 StreamHelper(std::stringstream *stream)
111 : mStream(stream)
112 {
113 ASSERT(stream);
114 }
115
116 std::stringstream *mStream;
117 };
118
119 template <typename T>
120 StreamHelper operator<<(const T &value)
121 {
122 StreamHelper helper(&mStream);
123 helper << value;
124 return helper;
125 }
126
127 std::string str() const { return mStream.str(); }
128
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000129 private:
Jamie Madill71c3b2c2015-05-07 11:49:20 -0400130 std::stringstream mStream;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000131};
132
Geoff Lang7dd2e102014-11-10 15:19:26 -0500133// Struct used for correlating uniforms/elements of uniform arrays to handles
134struct VariableLocation
135{
136 VariableLocation();
137 VariableLocation(const std::string &name, unsigned int element, unsigned int index);
138
139 std::string name;
140 unsigned int element;
141 unsigned int index;
142};
143
144struct LinkedVarying
145{
146 LinkedVarying();
147 LinkedVarying(const std::string &name, GLenum type, GLsizei size, const std::string &semanticName,
148 unsigned int semanticIndex, unsigned int semanticIndexCount);
149
150 // Original GL name
151 std::string name;
152
153 GLenum type;
154 GLsizei size;
155
156 // DirectX semantic information
157 std::string semanticName;
158 unsigned int semanticIndex;
159 unsigned int semanticIndexCount;
160};
161
Jamie Madillf0d10f82015-03-31 12:56:52 -0400162class Program : angle::NonCopyable
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000163{
164 public:
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400165 class Data final : angle::NonCopyable
166 {
167 public:
168 Data();
169 ~Data();
170
171 const Shader *getAttachedVertexShader() const { return mAttachedVertexShader; }
172 const Shader *getAttachedFragmentShader() const { return mAttachedFragmentShader; }
Jamie Madillccdf74b2015-08-18 10:46:12 -0400173 const std::vector<std::string> &getTransformFeedbackVaryingNames() const
Jamie Madillada9ecc2015-08-17 12:53:37 -0400174 {
Jamie Madillccdf74b2015-08-18 10:46:12 -0400175 return mTransformFeedbackVaryingNames;
Jamie Madillada9ecc2015-08-17 12:53:37 -0400176 }
177 GLint getTransformFeedbackBufferMode() const { return mTransformFeedbackBufferMode; }
Jamie Madilld1fe1642015-08-21 16:26:04 -0400178 GLuint getUniformBlockBinding(GLuint uniformBlockIndex) const
179 {
180 ASSERT(uniformBlockIndex < IMPLEMENTATION_MAX_COMBINED_SHADER_UNIFORM_BUFFERS);
181 return mUniformBlockBindings[uniformBlockIndex];
182 }
Jamie Madillc349ec02015-08-21 16:53:12 -0400183 const std::vector<sh::Attribute> &getAttributes() const { return mAttributes; }
Jamie Madill63805b42015-08-25 13:17:39 -0400184 const AttributesMask &getActiveAttribLocationsMask() const
185 {
186 return mActiveAttribLocationsMask;
187 }
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400188
189 private:
190 friend class Program;
191
192 Shader *mAttachedFragmentShader;
193 Shader *mAttachedVertexShader;
194
Jamie Madillccdf74b2015-08-18 10:46:12 -0400195 std::vector<std::string> mTransformFeedbackVaryingNames;
196 std::vector<sh::Varying> mTransformFeedbackVaryingVars;
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400197 GLenum mTransformFeedbackBufferMode;
198
Jamie Madilld1fe1642015-08-21 16:26:04 -0400199 GLuint mUniformBlockBindings[IMPLEMENTATION_MAX_COMBINED_SHADER_UNIFORM_BUFFERS];
200
Jamie Madillc349ec02015-08-21 16:53:12 -0400201 std::vector<sh::Attribute> mAttributes;
Jamie Madill63805b42015-08-25 13:17:39 -0400202 std::bitset<MAX_VERTEX_ATTRIBS> mActiveAttribLocationsMask;
Jamie Madillc349ec02015-08-21 16:53:12 -0400203
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400204 // TODO(jmadill): move more state into Data.
205 };
206
207 Program(rx::ImplFactory *factory, ResourceManager *manager, GLuint handle);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000208 ~Program();
209
Geoff Lang7dd2e102014-11-10 15:19:26 -0500210 GLuint id() const { return mHandle; }
211
212 rx::ProgramImpl *getImplementation() { return mProgram; }
213 const rx::ProgramImpl *getImplementation() const { return mProgram; }
214
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000215 bool attachShader(Shader *shader);
216 bool detachShader(Shader *shader);
217 int getAttachedShadersCount() const;
218
219 void bindAttributeLocation(GLuint index, const char *name);
220
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400221 Error link(const gl::Data &data);
daniel@transgaming.com716056c2012-07-24 18:38:59 +0000222 bool isLinked();
Geoff Lang7dd2e102014-11-10 15:19:26 -0500223
224 Error loadBinary(GLenum binaryFormat, const void *binary, GLsizei length);
225 Error saveBinary(GLenum *binaryFormat, void *binary, GLsizei bufSize, GLsizei *length) const;
226 GLint getBinaryLength() const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000227
228 int getInfoLogLength() const;
229 void getInfoLog(GLsizei bufSize, GLsizei *length, char *infoLog);
230 void getAttachedShaders(GLsizei maxCount, GLsizei *count, GLuint *shaders);
231
Geoff Lang7dd2e102014-11-10 15:19:26 -0500232 GLuint getAttributeLocation(const std::string &name);
Jamie Madill63805b42015-08-25 13:17:39 -0400233 bool isAttribLocationActive(size_t attribLocation) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500234
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000235 void getActiveAttribute(GLuint index, GLsizei bufsize, GLsizei *length, GLint *size, GLenum *type, GLchar *name);
236 GLint getActiveAttributeCount();
237 GLint getActiveAttributeMaxLength();
Jamie Madillc349ec02015-08-21 16:53:12 -0400238 const std::vector<sh::Attribute> &getAttributes() const { return mData.mAttributes; }
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000239
Geoff Lang7dd2e102014-11-10 15:19:26 -0500240 GLint getFragDataLocation(const std::string &name) const;
241
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000242 void getActiveUniform(GLuint index, GLsizei bufsize, GLsizei *length, GLint *size, GLenum *type, GLchar *name);
243 GLint getActiveUniformCount();
244 GLint getActiveUniformMaxLength();
Geoff Lang7dd2e102014-11-10 15:19:26 -0500245 GLint getActiveUniformi(GLuint index, GLenum pname) const;
246 bool isValidUniformLocation(GLint location) const;
247 LinkedUniform *getUniformByLocation(GLint location) const;
248 LinkedUniform *getUniformByName(const std::string &name) const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000249
Geoff Lang7dd2e102014-11-10 15:19:26 -0500250 GLint getUniformLocation(const std::string &name);
251 GLuint getUniformIndex(const std::string &name);
252 void setUniform1fv(GLint location, GLsizei count, const GLfloat *v);
253 void setUniform2fv(GLint location, GLsizei count, const GLfloat *v);
254 void setUniform3fv(GLint location, GLsizei count, const GLfloat *v);
255 void setUniform4fv(GLint location, GLsizei count, const GLfloat *v);
256 void setUniform1iv(GLint location, GLsizei count, const GLint *v);
257 void setUniform2iv(GLint location, GLsizei count, const GLint *v);
258 void setUniform3iv(GLint location, GLsizei count, const GLint *v);
259 void setUniform4iv(GLint location, GLsizei count, const GLint *v);
260 void setUniform1uiv(GLint location, GLsizei count, const GLuint *v);
261 void setUniform2uiv(GLint location, GLsizei count, const GLuint *v);
262 void setUniform3uiv(GLint location, GLsizei count, const GLuint *v);
263 void setUniform4uiv(GLint location, GLsizei count, const GLuint *v);
264 void setUniformMatrix2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
265 void setUniformMatrix3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
266 void setUniformMatrix4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
267 void setUniformMatrix2x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
268 void setUniformMatrix3x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
269 void setUniformMatrix2x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
270 void setUniformMatrix4x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
271 void setUniformMatrix3x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
272 void setUniformMatrix4x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
273
274 void getUniformfv(GLint location, GLfloat *params);
275 void getUniformiv(GLint location, GLint *params);
276 void getUniformuiv(GLint location, GLuint *params);
277
Geoff Lang7dd2e102014-11-10 15:19:26 -0500278 void getActiveUniformBlockName(GLuint uniformBlockIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformBlockName) const;
279 void getActiveUniformBlockiv(GLuint uniformBlockIndex, GLenum pname, GLint *params) const;
280 GLuint getActiveUniformBlockCount();
shannonwoods@chromium.orge684b582013-05-30 00:07:42 +0000281 GLint getActiveUniformBlockMaxLength();
282
Geoff Lang7dd2e102014-11-10 15:19:26 -0500283 GLuint getUniformBlockIndex(const std::string &name);
284
shannonwoods@chromium.org70eb1ea2013-05-30 00:07:20 +0000285 void bindUniformBlock(GLuint uniformBlockIndex, GLuint uniformBlockBinding);
286 GLuint getUniformBlockBinding(GLuint uniformBlockIndex) const;
287
Gregoire Payen de La Garanderie68694e92015-03-24 14:03:37 +0000288 const UniformBlock *getUniformBlockByIndex(GLuint index) const;
289
Geoff Lang48dcae72014-02-05 16:28:24 -0500290 void setTransformFeedbackVaryings(GLsizei count, const GLchar *const *varyings, GLenum bufferMode);
291 void getTransformFeedbackVarying(GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name) const;
Geoff Lang1b6edcb2014-02-03 14:27:56 -0500292 GLsizei getTransformFeedbackVaryingCount() const;
293 GLsizei getTransformFeedbackVaryingMaxLength() const;
294 GLenum getTransformFeedbackBufferMode() const;
295
Geoff Lang7dd2e102014-11-10 15:19:26 -0500296 static bool linkValidateUniforms(InfoLog &infoLog, const std::string &uniformName, const sh::Uniform &vertexUniform, const sh::Uniform &fragmentUniform);
297 static bool linkValidateInterfaceBlockFields(InfoLog &infoLog, const std::string &uniformName, const sh::InterfaceBlockField &vertexUniform, const sh::InterfaceBlockField &fragmentUniform);
298
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000299 void addRef();
300 void release();
301 unsigned int getRefCount() const;
302 void flagForDeletion();
303 bool isFlaggedForDeletion() const;
304
Brandon Jones43a53e22014-08-28 16:23:22 -0700305 void validate(const Caps &caps);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500306 bool validateSamplers(InfoLog *infoLog, const Caps &caps);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000307 bool isValidated() const;
apatrick@chromium.org90080e32012-07-09 22:15:33 +0000308
Jamie Madill63805b42015-08-25 13:17:39 -0400309 const AttributesMask &getActiveAttribLocationsMask() const
310 {
311 return mData.mActiveAttribLocationsMask;
312 }
313
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000314 private:
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000315 void unlink(bool destroy = false);
shannonwoods@chromium.org70eb1ea2013-05-30 00:07:20 +0000316 void resetUniformBlockBindings();
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000317
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400318 bool linkAttributes(const gl::Data &data,
Jamie Madill3da79b72015-04-27 11:09:17 -0400319 InfoLog &infoLog,
320 const AttributeBindings &attributeBindings,
321 const Shader *vertexShader);
Jamie Madille473dee2015-08-18 14:49:01 -0400322 bool linkUniformBlocks(InfoLog &infoLog, const Caps &caps);
Jamie Madillada9ecc2015-08-17 12:53:37 -0400323 static bool linkVaryings(InfoLog &infoLog,
324 const Shader *vertexShader,
325 const Shader *fragmentShader);
Jamie Madillea918db2015-08-18 14:48:59 -0400326 bool linkUniforms(gl::InfoLog &infoLog, const gl::Caps &caps) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500327 bool areMatchingInterfaceBlocks(gl::InfoLog &infoLog, const sh::InterfaceBlock &vertexInterfaceBlock,
328 const sh::InterfaceBlock &fragmentInterfaceBlock);
329
330 static bool linkValidateVariablesBase(InfoLog &infoLog,
331 const std::string &variableName,
332 const sh::ShaderVariable &vertexVariable,
333 const sh::ShaderVariable &fragmentVariable,
334 bool validatePrecision);
335
336 static bool linkValidateVaryings(InfoLog &infoLog, const std::string &varyingName, const sh::Varying &vertexVarying, const sh::Varying &fragmentVarying);
Jamie Madillccdf74b2015-08-18 10:46:12 -0400337 bool linkValidateTransformFeedback(InfoLog &infoLog,
338 const std::vector<const sh::Varying *> &linkedVaryings,
339 const Caps &caps) const;
340
341 void gatherTransformFeedbackVaryings(const std::vector<const sh::Varying *> &varyings);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500342 bool assignUniformBlockRegister(InfoLog &infoLog, UniformBlock *uniformBlock, GLenum shader, unsigned int registerIndex, const Caps &caps);
343 void defineOutputVariables(Shader *fragmentShader);
344
Jamie Madillccdf74b2015-08-18 10:46:12 -0400345 std::vector<const sh::Varying *> getMergedVaryings() const;
346
Jamie Madill5c6b7bf2015-08-17 12:53:35 -0400347 Data mData;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500348 rx::ProgramImpl *mProgram;
349
Geoff Lang7dd2e102014-11-10 15:19:26 -0500350 std::map<int, VariableLocation> mOutputVariables;
351
352 bool mValidated;
353
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000354 AttributeBindings mAttributeBindings;
355
daniel@transgaming.com716056c2012-07-24 18:38:59 +0000356 bool mLinked;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000357 bool mDeleteStatus; // Flag to indicate that the program can be deleted when no longer in use
daniel@transgaming.com4fa08332010-05-11 02:29:27 +0000358
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +0000359 unsigned int mRefCount;
360
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +0000361 ResourceManager *mResourceManager;
362 const GLuint mHandle;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000363
364 InfoLog mInfoLog;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000365};
366}
367
Geoff Lang0a73dd82014-11-19 16:18:08 -0500368#endif // LIBANGLE_PROGRAM_H_