1 /*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 #ifndef _GL_CLIENT_STATE_H_
17 #define _GL_CLIENT_STATE_H_
18 
19 #define GL_API
20 #ifndef ANDROID
21 #define GL_APIENTRY
22 #define GL_APIENTRYP
23 #endif
24 
25 #include <GLES/gl.h>
26 #include <GLES/glext.h>
27 #include <GLES2/gl2.h>
28 #include <GLES2/gl2ext.h>
29 
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include "ErrorLog.h"
33 #include "codec_defs.h"
34 
35 class GLClientState {
36 public:
37     typedef enum {
38         VERTEX_LOCATION = 0,
39         NORMAL_LOCATION = 1,
40         COLOR_LOCATION = 2,
41         POINTSIZE_LOCATION = 3,
42         TEXCOORD0_LOCATION = 4,
43         TEXCOORD1_LOCATION = 5,
44         TEXCOORD2_LOCATION = 6,
45         TEXCOORD3_LOCATION = 7,
46         TEXCOORD4_LOCATION = 8,
47         TEXCOORD5_LOCATION = 9,
48         TEXCOORD6_LOCATION = 10,
49         TEXCOORD7_LOCATION = 11,
50         MATRIXINDEX_LOCATION = 12,
51         WEIGHT_LOCATION = 13,
52         LAST_LOCATION = 14
53     } StateLocation;
54 
55     typedef struct {
56         GLint enabled;
57         GLint size;
58         GLenum type;
59         GLsizei stride;
60         void *data;
61         GLuint bufferObject;
62         GLenum glConst;
63         unsigned int elementSize;
64         bool enableDirty;  // true if any enable state has changed since last draw
65         bool normalized;
66     } VertexAttribState;
67 
68     typedef struct {
69         int unpack_alignment;
70         int pack_alignment;
71     } PixelStoreState;
72 
73     enum {
74         MAX_TEXTURE_UNITS = 32,
75     };
76 
77 public:
78     GLClientState(int nLocations = CODEC_MAX_VERTEX_ATTRIBUTES);
79     ~GLClientState();
nLocations()80     int nLocations() { return m_nLocations; }
pixelStoreState()81     const PixelStoreState *pixelStoreState() { return &m_pixelStore; }
82     int setPixelStore(GLenum param, GLint value);
currentArrayVbo()83     GLuint currentArrayVbo() { return m_currentArrayVbo; }
currentIndexVbo()84     GLuint currentIndexVbo() { return m_currentIndexVbo; }
85     void enable(int location, int state);
86     void setState(int  location, int size, GLenum type, GLboolean normalized, GLsizei stride, const void *data);
87     void setBufferObject(int location, GLuint id);
88     const VertexAttribState  *getState(int location);
89     const VertexAttribState  *getStateAndEnableDirty(int location, bool *enableChanged);
90     int getLocation(GLenum loc);
setActiveTexture(int texUnit)91     void setActiveTexture(int texUnit) {m_activeTexture = texUnit; };
getActiveTexture()92     int getActiveTexture() const { return m_activeTexture; }
93 
bindBuffer(GLenum target,GLuint id)94     int bindBuffer(GLenum target, GLuint id)
95     {
96         int err = 0;
97         switch(target) {
98         case GL_ARRAY_BUFFER:
99             m_currentArrayVbo = id;
100             break;
101         case GL_ELEMENT_ARRAY_BUFFER:
102             m_currentIndexVbo = id;
103             break;
104         default:
105             err = -1;
106         }
107         return err;
108     }
109 
getBuffer(GLenum target)110     int getBuffer(GLenum target)
111     {
112       int ret=0;
113       switch (target) {
114       case GL_ARRAY_BUFFER:
115           ret = m_currentArrayVbo;
116           break;
117       case GL_ELEMENT_ARRAY_BUFFER:
118           ret = m_currentIndexVbo;
119           break;
120       default:
121           ret = -1;
122       }
123       return ret;
124     }
125     size_t pixelDataSize(GLsizei width, GLsizei height, GLenum format, GLenum type, int pack) const;
126 
setCurrentProgram(GLint program)127     void setCurrentProgram(GLint program) { m_currentProgram = program; }
currentProgram()128     GLint currentProgram() const { return m_currentProgram; }
129 
130     /* OES_EGL_image_external
131      *
132      * These functions manipulate GL state which interacts with the
133      * OES_EGL_image_external extension, to support client-side emulation on
134      * top of host implementations that don't have it.
135      *
136      * Most of these calls should only be used with TEXTURE_2D or
137      * TEXTURE_EXTERNAL_OES texture targets; TEXTURE_CUBE_MAP or other extension
138      * targets should bypass this. An exception is bindTexture(), which should
139      * see all glBindTexture() calls for any target.
140      */
141 
142     // glActiveTexture(GL_TEXTURE0 + i)
143     // Sets the active texture unit. Up to MAX_TEXTURE_UNITS are supported.
144     GLenum setActiveTextureUnit(GLenum texture);
145     GLenum getActiveTextureUnit() const;
146 
147     // glEnable(GL_TEXTURE_(2D|EXTERNAL_OES))
148     void enableTextureTarget(GLenum target);
149 
150     // glDisable(GL_TEXTURE_(2D|EXTERNAL_OES))
151     void disableTextureTarget(GLenum target);
152 
153     // Implements the target priority logic:
154     // * Return GL_TEXTURE_EXTERNAL_OES if enabled, else
155     // * Return GL_TEXTURE_2D if enabled, else
156     // * Return the allDisabled value.
157     // For some cases passing GL_TEXTURE_2D for allDisabled makes callee code
158     // simpler; for other cases passing a recognizable enum like GL_ZERO or
159     // GL_INVALID_ENUM is appropriate.
160     GLenum getPriorityEnabledTarget(GLenum allDisabled) const;
161 
162     // glBindTexture(GL_TEXTURE_*, ...)
163     // Set the target binding of the active texture unit to texture. Returns
164     // GL_NO_ERROR on success or GL_INVALID_OPERATION if the texture has
165     // previously been bound to a different target. If firstUse is not NULL,
166     // it is set to indicate whether this is the first use of the texture.
167     // For accurate error detection, bindTexture should be called for *all*
168     // targets, not just 2D and EXTERNAL_OES.
169     GLenum bindTexture(GLenum target, GLuint texture, GLboolean* firstUse);
170 
171     // Return the texture currently bound to GL_TEXTURE_(2D|EXTERNAL_OES).
172     GLuint getBoundTexture(GLenum target) const;
173 
174     // glDeleteTextures(...)
175     // Remove references to the to-be-deleted textures.
176     void deleteTextures(GLsizei n, const GLuint* textures);
177 
178 private:
179     PixelStoreState m_pixelStore;
180     VertexAttribState *m_states;
181     int m_nLocations;
182     GLuint m_currentArrayVbo;
183     GLuint m_currentIndexVbo;
184     int m_activeTexture;
185     GLint m_currentProgram;
186 
validLocation(int location)187     bool validLocation(int location) { return (location >= 0 && location < m_nLocations); }
188 
189     enum TextureTarget {
190         TEXTURE_2D = 0,
191         TEXTURE_EXTERNAL = 1,
192         TEXTURE_TARGET_COUNT
193     };
194     struct TextureUnit {
195         unsigned int enables;
196         GLuint texture[TEXTURE_TARGET_COUNT];
197     };
198     struct TextureRec {
199         GLuint id;
200         GLenum target;
201     };
202     struct TextureState {
203         TextureUnit unit[MAX_TEXTURE_UNITS];
204         TextureUnit* activeUnit;
205         TextureRec* textures;
206         GLuint numTextures;
207         GLuint allocTextures;
208     };
209     TextureState m_tex;
210 
211     static int compareTexId(const void* pid, const void* prec);
212     TextureRec* addTextureRec(GLuint id, GLenum target);
213 
214 public:
215     void getClientStatePointer(GLenum pname, GLvoid** params);
216 
217     template <class T>
getVertexAttribParameter(GLuint index,GLenum param,T * ptr)218     int getVertexAttribParameter(GLuint index, GLenum param, T *ptr)
219     {
220         bool handled = true;
221         const VertexAttribState *vertexAttrib = getState(index);
222         if (vertexAttrib == NULL) {
223             ERR("getVeterxAttriParameter for non existant index %d\n", index);
224             // set gl error;
225             return handled;
226         }
227 
228         switch(param) {
229         case GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING:
230             *ptr = (T)(vertexAttrib->bufferObject);
231             break;
232         case GL_VERTEX_ATTRIB_ARRAY_ENABLED:
233             *ptr = (T)(vertexAttrib->enabled);
234             break;
235         case GL_VERTEX_ATTRIB_ARRAY_SIZE:
236             *ptr = (T)(vertexAttrib->size);
237             break;
238         case GL_VERTEX_ATTRIB_ARRAY_STRIDE:
239             *ptr = (T)(vertexAttrib->stride);
240             break;
241         case GL_VERTEX_ATTRIB_ARRAY_TYPE:
242             *ptr = (T)(vertexAttrib->type);
243             break;
244         case GL_VERTEX_ATTRIB_ARRAY_NORMALIZED:
245             *ptr = (T)(vertexAttrib->normalized);
246             break;
247         case GL_CURRENT_VERTEX_ATTRIB:
248             handled = false;
249             break;
250         default:
251             handled = false;
252             ERR("unknown vertex-attrib parameter param %d\n", param);
253         }
254         return handled;
255     }
256 
257     template <class T>
getClientStateParameter(GLenum param,T * ptr)258     bool getClientStateParameter(GLenum param, T* ptr)
259     {
260         bool isClientStateParam = false;
261         switch (param) {
262         case GL_CLIENT_ACTIVE_TEXTURE: {
263             GLint tex = getActiveTexture() + GL_TEXTURE0;
264             *ptr = tex;
265             isClientStateParam = true;
266             break;
267             }
268         case GL_VERTEX_ARRAY_SIZE: {
269             const GLClientState::VertexAttribState *state = getState(GLClientState::VERTEX_LOCATION);
270             *ptr = state->size;
271             isClientStateParam = true;
272             break;
273             }
274         case GL_VERTEX_ARRAY_TYPE: {
275             const GLClientState::VertexAttribState *state = getState(GLClientState::VERTEX_LOCATION);
276             *ptr = state->type;
277             isClientStateParam = true;
278             break;
279             }
280         case GL_VERTEX_ARRAY_STRIDE: {
281             const GLClientState::VertexAttribState *state = getState(GLClientState::VERTEX_LOCATION);
282             *ptr = state->stride;
283             isClientStateParam = true;
284             break;
285             }
286         case GL_COLOR_ARRAY_SIZE: {
287             const GLClientState::VertexAttribState *state = getState(GLClientState::COLOR_LOCATION);
288             *ptr = state->size;
289             isClientStateParam = true;
290             break;
291             }
292         case GL_COLOR_ARRAY_TYPE: {
293             const GLClientState::VertexAttribState *state = getState(GLClientState::COLOR_LOCATION);
294             *ptr = state->type;
295             isClientStateParam = true;
296             break;
297             }
298         case GL_COLOR_ARRAY_STRIDE: {
299             const GLClientState::VertexAttribState *state = getState(GLClientState::COLOR_LOCATION);
300             *ptr = state->stride;
301             isClientStateParam = true;
302             break;
303             }
304         case GL_NORMAL_ARRAY_TYPE: {
305             const GLClientState::VertexAttribState *state = getState(GLClientState::NORMAL_LOCATION);
306             *ptr = state->type;
307             isClientStateParam = true;
308             break;
309             }
310         case GL_NORMAL_ARRAY_STRIDE: {
311             const GLClientState::VertexAttribState *state = getState(GLClientState::NORMAL_LOCATION);
312             *ptr = state->stride;
313             isClientStateParam = true;
314             break;
315             }
316         case GL_TEXTURE_COORD_ARRAY_SIZE: {
317             const GLClientState::VertexAttribState *state = getState(getActiveTexture() + GLClientState::TEXCOORD0_LOCATION);
318             *ptr = state->size;
319             isClientStateParam = true;
320             break;
321             }
322         case GL_TEXTURE_COORD_ARRAY_TYPE: {
323             const GLClientState::VertexAttribState *state = getState(getActiveTexture() + GLClientState::TEXCOORD0_LOCATION);
324             *ptr = state->type;
325             isClientStateParam = true;
326             break;
327             }
328         case GL_TEXTURE_COORD_ARRAY_STRIDE: {
329             const GLClientState::VertexAttribState *state = getState(getActiveTexture() + GLClientState::TEXCOORD0_LOCATION);
330             *ptr = state->stride;
331             isClientStateParam = true;
332             break;
333             }
334         case GL_POINT_SIZE_ARRAY_TYPE_OES: {
335             const GLClientState::VertexAttribState *state = getState(GLClientState::POINTSIZE_LOCATION);
336             *ptr = state->type;
337             isClientStateParam = true;
338             break;
339             }
340         case GL_POINT_SIZE_ARRAY_STRIDE_OES: {
341             const GLClientState::VertexAttribState *state = getState(GLClientState::POINTSIZE_LOCATION);
342             *ptr = state->stride;
343             isClientStateParam = true;
344             break;
345             }
346         case GL_MATRIX_INDEX_ARRAY_SIZE_OES: {
347             const GLClientState::VertexAttribState *state = getState(GLClientState::MATRIXINDEX_LOCATION);
348             *ptr = state->size;
349             isClientStateParam = true;
350             break;
351             }
352         case GL_MATRIX_INDEX_ARRAY_TYPE_OES: {
353             const GLClientState::VertexAttribState *state = getState(GLClientState::MATRIXINDEX_LOCATION);
354             *ptr = state->type;
355             isClientStateParam = true;
356             break;
357             }
358         case GL_MATRIX_INDEX_ARRAY_STRIDE_OES: {
359             const GLClientState::VertexAttribState *state = getState(GLClientState::MATRIXINDEX_LOCATION);
360             *ptr = state->stride;
361             isClientStateParam = true;
362             break;
363             }
364         case GL_WEIGHT_ARRAY_SIZE_OES: {
365             const GLClientState::VertexAttribState *state = getState(GLClientState::WEIGHT_LOCATION);
366             *ptr = state->size;
367             isClientStateParam = true;
368             break;
369             }
370         case GL_WEIGHT_ARRAY_TYPE_OES: {
371             const GLClientState::VertexAttribState *state = getState(GLClientState::WEIGHT_LOCATION);
372             *ptr = state->type;
373             isClientStateParam = true;
374             break;
375             }
376         case GL_WEIGHT_ARRAY_STRIDE_OES: {
377             const GLClientState::VertexAttribState *state = getState(GLClientState::WEIGHT_LOCATION);
378             *ptr = state->stride;
379             isClientStateParam = true;
380             break;
381             }
382         case GL_VERTEX_ARRAY_BUFFER_BINDING: {
383             const GLClientState::VertexAttribState *state = getState(GLClientState::VERTEX_LOCATION);
384             *ptr = state->bufferObject;
385             isClientStateParam = true;
386             break;
387             }
388         case GL_NORMAL_ARRAY_BUFFER_BINDING: {
389             const GLClientState::VertexAttribState *state = getState(GLClientState::NORMAL_LOCATION);
390             *ptr = state->bufferObject;
391             isClientStateParam = true;
392             break;
393             }
394         case GL_COLOR_ARRAY_BUFFER_BINDING: {
395             const GLClientState::VertexAttribState *state = getState(GLClientState::COLOR_LOCATION);
396             *ptr = state->bufferObject;
397             isClientStateParam = true;
398             break;
399             }
400         case GL_TEXTURE_COORD_ARRAY_BUFFER_BINDING: {
401             const GLClientState::VertexAttribState *state = getState(getActiveTexture()+GLClientState::TEXCOORD0_LOCATION);
402             *ptr = state->bufferObject;
403             isClientStateParam = true;
404             break;
405             }
406         case GL_POINT_SIZE_ARRAY_BUFFER_BINDING_OES: {
407             const GLClientState::VertexAttribState *state = getState(GLClientState::POINTSIZE_LOCATION);
408             *ptr = state->bufferObject;
409             isClientStateParam = true;
410             break;
411             }
412         case GL_MATRIX_INDEX_ARRAY_BUFFER_BINDING_OES: {
413             const GLClientState::VertexAttribState *state = getState(GLClientState::MATRIXINDEX_LOCATION);
414             *ptr = state->bufferObject;
415             isClientStateParam = true;
416             break;
417             }
418         case GL_WEIGHT_ARRAY_BUFFER_BINDING_OES: {
419             const GLClientState::VertexAttribState *state = getState(GLClientState::WEIGHT_LOCATION);
420             *ptr = state->bufferObject;
421             isClientStateParam = true;
422             break;
423             }
424         case GL_ARRAY_BUFFER_BINDING: {
425             int buffer = getBuffer(GL_ARRAY_BUFFER);
426             *ptr = buffer;
427             isClientStateParam = true;
428             break;
429             }
430         case GL_ELEMENT_ARRAY_BUFFER_BINDING: {
431             int buffer = getBuffer(GL_ELEMENT_ARRAY_BUFFER);
432             *ptr = buffer;
433             isClientStateParam = true;
434             break;
435             }
436         }
437         return isClientStateParam;
438     }
439 
440 };
441 #endif
442