/*
 * (C) Copyright IBM Corporation 2006
 * All Rights Reserved.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * on the rights to use, copy, modify, merge, publish, distribute, sub
 * license, and/or sell copies of the Software, and to permit persons to whom
 * the Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.  IN NO EVENT SHALL
 * IBM AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 */

/**
 * \file prog_parameter.c
 * 
 * Test various aspects of setting (and getting) low-level program parameters.
 * This is primarilly intended as a test for GL_EXT_gpu_program_parameters,
 * but it turns out that it hits some other functionality along the way.
 * 
 * \author Ian Romanick <idr@us.ibm.com>
 */

#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include <GL/glut.h>

#ifndef GL_EXT_gpu_program_parameters
typedef void (APIENTRYP PFNGLPROGRAMLOCALPARAMETERS4FVEXTPROC)(GLenum,
    GLuint, GLsizei, const GLfloat *);
typedef void (APIENTRYP PFNGLPROGRAMENVPARAMETERS4FVEXTPROC)(GLenum,
    GLuint, GLsizei, const GLfloat *);
#endif

static PFNGLPROGRAMLOCALPARAMETER4FVARBPROC program_local_parameter4fv = NULL;
static PFNGLGETPROGRAMLOCALPARAMETERFVARBPROC get_program_local_parameterfv = NULL;
static PFNGLPROGRAMENVPARAMETER4FVARBPROC program_env_parameter4fv = NULL;
static PFNGLGETPROGRAMENVPARAMETERFVARBPROC get_program_env_parameterfv = NULL;
static PFNGLBINDPROGRAMARBPROC bind_program = NULL;
static PFNGLGETPROGRAMIVARBPROC get_program = NULL;

static PFNGLPROGRAMLOCALPARAMETERS4FVEXTPROC program_local_parameters4fv = NULL;
static PFNGLPROGRAMENVPARAMETERS4FVEXTPROC program_env_parameters4fv = NULL;

static int Width = 400;
static int Height = 200;
static const GLfloat Near = 5.0, Far = 25.0;


static void Display( void )
{
}


static void Idle( void )
{
}


static void Visible( int vis )
{
   if ( vis == GLUT_VISIBLE ) {
      glutIdleFunc( Idle );
   }
   else {
      glutIdleFunc( NULL );
   }
}
static void Reshape( int width, int height )
{
   GLfloat ar = (float) width / (float) height;
   Width = width;
   Height = height;
   glViewport( 0, 0, width, height );
   glMatrixMode( GL_PROJECTION );
   glLoadIdentity();
   glFrustum( -ar, ar, -1.0, 1.0, Near, Far );
}


static void Key( unsigned char key, int x, int y )
{
   (void) x;
   (void) y;
   switch (key) {
      case 27:
         exit(0);
         break;
   }
   glutPostRedisplay();
}


static int set_parameter_batch( GLsizei count, GLfloat * param,
				const char * name,
				PFNGLPROGRAMLOCALPARAMETER4FVARBPROC set_parameter,
				PFNGLPROGRAMLOCALPARAMETERS4FVEXTPROC set_parameters,
				PFNGLGETPROGRAMLOCALPARAMETERFVARBPROC get_parameter
				)
{
   unsigned i;
   int pass = 1;


   for ( i = 0 ; i < (4 * count) ; i++ ) {
      param[i] = (GLfloat) random() / (GLfloat) random();
   }

   /* Try using the "classic" interface.
    */
   printf("Testing glProgram%sParameter4fvARB (count = %u)...\n", name, count);
   for ( i = 0 ; i < count ; i++ ) {
      (*set_parameter)(GL_VERTEX_PROGRAM_ARB, i, & param[i * 4]);
   }

   for ( i = 0 ; i < count ; i++ ) {
      GLfloat temp[4];

      (*get_parameter)(GL_VERTEX_PROGRAM_ARB, i, temp);

      if ( (temp[0] != param[(i * 4) + 0])
	   || (temp[1] != param[(i * 4) + 1])
	   || (temp[2] != param[(i * 4) + 2])
	   || (temp[3] != param[(i * 4) + 3]) ) {
	 printf("Mismatch in glProgram%sParameter4fvARB index %u!\n", name, i);
	 printf("Got { %f, %f, %f, %f }, expected { %f, %f, %f, %f }!\n",
		temp[0],            temp[1],
		temp[2],            temp[3],
		param[(i * 4) + 0], param[(i * 4) + 1],
		param[(i * 4) + 2], param[(i * 4) + 3]);
	 pass = 0;
	 break;
      }
   }


   if ( set_parameters == NULL ) {
      return pass;
   }


   for ( i = 0 ; i < (4 * count) ; i++ ) {
      param[i] = (GLfloat) random() / (GLfloat) random();
   }

   printf("Testing glProgram%sParameters4fvEXT (count = %u)...\n", name, count);
   (*set_parameters)(GL_VERTEX_PROGRAM_ARB, 0, count, param);

   for ( i = 0 ; i < count ; i++ ) {
      GLfloat temp[4];

      (*get_parameter)(GL_VERTEX_PROGRAM_ARB, i, temp);

      if ( (temp[0] != param[(i * 4) + 0])
	   || (temp[1] != param[(i * 4) + 1])
	   || (temp[2] != param[(i * 4) + 2])
	   || (temp[3] != param[(i * 4) + 3]) ) {
	 printf("Mismatch in glProgram%sParameters4fvEXT index %u!\n", name, i);
	 printf("Got { %f, %f, %f, %f }, expected { %f, %f, %f, %f }!\n",
		temp[0],            temp[1],
		temp[2],            temp[3],
		param[(i * 4) + 0], param[(i * 4) + 1],
		param[(i * 4) + 2], param[(i * 4) + 3]);
	 pass = 0;
	 break;
      }
   }


   return pass;
}


static void Init( void )
{
   const char * const ver_string = (const char * const)
       glGetString( GL_VERSION );
   int pass = 1;
   GLfloat * params;
   GLint max_program_env_parameters;
   GLint max_program_local_parameters;


   printf("GL_RENDERER = %s\n", (char *) glGetString(GL_RENDERER));
   printf("GL_VERSION = %s\n\n", ver_string);

   if ( !glutExtensionSupported("GL_ARB_vertex_program") ) {
      printf("Sorry, this program requires GL_ARB_vertex_program\n");
      exit(2);
   }


   program_local_parameter4fv = glutGetProcAddress( "glProgramLocalParameter4fvARB" );
   program_env_parameter4fv = glutGetProcAddress( "glProgramEnvParameter4fvARB" );

   get_program_local_parameterfv = glutGetProcAddress( "glGetProgramLocalParameterfvARB" );
   get_program_env_parameterfv = glutGetProcAddress( "glGetProgramEnvParameterfvARB" );

   bind_program = glutGetProcAddress( "glBindProgramARB" );
   get_program = glutGetProcAddress( "glGetProgramivARB" );

   if ( glutExtensionSupported("GL_EXT_gpu_program_parameters") ) {
      printf("GL_EXT_gpu_program_parameters available, testing that path.\n");

      program_local_parameters4fv = glutGetProcAddress( "glProgramLocalParameters4fvEXT" );
      program_env_parameters4fv = glutGetProcAddress( "glProgramEnvParameters4fvEXT" );
   }
   else {
      printf("GL_EXT_gpu_program_parameters not available.\n");

      program_local_parameters4fv = NULL;
      program_env_parameters4fv = NULL;
   }



   /* Since the test sets program local parameters, a program must be bound.
    * Program source, however, is not needed.
    */
   (*bind_program)(GL_VERTEX_PROGRAM_ARB, 1);


   (*get_program)(GL_VERTEX_PROGRAM_ARB, GL_MAX_PROGRAM_ENV_PARAMETERS_ARB,
		  & max_program_env_parameters);

   params = malloc(max_program_env_parameters * 4 * sizeof(GLfloat));

   pass &= set_parameter_batch(max_program_env_parameters, params, "Env",
			       program_env_parameter4fv,
			       program_env_parameters4fv,
			       get_program_env_parameterfv);


   (*get_program)(GL_VERTEX_PROGRAM_ARB, GL_MAX_PROGRAM_LOCAL_PARAMETERS_ARB,
		  & max_program_local_parameters);

   if (max_program_local_parameters > max_program_env_parameters) {
      params = realloc(params,
		       max_program_local_parameters * 4 * sizeof(GLfloat));
   }

   pass &= set_parameter_batch(max_program_local_parameters, params, "Local",
			       program_local_parameter4fv,
			       program_local_parameters4fv,
			       get_program_local_parameterfv);

   free(params);

   if (! pass) {
      printf("FAIL!\n");
      exit(1);
   }
   
   printf("PASS!\n");
}


int main( int argc, char *argv[] )
{
   glutInit( &argc, argv );
   glutInitWindowPosition( 0, 0 );
   glutInitWindowSize( Width, Height );
   glutInitDisplayMode( GLUT_RGB );
   glutCreateWindow( "Program Parameters Test" );
   glutReshapeFunc( Reshape );
   glutKeyboardFunc( Key );
   glutDisplayFunc( Display );
   glutVisibilityFunc( Visible );

   Init();

   return 0;
}