/* libs/opengles/state.cpp ** ** Copyright 2006, The Android Open Source Project ** ** Licensed under the Apache License, Version 2.0 (the "License"); ** you may not use this file except in compliance with the License. ** You may obtain a copy of the License at ** ** http://www.apache.org/licenses/LICENSE-2.0 ** ** Unless required by applicable law or agreed to in writing, software ** distributed under the License is distributed on an "AS IS" BASIS, ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ** See the License for the specific language governing permissions and ** limitations under the License. */ #include <stdlib.h> #include "context.h" #include "fp.h" #include "state.h" #include "array.h" #include "matrix.h" #include "vertex.h" #include "light.h" #include "texture.h" #include "BufferObjectManager.h" #include "TextureObjectManager.h" namespace android { // ---------------------------------------------------------------------------- static char const * const gVendorString = "Android"; static char const * const gRendererString = "Android PixelFlinger 1.4"; static char const * const gVersionString = "OpenGL ES-CM 1.0"; static char const * const gExtensionsString = "GL_OES_byte_coordinates " // OK "GL_OES_fixed_point " // OK "GL_OES_single_precision " // OK "GL_OES_read_format " // OK "GL_OES_compressed_paletted_texture " // OK "GL_OES_draw_texture " // OK "GL_OES_matrix_get " // OK "GL_OES_query_matrix " // OK // "GL_OES_point_size_array " // TODO // "GL_OES_point_sprite " // TODO "GL_OES_EGL_image " // OK #ifdef GL_OES_compressed_ETC1_RGB8_texture "GL_OES_compressed_ETC1_RGB8_texture " // OK #endif "GL_ARB_texture_compression " // OK "GL_ARB_texture_non_power_of_two " // OK "GL_ANDROID_user_clip_plane " // OK "GL_ANDROID_vertex_buffer_object " // OK "GL_ANDROID_generate_mipmap " // OK ; // ---------------------------------------------------------------------------- #if 0 #pragma mark - #endif ogles_context_t *ogles_init(size_t extra) { void* const base = malloc(extra + sizeof(ogles_context_t) + 32); if (!base) return 0; ogles_context_t *c = (ogles_context_t *)((ptrdiff_t(base) + extra + 31) & ~0x1FL); memset(c, 0, sizeof(ogles_context_t)); ggl_init_context(&(c->rasterizer)); // XXX: this should be passed as an argument sp<EGLSurfaceManager> smgr(new EGLSurfaceManager()); c->surfaceManager = smgr.get(); c->surfaceManager->incStrong(c); sp<EGLBufferObjectManager> bomgr(new EGLBufferObjectManager()); c->bufferObjectManager = bomgr.get(); c->bufferObjectManager->incStrong(c); ogles_init_array(c); ogles_init_matrix(c); ogles_init_vertex(c); ogles_init_light(c); ogles_init_texture(c); c->rasterizer.base = base; c->point.size = TRI_ONE; c->line.width = TRI_ONE; // in OpenGL, writing to the depth buffer is enabled by default. c->rasterizer.procs.depthMask(c, 1); // OpenGL enables dithering by default c->rasterizer.procs.enable(c, GL_DITHER); return c; } void ogles_uninit(ogles_context_t* c) { ogles_uninit_array(c); ogles_uninit_matrix(c); ogles_uninit_vertex(c); ogles_uninit_light(c); ogles_uninit_texture(c); c->surfaceManager->decStrong(c); c->bufferObjectManager->decStrong(c); ggl_uninit_context(&(c->rasterizer)); free(c->rasterizer.base); } void _ogles_error(ogles_context_t* c, GLenum error) { if (c->error == GL_NO_ERROR) c->error = error; } static bool stencilop_valid(GLenum op) { switch (op) { case GL_KEEP: case GL_ZERO: case GL_REPLACE: case GL_INCR: case GL_DECR: case GL_INVERT: return true; } return false; } static void enable_disable(ogles_context_t* c, GLenum cap, int enabled) { if ((cap >= GL_LIGHT0) && (cap<GL_LIGHT0+OGLES_MAX_LIGHTS)) { c->lighting.lights[cap-GL_LIGHT0].enable = enabled; c->lighting.enabledLights &= ~(1<<(cap-GL_LIGHT0)); c->lighting.enabledLights |= (enabled<<(cap-GL_LIGHT0)); return; } switch (cap) { case GL_POINT_SMOOTH: c->point.smooth = enabled; break; case GL_LINE_SMOOTH: c->line.smooth = enabled; break; case GL_POLYGON_OFFSET_FILL: c->polygonOffset.enable = enabled; break; case GL_CULL_FACE: c->cull.enable = enabled; break; case GL_LIGHTING: c->lighting.enable = enabled; break; case GL_COLOR_MATERIAL: c->lighting.colorMaterial.enable = enabled; break; case GL_NORMALIZE: case GL_RESCALE_NORMAL: c->transforms.rescaleNormals = enabled ? cap : 0; // XXX: invalidate mvit break; case GL_CLIP_PLANE0: case GL_CLIP_PLANE1: case GL_CLIP_PLANE2: case GL_CLIP_PLANE3: case GL_CLIP_PLANE4: case GL_CLIP_PLANE5: c->clipPlanes.enable &= ~(1<<(cap-GL_CLIP_PLANE0)); c->clipPlanes.enable |= (enabled<<(cap-GL_CLIP_PLANE0)); ogles_invalidate_perspective(c); break; case GL_FOG: case GL_DEPTH_TEST: ogles_invalidate_perspective(c); // fall-through... case GL_BLEND: case GL_SCISSOR_TEST: case GL_ALPHA_TEST: case GL_COLOR_LOGIC_OP: case GL_DITHER: case GL_STENCIL_TEST: case GL_TEXTURE_2D: // these need to fall through into the rasterizer c->rasterizer.procs.enableDisable(c, cap, enabled); break; case GL_MULTISAMPLE: case GL_SAMPLE_ALPHA_TO_COVERAGE: case GL_SAMPLE_ALPHA_TO_ONE: case GL_SAMPLE_COVERAGE: // not supported in this implementation break; default: ogles_error(c, GL_INVALID_ENUM); return; } } // ---------------------------------------------------------------------------- }; // namespace android // ---------------------------------------------------------------------------- using namespace android; #if 0 #pragma mark - #endif // These ones are super-easy, we're not supporting those features! void glSampleCoverage(GLclampf value, GLboolean invert) { } void glSampleCoveragex(GLclampx value, GLboolean invert) { } void glStencilFunc(GLenum func, GLint ref, GLuint mask) { ogles_context_t* c = ogles_context_t::get(); if (func < GL_NEVER || func > GL_ALWAYS) { ogles_error(c, GL_INVALID_ENUM); return; } // from OpenGL|ES 1.0 sepcification: // If there is no stencil buffer, no stencil modification can occur // and it is as if the stencil test always passes. } void glStencilOp(GLenum fail, GLenum zfail, GLenum zpass) { ogles_context_t* c = ogles_context_t::get(); if ((stencilop_valid(fail) & stencilop_valid(zfail) & stencilop_valid(zpass)) == 0) { ogles_error(c, GL_INVALID_ENUM); return; } } // ---------------------------------------------------------------------------- void glAlphaFunc(GLenum func, GLclampf ref) { glAlphaFuncx(func, gglFloatToFixed(ref)); } void glCullFace(GLenum mode) { ogles_context_t* c = ogles_context_t::get(); switch (mode) { case GL_FRONT: case GL_BACK: case GL_FRONT_AND_BACK: break; default: ogles_error(c, GL_INVALID_ENUM); } c->cull.cullFace = mode; } void glFrontFace(GLenum mode) { ogles_context_t* c = ogles_context_t::get(); switch (mode) { case GL_CW: case GL_CCW: break; default: ogles_error(c, GL_INVALID_ENUM); return; } c->cull.frontFace = mode; } void glHint(GLenum target, GLenum mode) { ogles_context_t* c = ogles_context_t::get(); switch (target) { case GL_FOG_HINT: case GL_GENERATE_MIPMAP_HINT: case GL_LINE_SMOOTH_HINT: break; case GL_POINT_SMOOTH_HINT: c->rasterizer.procs.enableDisable(c, GGL_POINT_SMOOTH_NICE, mode==GL_NICEST); break; case GL_PERSPECTIVE_CORRECTION_HINT: c->perspective = (mode == GL_NICEST) ? 1 : 0; break; default: ogles_error(c, GL_INVALID_ENUM); } } void glEnable(GLenum cap) { ogles_context_t* c = ogles_context_t::get(); enable_disable(c, cap, 1); } void glDisable(GLenum cap) { ogles_context_t* c = ogles_context_t::get(); enable_disable(c, cap, 0); } void glFinish() { // nothing to do for our software implementation } void glFlush() { // nothing to do for our software implementation } GLenum glGetError() { // From OpenGL|ES 1.0 specification: // If more than one flag has recorded an error, glGetError returns // and clears an arbitrary error flag value. Thus, glGetError should // always be called in a loop, until it returns GL_NO_ERROR, // if all error flags are to be reset. ogles_context_t* c = ogles_context_t::get(); if (c->error) { const GLenum ret(c->error); c->error = 0; return ret; } if (c->rasterizer.error) { const GLenum ret(c->rasterizer.error); c->rasterizer.error = 0; return ret; } return GL_NO_ERROR; } const GLubyte* glGetString(GLenum string) { switch (string) { case GL_VENDOR: return (const GLubyte*)gVendorString; case GL_RENDERER: return (const GLubyte*)gRendererString; case GL_VERSION: return (const GLubyte*)gVersionString; case GL_EXTENSIONS: return (const GLubyte*)gExtensionsString; } ogles_context_t* c = ogles_context_t::get(); ogles_error(c, GL_INVALID_ENUM); return 0; } void glGetIntegerv(GLenum pname, GLint *params) { int i; ogles_context_t* c = ogles_context_t::get(); switch (pname) { case GL_ALIASED_POINT_SIZE_RANGE: params[0] = 0; params[1] = GGL_MAX_ALIASED_POINT_SIZE; break; case GL_ALIASED_LINE_WIDTH_RANGE: params[0] = 0; params[1] = GGL_MAX_ALIASED_POINT_SIZE; break; case GL_ALPHA_BITS: { int index = c->rasterizer.state.buffers.color.format; GGLFormat const * formats = gglGetPixelFormatTable(); params[0] = formats[index].ah - formats[index].al; break; } case GL_RED_BITS: { int index = c->rasterizer.state.buffers.color.format; GGLFormat const * formats = gglGetPixelFormatTable(); params[0] = formats[index].rh - formats[index].rl; break; } case GL_GREEN_BITS: { int index = c->rasterizer.state.buffers.color.format; GGLFormat const * formats = gglGetPixelFormatTable(); params[0] = formats[index].gh - formats[index].gl; break; } case GL_BLUE_BITS: { int index = c->rasterizer.state.buffers.color.format; GGLFormat const * formats = gglGetPixelFormatTable(); params[0] = formats[index].bh - formats[index].bl; break; } case GL_COMPRESSED_TEXTURE_FORMATS: params[ 0] = GL_PALETTE4_RGB8_OES; params[ 1] = GL_PALETTE4_RGBA8_OES; params[ 2] = GL_PALETTE4_R5_G6_B5_OES; params[ 3] = GL_PALETTE4_RGBA4_OES; params[ 4] = GL_PALETTE4_RGB5_A1_OES; params[ 5] = GL_PALETTE8_RGB8_OES; params[ 6] = GL_PALETTE8_RGBA8_OES; params[ 7] = GL_PALETTE8_R5_G6_B5_OES; params[ 8] = GL_PALETTE8_RGBA4_OES; params[ 9] = GL_PALETTE8_RGB5_A1_OES; i = 10; #ifdef GL_OES_compressed_ETC1_RGB8_texture params[i++] = GL_ETC1_RGB8_OES; #endif break; case GL_DEPTH_BITS: params[0] = c->rasterizer.state.buffers.depth.format ? 0 : 16; break; case GL_IMPLEMENTATION_COLOR_READ_FORMAT_OES: params[0] = GL_RGB; break; case GL_IMPLEMENTATION_COLOR_READ_TYPE_OES: params[0] = GL_UNSIGNED_SHORT_5_6_5; break; case GL_MAX_LIGHTS: params[0] = OGLES_MAX_LIGHTS; break; case GL_MAX_CLIP_PLANES: params[0] = OGLES_MAX_CLIP_PLANES; break; case GL_MAX_MODELVIEW_STACK_DEPTH: params[0] = OGLES_MODELVIEW_STACK_DEPTH; break; case GL_MAX_PROJECTION_STACK_DEPTH: params[0] = OGLES_PROJECTION_STACK_DEPTH; break; case GL_MAX_TEXTURE_STACK_DEPTH: params[0] = OGLES_TEXTURE_STACK_DEPTH; break; case GL_MAX_TEXTURE_SIZE: params[0] = GGL_MAX_TEXTURE_SIZE; break; case GL_MAX_TEXTURE_UNITS: params[0] = GGL_TEXTURE_UNIT_COUNT; break; case GL_MAX_VIEWPORT_DIMS: params[0] = GGL_MAX_VIEWPORT_DIMS; params[1] = GGL_MAX_VIEWPORT_DIMS; break; case GL_NUM_COMPRESSED_TEXTURE_FORMATS: params[0] = OGLES_NUM_COMPRESSED_TEXTURE_FORMATS; break; case GL_SMOOTH_LINE_WIDTH_RANGE: params[0] = 0; params[1] = GGL_MAX_SMOOTH_LINE_WIDTH; break; case GL_SMOOTH_POINT_SIZE_RANGE: params[0] = 0; params[1] = GGL_MAX_SMOOTH_POINT_SIZE; break; case GL_STENCIL_BITS: params[0] = 0; break; case GL_SUBPIXEL_BITS: params[0] = GGL_SUBPIXEL_BITS; break; case GL_MODELVIEW_MATRIX_FLOAT_AS_INT_BITS_OES: memcpy( params, c->transforms.modelview.top().elements(), 16*sizeof(GLint)); break; case GL_PROJECTION_MATRIX_FLOAT_AS_INT_BITS_OES: memcpy( params, c->transforms.projection.top().elements(), 16*sizeof(GLint)); break; case GL_TEXTURE_MATRIX_FLOAT_AS_INT_BITS_OES: memcpy( params, c->transforms.texture[c->textures.active].top().elements(), 16*sizeof(GLint)); break; default: ogles_error(c, GL_INVALID_ENUM); break; } } // ---------------------------------------------------------------------------- void glPointSize(GLfloat size) { ogles_context_t* c = ogles_context_t::get(); if (size <= 0) { ogles_error(c, GL_INVALID_ENUM); return; } c->point.size = TRI_FROM_FIXED(gglFloatToFixed(size)); } void glPointSizex(GLfixed size) { ogles_context_t* c = ogles_context_t::get(); if (size <= 0) { ogles_error(c, GL_INVALID_ENUM); return; } c->point.size = TRI_FROM_FIXED(size); } // ---------------------------------------------------------------------------- void glLineWidth(GLfloat width) { ogles_context_t* c = ogles_context_t::get(); if (width <= 0) { ogles_error(c, GL_INVALID_ENUM); return; } c->line.width = TRI_FROM_FIXED(gglFloatToFixed(width)); } void glLineWidthx(GLfixed width) { ogles_context_t* c = ogles_context_t::get(); if (width <= 0) { ogles_error(c, GL_INVALID_ENUM); return; } c->line.width = TRI_FROM_FIXED(width); } // ---------------------------------------------------------------------------- void glColorMask(GLboolean r, GLboolean g, GLboolean b, GLboolean a) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.colorMask(c, r, g, b, a); } void glDepthMask(GLboolean flag) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.depthMask(c, flag); } void glStencilMask(GLuint mask) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.stencilMask(c, mask); } void glDepthFunc(GLenum func) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.depthFunc(c, func); } void glLogicOp(GLenum opcode) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.logicOp(c, opcode); } void glAlphaFuncx(GLenum func, GLclampx ref) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.alphaFuncx(c, func, ref); } void glBlendFunc(GLenum sfactor, GLenum dfactor) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.blendFunc(c, sfactor, dfactor); } void glClear(GLbitfield mask) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.clear(c, mask); } void glClearColorx(GLclampx red, GLclampx green, GLclampx blue, GLclampx alpha) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.clearColorx(c, red, green, blue, alpha); } void glClearColor(GLclampf r, GLclampf g, GLclampf b, GLclampf a) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.clearColorx(c, gglFloatToFixed(r), gglFloatToFixed(g), gglFloatToFixed(b), gglFloatToFixed(a)); } void glClearDepthx(GLclampx depth) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.clearDepthx(c, depth); } void glClearDepthf(GLclampf depth) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.clearDepthx(c, gglFloatToFixed(depth)); } void glClearStencil(GLint s) { ogles_context_t* c = ogles_context_t::get(); c->rasterizer.procs.clearStencil(c, s); }