/**
**
** Copyright 2010, 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 "pixelflinger2.h"
#include "src/talloc/hieralloc.h"
#include <string>
void gglError(unsigned error)
{
std::string str;
if (GL_NO_ERROR == error)
return;
LOGD("\n*\n*\n pf2: gglError 0x%.4X \n*\n*\n", error);
assert(0);
}
static void DepthRangef(GGLInterface * iface, GLclampf zNear, GLclampf zFar)
{
GGL_GET_CONTEXT(ctx, iface);
ctx->viewport.n = VectorComp_t_CTR((zNear + zFar) / 2);
ctx->viewport.f = VectorComp_t_CTR((zFar - zNear) / 2);
}
static void Viewport(GGLInterface * iface, GLint x, GLint y, GLsizei width, GLsizei height)
{
GGL_GET_CONTEXT(ctx, iface);
ctx->viewport.x = VectorComp_t_CTR(x + width / 2);
ctx->viewport.y = VectorComp_t_CTR(y + height / 2);
ctx->viewport.w = VectorComp_t_CTR(width / 2);
ctx->viewport.h = VectorComp_t_CTR(height / 2);
}
static void CullFace(GGLInterface * iface, GLenum mode)
{
GGL_GET_CONTEXT(ctx, iface);
if (GL_FRONT > mode || GL_FRONT_AND_BACK < mode)
gglError(GL_INVALID_ENUM);
else
ctx->cullState.cullFace = mode - GL_FRONT;
}
static void FrontFace(GGLInterface * iface, GLenum mode)
{
GGL_GET_CONTEXT(ctx, iface);
if (GL_CW > mode || GL_CCW < mode)
gglError(GL_INVALID_ENUM);
else
ctx->cullState.frontFace = mode - GL_CW;
}
static void BlendColor(GGLInterface * iface, GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha)
{
GGL_GET_CONTEXT(ctx, iface);
ctx->state.blendState.color[0] = MIN2(MAX2(red * 255, 0.0f), 255.0f);
ctx->state.blendState.color[1] = MIN2(MAX2(green * 255, 0.0f), 255.0f);
ctx->state.blendState.color[2] = MIN2(MAX2(blue * 255, 0.0f), 255.0f);
ctx->state.blendState.color[3] = MIN2(MAX2(alpha * 255, 0.0f), 255.0f);
SetShaderVerifyFunctions(iface);
}
static void BlendEquationSeparate(GGLInterface * iface, GLenum modeRGB, GLenum modeAlpha)
{
GGL_GET_CONTEXT(ctx, iface);
if (GL_FUNC_ADD != modeRGB && (GL_FUNC_SUBTRACT > modeRGB ||
GL_FUNC_REVERSE_SUBTRACT < modeRGB))
return gglError(GL_INVALID_ENUM);
if (GL_FUNC_ADD != modeRGB && (GL_FUNC_SUBTRACT > modeRGB ||
GL_FUNC_REVERSE_SUBTRACT < modeRGB))
return gglError(GL_INVALID_ENUM);
ctx->state.blendState.ce = (GGLBlendState::GGLBlendFunc)(modeRGB - GL_FUNC_ADD);
ctx->state.blendState.ae = (GGLBlendState::GGLBlendFunc)(modeAlpha - GL_FUNC_ADD);
SetShaderVerifyFunctions(iface);
}
static inline GGLBlendState::GGLBlendFactor GLBlendFactor(const GLenum factor)
{
#define SWITCH_LINE(c) case c: return GGLBlendState::G##c;
switch (factor)
{
SWITCH_LINE(GL_ZERO);
SWITCH_LINE(GL_ONE);
SWITCH_LINE(GL_SRC_COLOR);
SWITCH_LINE(GL_ONE_MINUS_SRC_COLOR);
SWITCH_LINE(GL_DST_COLOR);
SWITCH_LINE(GL_ONE_MINUS_DST_COLOR);
SWITCH_LINE(GL_SRC_ALPHA);
SWITCH_LINE(GL_ONE_MINUS_SRC_ALPHA);
SWITCH_LINE(GL_DST_ALPHA);
SWITCH_LINE(GL_ONE_MINUS_DST_ALPHA);
SWITCH_LINE(GL_SRC_ALPHA_SATURATE);
SWITCH_LINE(GL_CONSTANT_COLOR);
SWITCH_LINE(GL_ONE_MINUS_CONSTANT_COLOR);
SWITCH_LINE(GL_CONSTANT_ALPHA);
SWITCH_LINE(GL_ONE_MINUS_CONSTANT_ALPHA);
default: assert(0); return GGLBlendState::GGL_ZERO;
}
#undef SWITCH_LINE
}
static void BlendFuncSeparate(GGLInterface * iface, GLenum srcRGB, GLenum dstRGB, GLenum srcAlpha, GLenum dstAlpha)
{
GGL_GET_CONTEXT(ctx, iface);
if (GL_ZERO != srcRGB && GL_ONE != srcRGB &&
(GL_SRC_COLOR > srcRGB || GL_SRC_ALPHA_SATURATE < srcRGB) &&
(GL_CONSTANT_COLOR > srcRGB || GL_ONE_MINUS_CONSTANT_ALPHA < srcRGB))
return gglError(GL_INVALID_ENUM);
if (GL_ZERO != srcAlpha && GL_ONE != srcAlpha &&
(GL_SRC_COLOR > srcAlpha || GL_SRC_ALPHA_SATURATE < srcAlpha) &&
(GL_CONSTANT_COLOR > dstRGB || GL_ONE_MINUS_CONSTANT_ALPHA < dstRGB))
return gglError(GL_INVALID_ENUM);
if (GL_ZERO != dstRGB && GL_ONE != dstRGB &&
(GL_SRC_COLOR > dstRGB || GL_ONE_MINUS_DST_COLOR < dstRGB) && // GL_SRC_ALPHA_SATURATE only for source
(GL_CONSTANT_COLOR > dstRGB || GL_ONE_MINUS_CONSTANT_ALPHA < dstRGB))
return gglError(GL_INVALID_ENUM);
if (GL_ZERO != dstAlpha && GL_ONE != dstAlpha &&
(GL_SRC_COLOR > dstAlpha || GL_ONE_MINUS_DST_COLOR < dstAlpha) &&
(GL_CONSTANT_COLOR > dstRGB || GL_ONE_MINUS_CONSTANT_ALPHA < dstRGB))
return gglError(GL_INVALID_ENUM);
if (srcAlpha == GL_SRC_ALPHA_SATURATE) // it's just 1 instead of min(sa, 1 - da) for alpha channel
srcAlpha = GL_ONE;
// in c++ it's templated function for color and alpha,
// so it requires setting srcAlpha to GL_ONE to run template again only for alpha
ctx->state.blendState.scf = GLBlendFactor(srcRGB);
ctx->state.blendState.saf = GLBlendFactor(srcAlpha);
ctx->state.blendState.dcf = GLBlendFactor(dstRGB);
ctx->state.blendState.daf = GLBlendFactor(dstAlpha);
SetShaderVerifyFunctions(iface);
}
static void EnableDisable(GGLInterface * iface, GLenum cap, GLboolean enable)
{
GGL_GET_CONTEXT(ctx, iface);
bool changed = false;
switch (cap) {
case GL_BLEND:
changed |= ctx->state.blendState.enable ^ enable;
ctx->state.blendState.enable = enable;
break;
case GL_CULL_FACE:
changed |= ctx->cullState.enable ^ enable;
ctx->cullState.enable = enable;
break;
case GL_DEPTH_TEST:
changed |= ctx->state.bufferState.depthTest ^ enable;
ctx->state.bufferState.depthTest = enable;
break;
case GL_STENCIL_TEST:
changed |= ctx->state.bufferState.stencilTest ^ enable;
ctx->state.bufferState.stencilTest = enable;
break;
case GL_DITHER:
// LOGD("pf2: EnableDisable GL_DITHER \n");
break;
case GL_SCISSOR_TEST:
// LOGD("pf2: EnableDisable GL_SCISSOR_TEST \n");
break;
case GL_TEXTURE_2D:
// LOGD("pf2: EnableDisable GL_SCISSOR_TEST %d", enable);
break;
default:
LOGD("pf2: EnableDisable 0x%.4X causes GL_INVALID_ENUM (maybe not implemented or ES 1.0) \n", cap);
// gglError(GL_INVALID_ENUM);
assert(0);
break;
}
if (changed)
SetShaderVerifyFunctions(iface);
}
void InitializeGGLState(GGLInterface * iface)
{
#if USE_DUAL_THREAD
reinterpret_cast<GGLContext *>(iface)->worker = GGLContext::Worker();
#endif
iface->DepthRangef = DepthRangef;
iface->Viewport = Viewport;
iface->CullFace = CullFace;
iface->FrontFace = FrontFace;
iface->BlendColor = BlendColor;
iface->BlendEquationSeparate = BlendEquationSeparate;
iface->BlendFuncSeparate = BlendFuncSeparate;
iface->EnableDisable = EnableDisable;
InitializeBufferFunctions(iface);
InitializeRasterFunctions(iface);
InitializeScanLineFunctions(iface);
InitializeShaderFunctions(iface);
InitializeTextureFunctions(iface);
iface->EnableDisable(iface, GL_DEPTH_TEST, false);
iface->DepthFunc(iface, GL_LESS);
iface->ClearColor(iface, 0, 0, 0, 0);
iface->ClearDepthf(iface, 1.0f);
iface->EnableDisable(iface, GL_STENCIL_TEST, false);
iface->StencilFuncSeparate(iface, GL_FRONT_AND_BACK, GL_ALWAYS, 0, 0xff);
iface->StencilOpSeparate(iface, GL_FRONT_AND_BACK, GL_KEEP, GL_KEEP, GL_KEEP);
iface->FrontFace(iface, GL_CCW);
iface->CullFace(iface, GL_BACK);
iface->EnableDisable(iface, GL_CULL_FACE, false);
iface->EnableDisable(iface, GL_BLEND, false);
iface->BlendColor(iface, 0, 0, 0, 0);
iface->BlendEquationSeparate(iface, GL_FUNC_ADD, GL_FUNC_ADD);
iface->BlendFuncSeparate(iface, GL_ONE, GL_ZERO, GL_ONE, GL_ZERO);
for (unsigned i = 0; i < GGL_MAXCOMBINEDTEXTUREIMAGEUNITS; i++)
iface->SetSampler(iface, i, NULL);
iface->SetBuffer(iface, GL_COLOR_BUFFER_BIT, NULL);
iface->SetBuffer(iface, GL_DEPTH_BUFFER_BIT, NULL);
iface->SetBuffer(iface, GL_STENCIL_BUFFER_BIT, NULL);
SetShaderVerifyFunctions(iface);
}
GGLInterface * CreateGGLInterface()
{
GGLContext * const ctx = (GGLContext *)calloc(1, sizeof(GGLContext));
if (!ctx)
return NULL;
assert((void *)ctx == (void *)&ctx->interface);
//_glapi_set_context(ctx->glCtx);
//_mesa_init_constants(&Const);
puts("InitializeGGLState");
InitializeGGLState(&ctx->interface);
return &ctx->interface;
}
void UninitializeGGLState(GGLInterface * iface)
{
#if USE_DUAL_THREAD
reinterpret_cast<GGLContext *>(iface)->worker.~Worker();
#endif
DestroyShaderFunctions(iface);
#if USE_LLVM_TEXTURE_SAMPLER
puts("USE_LLVM_TEXTURE_SAMPLER");
#endif
#if USE_LLVM_SCANLINE
puts("USE_LLVM_SCANLINE");
#endif
#if USE_LLVM_EXECUTIONENGINE
puts("USE_LLVM_EXECUTIONENGINE");
#endif
#if USE_DUAL_THREAD
puts("USE_DUAL_THREAD");
#endif
hieralloc_report_brief(NULL, stdout);
}
void DestroyGGLInterface(GGLInterface * iface)
{
GGLContext * const ctx = reinterpret_cast<GGLContext *>(iface);
UninitializeGGLState(iface);
free(ctx);
}