/**
 **
 ** 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);
}