blob: 4b8137a2ac21bf93cc8edfc278f6a0ec54ed2915 [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 Lang0b7eef72014-06-12 14:10:47 -040013#include <GLES2/gl2.h>
Jamie Madill861ebb32016-11-17 16:40:22 -050014#include <GLSLANG/ShaderVars.h>
Geoff Lang0b7eef72014-06-12 14:10:47 -040015
Jamie Madilla7d12dc2016-12-13 15:08:19 -050016#include <array>
Jamie Madill861ebb32016-11-17 16:40:22 -050017#include <map>
Jamie Madill63805b42015-08-25 13:17:39 -040018#include <set>
Jamie Madill71c3b2c2015-05-07 11:49:20 -040019#include <sstream>
Geoff Lang0b7eef72014-06-12 14:10:47 -040020#include <string>
Jamie Madill63805b42015-08-25 13:17:39 -040021#include <vector>
Geoff Lang0b7eef72014-06-12 14:10:47 -040022
Jamie Madill3d3d2f22015-09-23 16:47:51 -040023#include "common/angleutils.h"
24#include "common/mathutil.h"
25#include "common/Optional.h"
26
Jamie Madill3d3d2f22015-09-23 16:47:51 -040027#include "libANGLE/Constants.h"
Geoff Lang70d0f492015-12-10 17:45:46 -050028#include "libANGLE/Debug.h"
Jamie Madill3d3d2f22015-09-23 16:47:51 -040029#include "libANGLE/Error.h"
30#include "libANGLE/RefCountObject.h"
jchen107a20b972017-06-13 14:25:26 +080031#include "libANGLE/Uniform.h"
32#include "libANGLE/angletypes.h"
Jamie Madill3d3d2f22015-09-23 16:47:51 -040033
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000034namespace rx
35{
Jamie Madill7aea7e02016-05-10 10:39:45 -040036class GLImplFactory;
Geoff Lang7dd2e102014-11-10 15:19:26 -050037class ProgramImpl;
Jamie Madill5c6b7bf2015-08-17 12:53:35 -040038struct TranslatedAttribute;
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000039}
daniel@transgaming.come6842292010-04-20 18:52:50 +000040
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000041namespace gl
42{
Brandon Jones43a53e22014-08-28 16:23:22 -070043struct Caps;
Jamie Madilla2c74982016-12-12 11:20:42 -050044class Context;
Jamie Madilldfde6ab2016-06-09 07:07:18 -070045class ContextState;
shannon.woods@transgaming.com486d9e92013-02-28 23:15:41 +000046class Shader;
Geoff Lang4ddf5af2016-12-01 14:30:44 -050047class ShaderProgramManager;
Jamie Madillf4f8db82017-02-15 09:31:39 -050048class State;
Geoff Lang7dd2e102014-11-10 15:19:26 -050049class InfoLog;
Geoff Lang7dd2e102014-11-10 15:19:26 -050050class Buffer;
51class Framebuffer;
Jamie Madill192745a2016-12-22 15:58:21 -050052struct PackedVarying;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000053
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000054extern const char * const g_fakepath;
55
Jamie Madillf0d10f82015-03-31 12:56:52 -040056class InfoLog : angle::NonCopyable
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000057{
58 public:
59 InfoLog();
60 ~InfoLog();
61
Jamie Madill71c3b2c2015-05-07 11:49:20 -040062 size_t getLength() const;
Geoff Lange1a27752015-10-05 13:16:04 -040063 void getLog(GLsizei bufSize, GLsizei *length, char *infoLog) const;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000064
65 void appendSanitized(const char *message);
apatrick@chromium.org253b8d22012-06-22 19:27:21 +000066 void reset();
Jamie Madillf6113162015-05-07 11:49:21 -040067
68 // This helper class ensures we append a newline after writing a line.
69 class StreamHelper : angle::NonCopyable
70 {
71 public:
72 StreamHelper(StreamHelper &&rhs)
73 : mStream(rhs.mStream)
74 {
75 rhs.mStream = nullptr;
76 }
77
78 StreamHelper &operator=(StreamHelper &&rhs)
79 {
80 std::swap(mStream, rhs.mStream);
81 return *this;
82 }
83
84 ~StreamHelper()
85 {
86 // Write newline when destroyed on the stack
87 if (mStream)
88 {
89 (*mStream) << std::endl;
90 }
91 }
92
93 template <typename T>
94 StreamHelper &operator<<(const T &value)
95 {
96 (*mStream) << value;
97 return *this;
98 }
99
100 private:
101 friend class InfoLog;
102
103 StreamHelper(std::stringstream *stream)
104 : mStream(stream)
105 {
106 ASSERT(stream);
107 }
108
109 std::stringstream *mStream;
110 };
111
112 template <typename T>
113 StreamHelper operator<<(const T &value)
114 {
115 StreamHelper helper(&mStream);
116 helper << value;
117 return helper;
118 }
119
120 std::string str() const { return mStream.str(); }
121
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000122 private:
Jamie Madill71c3b2c2015-05-07 11:49:20 -0400123 std::stringstream mStream;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000124};
125
Geoff Lang7dd2e102014-11-10 15:19:26 -0500126// Struct used for correlating uniforms/elements of uniform arrays to handles
127struct VariableLocation
128{
129 VariableLocation();
130 VariableLocation(const std::string &name, unsigned int element, unsigned int index);
131
132 std::string name;
133 unsigned int element;
134 unsigned int index;
Geoff Langd8605522016-04-13 10:19:12 -0400135
136 // If this is a valid uniform location
137 bool used;
138
139 // If this location was bound to an unreferenced uniform. Setting data on this uniform is a
140 // no-op.
141 bool ignored;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500142};
143
Sami Väisänen46eaa942016-06-29 10:26:37 +0300144// Information about a variable binding.
145// Currently used by CHROMIUM_path_rendering
146struct BindingInfo
147{
148 // The type of binding, for example GL_FLOAT_VEC3.
149 // This can be GL_NONE if the variable is optimized away.
150 GLenum type;
151
152 // This is the name of the variable in
153 // the translated shader program. Note that
154 // this can be empty in the case where the
155 // variable has been optimized away.
156 std::string name;
157
158 // True if the binding is valid, otherwise false.
159 bool valid;
160};
161
Jamie Madille7d84322017-01-10 18:21:59 -0500162// This small structure encapsulates binding sampler uniforms to active GL textures.
163struct SamplerBinding
164{
165 SamplerBinding(GLenum textureTypeIn, size_t elementCount)
166 : textureType(textureTypeIn), boundTextureUnits(elementCount, 0)
167 {
168 }
169
170 // Necessary for retrieving active textures from the GL state.
171 GLenum textureType;
172
173 // List of all textures bound to this sampler, of type textureType.
174 std::vector<GLuint> boundTextureUnits;
175};
176
jchen10a9042d32017-03-17 08:50:45 +0800177// A varying with tranform feedback enabled. If it's an array, either the whole array or one of its
178// elements specified by 'arrayIndex' can set to be enabled.
179struct TransformFeedbackVarying : public sh::Varying
180{
181 TransformFeedbackVarying(const sh::Varying &varyingIn, GLuint index)
182 : sh::Varying(varyingIn), arrayIndex(index)
183 {
184 }
185 std::string nameWithArrayIndex() const
186 {
187 std::stringstream fullNameStr;
188 fullNameStr << name;
189 if (arrayIndex != GL_INVALID_INDEX)
190 {
191 fullNameStr << "[" << arrayIndex << "]";
192 }
193 return fullNameStr.str();
194 }
195 GLsizei size() const { return (arrayIndex == GL_INVALID_INDEX ? elementCount() : 1); }
196
197 GLuint arrayIndex;
198};
199
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800200struct ImageBinding
201{
202 ImageBinding(GLuint imageUnit, size_t count) : boundImageUnit(imageUnit), elementCount(count) {}
203
204 GLuint boundImageUnit;
205 size_t elementCount;
206};
207
Jamie Madill48ef11b2016-04-27 15:21:52 -0400208class ProgramState final : angle::NonCopyable
209{
210 public:
211 ProgramState();
212 ~ProgramState();
213
214 const std::string &getLabel();
215
Jamie Madillbd044ed2017-06-05 12:59:21 -0400216 Shader *getAttachedVertexShader() const { return mAttachedVertexShader; }
217 Shader *getAttachedFragmentShader() const { return mAttachedFragmentShader; }
218 Shader *getAttachedComputeShader() const { return mAttachedComputeShader; }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400219 const std::vector<std::string> &getTransformFeedbackVaryingNames() const
220 {
221 return mTransformFeedbackVaryingNames;
222 }
223 GLint getTransformFeedbackBufferMode() const { return mTransformFeedbackBufferMode; }
224 GLuint getUniformBlockBinding(GLuint uniformBlockIndex) const
225 {
jchen107a20b972017-06-13 14:25:26 +0800226 ASSERT(uniformBlockIndex < mUniformBlocks.size());
227 return mUniformBlocks[uniformBlockIndex].binding;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400228 }
229 const UniformBlockBindingMask &getActiveUniformBlockBindingsMask() const
230 {
231 return mActiveUniformBlockBindings;
232 }
233 const std::vector<sh::Attribute> &getAttributes() const { return mAttributes; }
234 const AttributesMask &getActiveAttribLocationsMask() const
235 {
236 return mActiveAttribLocationsMask;
237 }
Corentin Walleze7557742017-06-01 13:09:57 -0400238 DrawBufferMask getActiveOutputVariables() const { return mActiveOutputVariables; }
Jamie Madill4f86d052017-06-05 12:59:26 -0400239 const std::vector<sh::OutputVariable> &getOutputVariables() const { return mOutputVariables; }
jchen1015015f72017-03-16 13:54:21 +0800240 const std::map<int, VariableLocation> &getOutputLocations() const { return mOutputLocations; }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400241 const std::vector<LinkedUniform> &getUniforms() const { return mUniforms; }
242 const std::vector<VariableLocation> &getUniformLocations() const { return mUniformLocations; }
243 const std::vector<UniformBlock> &getUniformBlocks() const { return mUniformBlocks; }
Jamie Madille7d84322017-01-10 18:21:59 -0500244 const std::vector<SamplerBinding> &getSamplerBindings() const { return mSamplerBindings; }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800245 const std::vector<ImageBinding> &getImageBindings() const { return mImageBindings; }
Jamie Madill4f86d052017-06-05 12:59:26 -0400246 const sh::WorkGroupSize &getComputeShaderLocalSize() const { return mComputeShaderLocalSize; }
247 const RangeUI &getSamplerUniformRange() const { return mSamplerUniformRange; }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800248 const RangeUI &getImageUniformRange() const { return mImageUniformRange; }
Jamie Madill4f86d052017-06-05 12:59:26 -0400249
Jamie Madill4f86d052017-06-05 12:59:26 -0400250 const std::vector<TransformFeedbackVarying> &getLinkedTransformFeedbackVaryings() const
251 {
252 return mLinkedTransformFeedbackVaryings;
253 }
Jamie Madill48ef11b2016-04-27 15:21:52 -0400254
Jamie Madill48ef11b2016-04-27 15:21:52 -0400255 GLint getUniformLocation(const std::string &name) const;
Jamie Madille7d84322017-01-10 18:21:59 -0500256 GLuint getUniformIndexFromName(const std::string &name) const;
257 GLuint getUniformIndexFromLocation(GLint location) const;
258 Optional<GLuint> getSamplerIndex(GLint location) const;
259 bool isSamplerUniformIndex(GLuint index) const;
260 GLuint getSamplerIndexFromUniformIndex(GLuint uniformIndex) const;
Jamie Madill34ca4f52017-06-13 11:49:39 -0400261 GLuint getAttributeLocation(const std::string &name) const;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400262
263 private:
Jamie Madill4f86d052017-06-05 12:59:26 -0400264 friend class MemoryProgramCache;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400265 friend class Program;
266
267 std::string mLabel;
268
Martin Radev4c4c8e72016-08-04 12:25:34 +0300269 sh::WorkGroupSize mComputeShaderLocalSize;
270
Jamie Madill48ef11b2016-04-27 15:21:52 -0400271 Shader *mAttachedFragmentShader;
272 Shader *mAttachedVertexShader;
Martin Radev4c4c8e72016-08-04 12:25:34 +0300273 Shader *mAttachedComputeShader;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400274
275 std::vector<std::string> mTransformFeedbackVaryingNames;
jchen10a9042d32017-03-17 08:50:45 +0800276 std::vector<TransformFeedbackVarying> mLinkedTransformFeedbackVaryings;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400277 GLenum mTransformFeedbackBufferMode;
278
jchen107a20b972017-06-13 14:25:26 +0800279 // For faster iteration on the blocks currently being bound.
Jamie Madill48ef11b2016-04-27 15:21:52 -0400280 UniformBlockBindingMask mActiveUniformBlockBindings;
281
282 std::vector<sh::Attribute> mAttributes;
Jamie Madill6de51852017-04-12 09:53:01 -0400283 angle::BitSet<MAX_VERTEX_ATTRIBS> mActiveAttribLocationsMask;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400284
285 // Uniforms are sorted in order:
286 // 1. Non-sampler uniforms
287 // 2. Sampler uniforms
288 // 3. Uniform block uniforms
289 // This makes sampler validation easier, since we don't need a separate list.
290 std::vector<LinkedUniform> mUniforms;
291 std::vector<VariableLocation> mUniformLocations;
292 std::vector<UniformBlock> mUniformBlocks;
Jamie Madille7d84322017-01-10 18:21:59 -0500293 RangeUI mSamplerUniformRange;
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800294 RangeUI mImageUniformRange;
Jamie Madille7d84322017-01-10 18:21:59 -0500295
296 // An array of the samplers that are used by the program
297 std::vector<gl::SamplerBinding> mSamplerBindings;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400298
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800299 // An array of the images that are used by the program
300 std::vector<gl::ImageBinding> mImageBindings;
301
jchen1015015f72017-03-16 13:54:21 +0800302 std::vector<sh::OutputVariable> mOutputVariables;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400303 // TODO(jmadill): use unordered/hash map when available
jchen1015015f72017-03-16 13:54:21 +0800304 std::map<int, VariableLocation> mOutputLocations;
Corentin Walleze7557742017-06-01 13:09:57 -0400305 DrawBufferMask mActiveOutputVariables;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400306
Geoff Lange0cff192017-05-30 13:04:56 -0400307 // Fragment output variable base types: FLOAT, INT, or UINT. Ordered by location.
308 std::vector<GLenum> mOutputVariableTypes;
309
Jamie Madill48ef11b2016-04-27 15:21:52 -0400310 bool mBinaryRetrieveableHint;
Yunchao He61afff12017-03-14 15:34:03 +0800311 bool mSeparable;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400312};
313
Geoff Lang70d0f492015-12-10 17:45:46 -0500314class Program final : angle::NonCopyable, public LabeledObject
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000315{
316 public:
Geoff Lang4ddf5af2016-12-01 14:30:44 -0500317 Program(rx::GLImplFactory *factory, ShaderProgramManager *manager, GLuint handle);
Jamie Madill4928b7c2017-06-20 12:57:39 -0400318 void onDestroy(const Context *context);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000319
Geoff Lang7dd2e102014-11-10 15:19:26 -0500320 GLuint id() const { return mHandle; }
321
Geoff Lang70d0f492015-12-10 17:45:46 -0500322 void setLabel(const std::string &label) override;
323 const std::string &getLabel() const override;
324
Geoff Lang47110bf2016-04-20 11:13:22 -0700325 rx::ProgramImpl *getImplementation() const { return mProgram; }
Geoff Lang7dd2e102014-11-10 15:19:26 -0500326
Jamie Madillef300b12016-10-07 15:12:09 -0400327 void attachShader(Shader *shader);
Jamie Madillc1d770e2017-04-13 17:31:24 -0400328 void detachShader(const Context *context, Shader *shader);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000329 int getAttachedShadersCount() const;
330
Jamie Madillef300b12016-10-07 15:12:09 -0400331 const Shader *getAttachedVertexShader() const { return mState.mAttachedVertexShader; }
332 const Shader *getAttachedFragmentShader() const { return mState.mAttachedFragmentShader; }
333 const Shader *getAttachedComputeShader() const { return mState.mAttachedComputeShader; }
334
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000335 void bindAttributeLocation(GLuint index, const char *name);
Geoff Langd8605522016-04-13 10:19:12 -0400336 void bindUniformLocation(GLuint index, const char *name);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000337
Sami Väisänen46eaa942016-06-29 10:26:37 +0300338 // CHROMIUM_path_rendering
Jamie Madillbd044ed2017-06-05 12:59:21 -0400339 BindingInfo getFragmentInputBindingInfo(const Context *context, GLint index) const;
Sami Väisänen46eaa942016-06-29 10:26:37 +0300340 void bindFragmentInputLocation(GLint index, const char *name);
Jamie Madillbd044ed2017-06-05 12:59:21 -0400341 void pathFragmentInputGen(const Context *context,
342 GLint index,
Sami Väisänen46eaa942016-06-29 10:26:37 +0300343 GLenum genMode,
344 GLint components,
345 const GLfloat *coeffs);
346
Jamie Madill8ecf7f92017-01-13 17:29:52 -0500347 Error link(const gl::Context *context);
Geoff Lange1a27752015-10-05 13:16:04 -0400348 bool isLinked() const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500349
Jamie Madilla2c74982016-12-12 11:20:42 -0500350 Error loadBinary(const Context *context,
351 GLenum binaryFormat,
352 const void *binary,
353 GLsizei length);
354 Error saveBinary(const Context *context,
355 GLenum *binaryFormat,
356 void *binary,
357 GLsizei bufSize,
358 GLsizei *length) const;
Jamie Madillffe00c02017-06-27 16:26:55 -0400359 GLint getBinaryLength(const Context *context) const;
Geoff Langc5629752015-12-07 16:29:04 -0500360 void setBinaryRetrievableHint(bool retrievable);
361 bool getBinaryRetrievableHint() const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000362
Yunchao He61afff12017-03-14 15:34:03 +0800363 void setSeparable(bool separable);
364 bool isSeparable() const;
365
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000366 int getInfoLogLength() const;
Geoff Lange1a27752015-10-05 13:16:04 -0400367 void getInfoLog(GLsizei bufSize, GLsizei *length, char *infoLog) const;
368 void getAttachedShaders(GLsizei maxCount, GLsizei *count, GLuint *shaders) const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000369
Geoff Lange1a27752015-10-05 13:16:04 -0400370 GLuint getAttributeLocation(const std::string &name) const;
Jamie Madill63805b42015-08-25 13:17:39 -0400371 bool isAttribLocationActive(size_t attribLocation) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500372
jchen10fd7c3b52017-03-21 15:36:03 +0800373 void getActiveAttribute(GLuint index,
374 GLsizei bufsize,
375 GLsizei *length,
376 GLint *size,
377 GLenum *type,
378 GLchar *name) const;
Geoff Lange1a27752015-10-05 13:16:04 -0400379 GLint getActiveAttributeCount() const;
380 GLint getActiveAttributeMaxLength() const;
Jamie Madill48ef11b2016-04-27 15:21:52 -0400381 const std::vector<sh::Attribute> &getAttributes() const { return mState.mAttributes; }
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000382
Geoff Lang7dd2e102014-11-10 15:19:26 -0500383 GLint getFragDataLocation(const std::string &name) const;
jchen10fd7c3b52017-03-21 15:36:03 +0800384 size_t getOutputResourceCount() const;
Geoff Lange0cff192017-05-30 13:04:56 -0400385 const std::vector<GLenum> &getOutputVariableTypes() const
386 {
387 return mState.mOutputVariableTypes;
388 }
Corentin Wallezdb9e5d32017-06-12 12:05:45 -0700389 DrawBufferMask getActiveOutputVariables() const { return mState.mActiveOutputVariables; }
Geoff Lang7dd2e102014-11-10 15:19:26 -0500390
Geoff Lange1a27752015-10-05 13:16:04 -0400391 void getActiveUniform(GLuint index,
392 GLsizei bufsize,
393 GLsizei *length,
394 GLint *size,
395 GLenum *type,
396 GLchar *name) const;
397 GLint getActiveUniformCount() const;
398 GLint getActiveUniformMaxLength() const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500399 GLint getActiveUniformi(GLuint index, GLenum pname) const;
400 bool isValidUniformLocation(GLint location) const;
Jamie Madill62d31cb2015-09-11 13:25:51 -0400401 const LinkedUniform &getUniformByLocation(GLint location) const;
Jamie Madillac4e9c32017-01-13 14:07:12 -0500402 const VariableLocation &getUniformLocation(GLint location) const;
403 const std::vector<VariableLocation> &getUniformLocations() const;
404 const LinkedUniform &getUniformByIndex(GLuint index) const;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000405
Jamie Madill62d31cb2015-09-11 13:25:51 -0400406 GLint getUniformLocation(const std::string &name) const;
407 GLuint getUniformIndex(const std::string &name) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500408 void setUniform1fv(GLint location, GLsizei count, const GLfloat *v);
409 void setUniform2fv(GLint location, GLsizei count, const GLfloat *v);
410 void setUniform3fv(GLint location, GLsizei count, const GLfloat *v);
411 void setUniform4fv(GLint location, GLsizei count, const GLfloat *v);
412 void setUniform1iv(GLint location, GLsizei count, const GLint *v);
413 void setUniform2iv(GLint location, GLsizei count, const GLint *v);
414 void setUniform3iv(GLint location, GLsizei count, const GLint *v);
415 void setUniform4iv(GLint location, GLsizei count, const GLint *v);
416 void setUniform1uiv(GLint location, GLsizei count, const GLuint *v);
417 void setUniform2uiv(GLint location, GLsizei count, const GLuint *v);
418 void setUniform3uiv(GLint location, GLsizei count, const GLuint *v);
419 void setUniform4uiv(GLint location, GLsizei count, const GLuint *v);
420 void setUniformMatrix2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
421 void setUniformMatrix3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
422 void setUniformMatrix4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
423 void setUniformMatrix2x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
424 void setUniformMatrix3x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
425 void setUniformMatrix2x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
426 void setUniformMatrix4x2fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
427 void setUniformMatrix3x4fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
428 void setUniformMatrix4x3fv(GLint location, GLsizei count, GLboolean transpose, const GLfloat *value);
429
Geoff Lange1a27752015-10-05 13:16:04 -0400430 void getUniformfv(GLint location, GLfloat *params) const;
431 void getUniformiv(GLint location, GLint *params) const;
432 void getUniformuiv(GLint location, GLuint *params) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500433
Geoff Lang7dd2e102014-11-10 15:19:26 -0500434 void getActiveUniformBlockName(GLuint uniformBlockIndex, GLsizei bufSize, GLsizei *length, GLchar *uniformBlockName) const;
Geoff Lange1a27752015-10-05 13:16:04 -0400435 GLuint getActiveUniformBlockCount() const;
436 GLint getActiveUniformBlockMaxLength() const;
shannonwoods@chromium.orge684b582013-05-30 00:07:42 +0000437
Geoff Lange1a27752015-10-05 13:16:04 -0400438 GLuint getUniformBlockIndex(const std::string &name) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500439
shannonwoods@chromium.org70eb1ea2013-05-30 00:07:20 +0000440 void bindUniformBlock(GLuint uniformBlockIndex, GLuint uniformBlockBinding);
441 GLuint getUniformBlockBinding(GLuint uniformBlockIndex) const;
442
Jamie Madill62d31cb2015-09-11 13:25:51 -0400443 const UniformBlock &getUniformBlockByIndex(GLuint index) const;
Gregoire Payen de La Garanderie68694e92015-03-24 14:03:37 +0000444
Geoff Lang48dcae72014-02-05 16:28:24 -0500445 void setTransformFeedbackVaryings(GLsizei count, const GLchar *const *varyings, GLenum bufferMode);
446 void getTransformFeedbackVarying(GLuint index, GLsizei bufSize, GLsizei *length, GLsizei *size, GLenum *type, GLchar *name) const;
Geoff Lang1b6edcb2014-02-03 14:27:56 -0500447 GLsizei getTransformFeedbackVaryingCount() const;
448 GLsizei getTransformFeedbackVaryingMaxLength() const;
449 GLenum getTransformFeedbackBufferMode() const;
450
Frank Henigmanfccbac22017-05-28 17:29:26 -0400451 static bool linkValidateInterfaceBlockFields(InfoLog &infoLog,
452 const std::string &uniformName,
453 const sh::InterfaceBlockField &vertexUniform,
454 const sh::InterfaceBlockField &fragmentUniform,
455 bool webglCompatibility);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500456
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000457 void addRef();
Jamie Madill6c1f6712017-02-14 19:08:04 -0500458 void release(const Context *context);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000459 unsigned int getRefCount() const;
460 void flagForDeletion();
461 bool isFlaggedForDeletion() const;
462
Brandon Jones43a53e22014-08-28 16:23:22 -0700463 void validate(const Caps &caps);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500464 bool validateSamplers(InfoLog *infoLog, const Caps &caps);
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000465 bool isValidated() const;
Jamie Madilla4595b82017-01-11 17:36:34 -0500466 bool samplesFromTexture(const gl::State &state, GLuint textureID) const;
apatrick@chromium.org90080e32012-07-09 22:15:33 +0000467
Jamie Madill63805b42015-08-25 13:17:39 -0400468 const AttributesMask &getActiveAttribLocationsMask() const
469 {
Jamie Madill48ef11b2016-04-27 15:21:52 -0400470 return mState.mActiveAttribLocationsMask;
Jamie Madill63805b42015-08-25 13:17:39 -0400471 }
472
Jamie Madille7d84322017-01-10 18:21:59 -0500473 const std::vector<SamplerBinding> &getSamplerBindings() const
474 {
475 return mState.mSamplerBindings;
476 }
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800477
478 const std::vector<ImageBinding> &getImageBindings() const { return mState.mImageBindings; }
479
Jamie Madilldf68a6f2017-01-13 17:29:53 -0500480 const ProgramState &getState() const { return mState; }
Jamie Madille7d84322017-01-10 18:21:59 -0500481
Olli Etuahob78707c2017-03-09 15:03:11 +0000482 static bool linkValidateVariablesBase(InfoLog &infoLog,
483 const std::string &variableName,
484 const sh::ShaderVariable &vertexVariable,
485 const sh::ShaderVariable &fragmentVariable,
486 bool validatePrecision);
487
jchen1015015f72017-03-16 13:54:21 +0800488 GLuint getInputResourceIndex(const GLchar *name) const;
489 GLuint getOutputResourceIndex(const GLchar *name) const;
jchen10fd7c3b52017-03-21 15:36:03 +0800490 void getInputResourceName(GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name) const;
491 void getOutputResourceName(GLuint index, GLsizei bufSize, GLsizei *length, GLchar *name) const;
jchen1015015f72017-03-16 13:54:21 +0800492
Geoff Langd8605522016-04-13 10:19:12 -0400493 class Bindings final : angle::NonCopyable
494 {
495 public:
496 void bindLocation(GLuint index, const std::string &name);
497 int getBinding(const std::string &name) const;
498
499 typedef std::unordered_map<std::string, GLuint>::const_iterator const_iterator;
500 const_iterator begin() const;
501 const_iterator end() const;
502
503 private:
504 std::unordered_map<std::string, GLuint> mBindings;
505 };
506
Jamie Madill32447362017-06-28 14:53:52 -0400507 const Bindings &getAttributeBindings() const { return mAttributeBindings; }
508 const Bindings &getUniformLocationBindings() const { return mUniformLocationBindings; }
509 const Bindings &getFragmentInputBindings() const { return mFragmentInputBindings; }
510
Olli Etuahob78707c2017-03-09 15:03:11 +0000511 private:
Jamie Madill4928b7c2017-06-20 12:57:39 -0400512 ~Program();
513
Jamie Madill192745a2016-12-22 15:58:21 -0500514 struct VaryingRef
515 {
516 const sh::Varying *get() const { return vertex ? vertex : fragment; }
517
518 const sh::Varying *vertex = nullptr;
519 const sh::Varying *fragment = nullptr;
520 };
521
522 using MergedVaryings = std::map<std::string, VaryingRef>;
523
Jamie Madill6c1f6712017-02-14 19:08:04 -0500524 void unlink();
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000525
Jamie Madillbd044ed2017-06-05 12:59:21 -0400526 bool linkAttributes(const Context *context, InfoLog &infoLog);
Martin Radev4c4c8e72016-08-04 12:25:34 +0300527 bool validateUniformBlocksCount(GLuint maxUniformBlocks,
528 const std::vector<sh::InterfaceBlock> &block,
529 const std::string &errorMessage,
530 InfoLog &infoLog) const;
531 bool validateVertexAndFragmentInterfaceBlocks(
532 const std::vector<sh::InterfaceBlock> &vertexInterfaceBlocks,
533 const std::vector<sh::InterfaceBlock> &fragmentInterfaceBlocks,
Frank Henigmanfccbac22017-05-28 17:29:26 -0400534 InfoLog &infoLog,
535 bool webglCompatibility) const;
Jamie Madillbd044ed2017-06-05 12:59:21 -0400536 bool linkUniformBlocks(const Context *context, InfoLog &infoLog);
537 bool linkVaryings(const Context *context, InfoLog &infoLog) const;
Olli Etuaho6ca2b652017-02-19 18:05:10 +0000538
Jamie Madillbd044ed2017-06-05 12:59:21 -0400539 bool linkUniforms(const Context *context,
540 InfoLog &infoLog,
541 const Bindings &uniformLocationBindings);
Xinghua Cao65ec0b22017-03-28 16:10:52 +0800542 void linkSamplerAndImageBindings();
Olli Etuaho6ca2b652017-02-19 18:05:10 +0000543
Jamie Madilla2c74982016-12-12 11:20:42 -0500544 bool areMatchingInterfaceBlocks(InfoLog &infoLog,
Martin Radev4c4c8e72016-08-04 12:25:34 +0300545 const sh::InterfaceBlock &vertexInterfaceBlock,
Frank Henigmanfccbac22017-05-28 17:29:26 -0400546 const sh::InterfaceBlock &fragmentInterfaceBlock,
547 bool webglCompatibility) const;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500548
Yuly Novikova1f6dc92016-06-15 23:27:04 -0400549 static bool linkValidateVaryings(InfoLog &infoLog,
550 const std::string &varyingName,
551 const sh::Varying &vertexVarying,
552 const sh::Varying &fragmentVarying,
553 int shaderVersion);
Jamie Madillbd044ed2017-06-05 12:59:21 -0400554 bool linkValidateBuiltInVaryings(const Context *context, InfoLog &infoLog) const;
jchen10a9042d32017-03-17 08:50:45 +0800555 bool linkValidateTransformFeedback(const gl::Context *context,
556 InfoLog &infoLog,
Jamie Madill192745a2016-12-22 15:58:21 -0500557 const MergedVaryings &linkedVaryings,
Jamie Madillccdf74b2015-08-18 10:46:12 -0400558 const Caps &caps) const;
559
Jamie Madill192745a2016-12-22 15:58:21 -0500560 void gatherTransformFeedbackVaryings(const MergedVaryings &varyings);
Geoff Lang7dd2e102014-11-10 15:19:26 -0500561
Jamie Madillbd044ed2017-06-05 12:59:21 -0400562 MergedVaryings getMergedVaryings(const Context *context) const;
Jamie Madill192745a2016-12-22 15:58:21 -0500563 std::vector<PackedVarying> getPackedVaryings(const MergedVaryings &mergedVaryings) const;
Jamie Madillbd044ed2017-06-05 12:59:21 -0400564 void linkOutputVariables(const Context *context);
Jamie Madillccdf74b2015-08-18 10:46:12 -0400565
Olli Etuaho48fed632017-03-16 12:05:30 +0000566 void setUniformValuesFromBindingQualifiers();
567
Jamie Madillbd044ed2017-06-05 12:59:21 -0400568 void gatherInterfaceBlockInfo(const Context *context);
Jamie Madill4a3c2342015-10-08 12:58:45 -0400569 template <typename VarT>
570 void defineUniformBlockMembers(const std::vector<VarT> &fields,
571 const std::string &prefix,
572 int blockIndex);
573
Jamie Madill62d31cb2015-09-11 13:25:51 -0400574 void defineUniformBlock(const sh::InterfaceBlock &interfaceBlock, GLenum shaderType);
575
Corentin Wallez8b7d8142016-11-15 13:40:37 -0500576 // Both these function update the cached uniform values and return a modified "count"
577 // so that the uniform update doesn't overflow the uniform.
Jamie Madill62d31cb2015-09-11 13:25:51 -0400578 template <typename T>
Corentin Wallez8b7d8142016-11-15 13:40:37 -0500579 GLsizei setUniformInternal(GLint location, GLsizei count, int vectorSize, const T *v);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400580 template <size_t cols, size_t rows, typename T>
Corentin Wallez8b7d8142016-11-15 13:40:37 -0500581 GLsizei setMatrixUniformInternal(GLint location,
582 GLsizei count,
583 GLboolean transpose,
584 const T *v);
Jamie Madille7d84322017-01-10 18:21:59 -0500585 template <typename T>
586 void updateSamplerUniform(const VariableLocation &locationInfo,
587 const uint8_t *destPointer,
588 GLsizei clampedCount,
589 const T *v);
Jamie Madill62d31cb2015-09-11 13:25:51 -0400590
591 template <typename DestT>
592 void getUniformInternal(GLint location, DestT *dataOut) const;
593
Jamie Madill48ef11b2016-04-27 15:21:52 -0400594 ProgramState mState;
Geoff Lang7dd2e102014-11-10 15:19:26 -0500595 rx::ProgramImpl *mProgram;
596
Geoff Lang7dd2e102014-11-10 15:19:26 -0500597 bool mValidated;
598
Geoff Langd8605522016-04-13 10:19:12 -0400599 Bindings mAttributeBindings;
Olli Etuaho4a92ceb2017-02-19 17:51:24 +0000600
601 // Note that this has nothing to do with binding layout qualifiers that can be set for some
602 // uniforms in GLES3.1+. It is used to pre-set the location of uniforms.
603 Bindings mUniformLocationBindings;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000604
Sami Väisänen46eaa942016-06-29 10:26:37 +0300605 // CHROMIUM_path_rendering
606 Bindings mFragmentInputBindings;
607
daniel@transgaming.com716056c2012-07-24 18:38:59 +0000608 bool mLinked;
apatrick@chromium.orge2a59bb2012-06-07 21:09:53 +0000609 bool mDeleteStatus; // Flag to indicate that the program can be deleted when no longer in use
daniel@transgaming.com4fa08332010-05-11 02:29:27 +0000610
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +0000611 unsigned int mRefCount;
612
Geoff Lang4ddf5af2016-12-01 14:30:44 -0500613 ShaderProgramManager *mResourceManager;
daniel@transgaming.comda13f3e2010-07-28 19:20:56 +0000614 const GLuint mHandle;
apatrick@chromium.org253b8d22012-06-22 19:27:21 +0000615
616 InfoLog mInfoLog;
Jamie Madill3d3d2f22015-09-23 16:47:51 -0400617
618 // Cache for sampler validation
619 Optional<bool> mCachedValidateSamplersResult;
620 std::vector<GLenum> mTextureUnitTypesCache;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000621};
Jamie Madilla2c74982016-12-12 11:20:42 -0500622} // namespace gl
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000623
Geoff Lang0a73dd82014-11-19 16:18:08 -0500624#endif // LIBANGLE_PROGRAM_H_