Sophie

Sophie

distrib > * > cooker > x86_64 > by-pkgid > 0243c8b7bca94179c78b9bd6ac76c033 > files > 317

cg-examples-3.0.0018-0.1.x86_64.rpm


/* cgfx_buffer_lighting.cpp - OpenGL-based Cg 3.0 example showing use
   of buffers for material, lighting, and transforms in CgFX effects. */

/* Requires the OpenGL Utility Toolkit (GLUT) and Cg runtime (version
   2.0 or higher). */

#include <assert.h>   /* for assert */
#include <stdio.h>    /* for printf and NULL */
#include <stdlib.h>   /* for exit */
#include <string.h>   /* for memset */
#include <math.h>     /* for sin and cos */

#include <GL/glew.h>

#ifdef __APPLE__
#include <GLUT/glut.h>
#else
#include <GL/glut.h>
#endif

#ifdef _WIN32
#include <GL/wglew.h>
#else
#ifdef __APPLE__
#include <OpenGL/OpenGL.h>
#else
#include <GL/glxew.h>
#endif
#endif

#include <Cg/cg.h>    /* Can't include this?  Is Cg Toolkit installed? */
#include <Cg/cgGL.h>

#include "matrix.h"
#include "materials.h"
#include "request_vsync.h"


/* Structures */

typedef float float4x4[16];
typedef struct {
  float4x4 modelview;
  float4x4 inverse_modelview;
  float4x4 modelview_projection;
} Transform;

#define MAX_LIGHTS 8

typedef struct {
  int   enabled[4];
  float ambient[4];
  float diffuse[4];
  float specular[4];
  float k0[4], k1[4], k2[4];
} LightSourceStatic;

typedef struct {
  float global_ambient[4];
  LightSourceStatic source[MAX_LIGHTS];
} LightSet;

typedef struct {
  float position[4];
} LightSourcePerView;

typedef struct {
  LightSourcePerView source[MAX_LIGHTS];
} LightSetPerView;


/* Global Variables */

LightSet lightSet;
LightSetPerView lightSetPerView_world, lightSetPerView_eye;

int currentLight = 0;
int myAnimating = 0;
float eyeAngle = 1.6;
float myProjectionMatrix[16];

/* Cg global variables */

CGcontext myCgContext;
CGeffect  myCgEffect;
CGtechnique myCgTechnique;
CGbuffer transform_buffer, *material_buffer, lightSet_buffer, lightSetPerView_buffer;
int object_material[2] = { 0, 3 };
int transform_buffer_offset = -1;
int lightSetPerView_offset = -1;
int enableSync = 1;  /* Sync buffer swaps to monitor refresh rate. */

const char *myProgramName = "cgfx_buffer_lighting",
           *myEffectFileName = "buffer_lighting.cgfx";


/* Forward declared GLUT callbacks registered by main. */

void display();
void keyboard(unsigned char c, int x, int y);
void menu(int item);
void reshape(int width, int height);

/* Forward declarations */

void checkForCgError(const char *situation);
void parameterBufferInfo(const char *name);
void initLight(LightSet *lightSet, int ndx);
void initBuffers();

void advance(int ignored);
void gl_loadRowMajorMatrix(GLenum matrixMode, const float mat[16]);
void renderLightPositions(const float projection[16], const float view[16]);
void bind_material_buffer(int object);
void update_transform_buffer(Transform *transform);
void drawLitSphere(const float projectionMatrix[16], const float viewMatrix[16], int object, float xTranslate);
void drawLitSpheres(const float projectionMatrix[16], const float viewMatrix[16]);
void drawScene();

int main(int argc, char **argv)
{
  int i;

  glutInitWindowSize(400, 400);
  glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE | GLUT_DEPTH);
  glutInit(&argc, argv);

  for (i=1; i<argc; i++) {
    if (!strcmp("-nosync", argv[i])) {
      enableSync = 0;
    }
  }

  glutCreateWindow(myProgramName);
  glutDisplayFunc(display);
  glutKeyboardFunc(keyboard);
  glutReshapeFunc(reshape);

  /* Initialize OpenGL entry points. */
  if (glewInit()!=GLEW_OK || !GLEW_VERSION_1_1) {
    fprintf(stderr, "%s: Failed to initialize GLEW. OpenGL 1.1 required.\n", myProgramName);    
    exit(1);
  }

  requestSynchronizedSwapBuffers(enableSync);
  glClearColor(0.2, 0.2, 0.2, 0.0);  /* Gray background */
  glEnable(GL_DEPTH_TEST);

  myCgContext = cgCreateContext();
  checkForCgError("creating context");

  cgGLRegisterStates(myCgContext);

  myCgEffect = cgCreateEffectFromFile(myCgContext, myEffectFileName, NULL);
  checkForCgError("creating buffer_lighting.cgfx effect");
  assert(myCgEffect);

  myCgTechnique = cgGetFirstTechnique(myCgEffect);
  while (myCgTechnique && cgValidateTechnique(myCgTechnique) == CG_FALSE) {
    fprintf(stderr, "%s: Technique %s did not validate.  Skipping.\n",
      myProgramName, cgGetTechniqueName(myCgTechnique));
    myCgTechnique = cgGetNextTechnique(myCgTechnique);
  }
  if (myCgTechnique) {
    fprintf(stderr, "%s: Use technique %s.\n",
      myProgramName, cgGetTechniqueName(myCgTechnique));
  } else {
    fprintf(stderr, "%s: No valid technique\n",
      myProgramName);
    exit(1);
  }

  // Get buffer offsets once to use them during rendering.
  transform_buffer_offset = cgGetParameterBufferOffset(cgGetNamedParameter(
                              cgGetPassProgram(cgGetFirstPass(myCgTechnique), 
                                CG_VERTEX_DOMAIN), "cbuffer1_Transform"));

  lightSetPerView_offset = cgGetParameterBufferOffset(cgGetNamedParameter(
                             cgGetPassProgram(cgGetFirstPass(myCgTechnique), 
                                CG_FRAGMENT_DOMAIN), "cbuffer3_LightSetPerView"));

  initBuffers();

  printf("vertex program:\n%s\n\n", cgGetProgramString(cgGetPassProgram(
                           cgGetFirstPass(myCgTechnique), CG_VERTEX_DOMAIN), 
                           CG_COMPILED_PROGRAM));

  printf("fragment program:\n%s", cgGetProgramString(cgGetPassProgram(
                           cgGetFirstPass(myCgTechnique), CG_FRAGMENT_DOMAIN), 
                           CG_COMPILED_PROGRAM));

  glutCreateMenu(menu);
  glutAddMenuEntry("[ ] Animate", ' ');
  glutAddMenuEntry("[1] Update sphere 1", '1');
  glutAddMenuEntry("[2] Update sphere 2", '2');
  glutAddMenuEntry("[3] Change material of sphere 1", '3');
  glutAddMenuEntry("[4] Change material of  sphere 2", '4');
  glutAttachMenu(GLUT_RIGHT_BUTTON);

  glutMainLoop();
  return 0;
}

void display()
{
  if (myAnimating) {
    glutTimerFunc(10, advance, 0);
  }

  glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
  drawScene();
  glutSwapBuffers();
}

void keyboard(unsigned char c, int x, int y)
{
  switch (c) {
    case 27:  /* Esc key */
      /* Demonstrate proper deallocation of Cg runtime data structures.
         Not strictly necessary if we are simply going to exit. */
      for (int i=0; i<materialInfoCount; i++) {
        cgDestroyBuffer(material_buffer[i]);
      }
        
      delete [] material_buffer;

      cgDestroyBuffer(transform_buffer);
      cgDestroyBuffer(lightSet_buffer);
      cgDestroyBuffer(lightSetPerView_buffer);
      
      cgDestroyEffect(myCgEffect);
      cgDestroyContext(myCgContext);
      exit(0);
      return;
    case ' ':
      myAnimating = !myAnimating; /* Toggle */
      if (myAnimating)
        advance(0);
      return;
    case '1':
      currentLight = 0;
      return;
    case '2':
      currentLight = 1;
      return;
    case 'k':
      lightSetPerView_world.source[currentLight].position[1] += 0.1;      
      break;
    case 'j':
      lightSetPerView_world.source[currentLight].position[1] -= 0.1;
      break;
    case 'a':
      lightSetPerView_world.source[currentLight].position[2] += 0.1;
      break;
    case 'z':
      lightSetPerView_world.source[currentLight].position[2] -= 0.1;
      break;
    case 'h':
      lightSetPerView_world.source[currentLight].position[0] -= 0.1;
      break;
    case 'l':
      lightSetPerView_world.source[currentLight].position[0] += 0.1;
      break;
    case '3':
      object_material[0]++;
      if (object_material[0] >= materialInfoCount) {
        object_material[0] = 0;
      }
      break;
    case '4':
      object_material[1]++;
      if (object_material[1] >= materialInfoCount) {
        object_material[1] = 0;
      }
      break;
  }

  glutPostRedisplay();
}

void menu(int item)
{
  /* Pass menu item character code to keyboard callback. */
  keyboard((unsigned char)item, 0, 0);
}

void reshape(int width, int height)
{
  double aspectRatio = (float) width / (float) height;
  double fieldOfView = 70.0; /* Degrees */

  /* Build projection matrix once. */
  makePerspectiveMatrix(fieldOfView, aspectRatio,
                        1.0, 20.0,  /* Znear and Zfar */
                        myProjectionMatrix);
  glViewport(0, 0, width, height);
}

void checkForCgError(const char *situation)
{
  CGerror error;
  const char *string = cgGetLastErrorString(&error);
  
  if (error != CG_NO_ERROR) {
    if (error == CG_COMPILER_ERROR) {
      fprintf(stderr,
             "Program: %s\n"
             "Situation: %s\n"
             "Error: %s\n\n"
             "Cg compiler output...\n%s",
             myProgramName, situation, string,
             cgGetLastListing(myCgContext));
    } else {
      fprintf(stderr,
              "Program: %s\n"
              "Situation: %s\n"
              "Error: %s",
              myProgramName, situation, string);
    }
    exit(1);
  }
}

void parameterBufferInfo(const char *name)
{
  CGparameter param = cgGetNamedEffectParameter(myCgEffect, name);

  printf("%s (0x%p):\n", name, param);
  printf(" index = %d\n", cgGetParameterBufferIndex(param));
  printf(" offset = %d\n", cgGetParameterBufferOffset(param));
}

void initLight(LightSet *lightSet, int ndx)
{
  lightSet->source[ndx].enabled[0] = 1;
  lightSet->source[ndx].ambient[0] = 0;
  lightSet->source[ndx].ambient[1] = 0;
  lightSet->source[ndx].ambient[2] = 0;
  lightSet->source[ndx].ambient[3] = 0;
  lightSet->source[ndx].diffuse[0] = 0.9;
  lightSet->source[ndx].diffuse[1] = 0.9;
  lightSet->source[ndx].diffuse[2] = 0.9;
  lightSet->source[ndx].diffuse[3] = 0.9;
  lightSet->source[ndx].specular[0] = 0.9;
  lightSet->source[ndx].specular[1] = 0.9;
  lightSet->source[ndx].specular[2] = 0.9;
  lightSet->source[ndx].specular[3] = 0;
#if 1
  // Inverse square law attenuation
  lightSet->source[ndx].k0[0] = 0.7;
  lightSet->source[ndx].k1[0] = 0;
  lightSet->source[ndx].k2[0] = 0.001;
#else
  // No attenuation
  lightSet->source[ndx].k0[0] = 1;
  lightSet->source[ndx].k1[0] = 0;
  lightSet->source[ndx].k2[0] = 0;
#endif
}

void initBuffers()
{
  int i;

  memset(&lightSet, 0, sizeof(lightSet));
  lightSet.global_ambient[0] = 0.1;
  lightSet.global_ambient[1] = 0.1;
  lightSet.global_ambient[2] = 0.1;
  lightSet.global_ambient[3] = 0.1;

  initLight(&lightSet, 0);
  initLight(&lightSet, 1);

  lightSetPerView_world.source[0].position[0] = 0;
  lightSetPerView_world.source[0].position[1] = 2;
  lightSetPerView_world.source[0].position[2] = 4;
  lightSetPerView_world.source[0].position[3] = 1;

  lightSetPerView_world.source[1].position[0] = 0;
  lightSetPerView_world.source[1].position[1] = -2;
  lightSetPerView_world.source[1].position[2] = 4;
  lightSetPerView_world.source[1].position[3] = 1;

  transform_buffer = cgCreateBuffer(myCgContext, 3*16*sizeof(float), NULL, CG_BUFFER_USAGE_DYNAMIC_DRAW);
  lightSet_buffer = cgCreateBuffer(myCgContext, sizeof(lightSet), &lightSet, CG_BUFFER_USAGE_STATIC_DRAW);
  lightSetPerView_buffer = cgCreateBuffer(myCgContext, sizeof(LightSetPerView), NULL, CG_BUFFER_USAGE_DYNAMIC_DRAW);
  
  /* Set transform_buffer to every program in the effect that uses cbuffer1_Transform */
  cgSetEffectParameterBuffer(cgGetNamedEffectParameter(myCgEffect, "cbuffer1_Transform"), transform_buffer);

  /* Set lightSet_buffer to every program in the effect that uses cbuffer2_LightSetStatic */
  cgSetEffectParameterBuffer(cgGetNamedEffectParameter(myCgEffect, "cbuffer2_LightSetStatic"), lightSet_buffer);
  
  /* Set lightSetPerView_buffer to every program in the effect that uses cbuffer3_LightSetPerView */
  cgSetEffectParameterBuffer(cgGetNamedEffectParameter(myCgEffect, "cbuffer3_LightSetPerView"), lightSetPerView_buffer);

  material_buffer = new CGbuffer[materialInfoCount];

  /* Create a set of material buffers. */
  for (i=0; i<materialInfoCount; i++) {
    material_buffer[i] = cgCreateBuffer(myCgContext, sizeof(MaterialData), &materialInfo[i].data, CG_BUFFER_USAGE_STATIC_DRAW);
  }

  checkForCgError("initBuffers");
}

void advance(int ignored)
{
  eyeAngle += 0.01;

  if (!myAnimating)
    return;  /* Return without rendering or registering another timer func. */

  glutPostRedisplay();
}

void gl_loadRowMajorMatrix(GLenum matrixMode, const float mat[16])
{
  float transpose_mat[16];

  transposeMatrix(transpose_mat, mat);
  glMatrixMode(matrixMode);
  glLoadMatrixf(transpose_mat);
}

void renderLightPositions(const float projection[16], const float view[16])
{
  int i;

  gl_loadRowMajorMatrix(GL_PROJECTION, projection);
  gl_loadRowMajorMatrix(GL_MODELVIEW, view);
  for (i=0; i<2; i++) {
    if (lightSet.source[i].enabled[0]) {
      glPushMatrix();
        glTranslatef(lightSetPerView_world.source[i].position[0],
                     lightSetPerView_world.source[i].position[1],
                     lightSetPerView_world.source[i].position[2]);
        glColor3fv(lightSet.source[i].diffuse);
        glutSolidSphere(0.3, 10, 10);
      glPopMatrix();
    }
  }
}

void bind_material_buffer(int object)
{
  /* Set material_buffer[object_material[object]] buffer to every program in the effect that uses cbuffer0_Material */
  cgSetEffectParameterBuffer(cgGetNamedEffectParameter(myCgEffect, "cbuffer0_Material"), material_buffer[object_material[object]]);
}

void update_transform_buffer(Transform *transform)
{
  cgSetBufferSubData(transform_buffer, transform_buffer_offset, sizeof(Transform), transform);
}

void drawLitSphere(const float projectionMatrix[16], const float viewMatrix[16], int object, float xTranslate)
{
  Transform transform;
  float modelMatrix[16];

  makeTranslateMatrix(xTranslate, 0, 0, modelMatrix);
  multMatrix(transform.modelview, viewMatrix, modelMatrix);
  multMatrix(transform.modelview_projection, projectionMatrix, transform.modelview);
  invertMatrix(transform.inverse_modelview, transform.modelview);
  
  update_transform_buffer(&transform);

  bind_material_buffer(object);

  CGpass pass = cgGetFirstPass( myCgTechnique );
  while( pass ) 
  {
    cgSetPassState( pass );
   
    glColor3f(1,0,0);
    glutSolidSphere(2.0, 50, 50);
                
    cgResetPassState( pass );
    pass = cgGetNextPass( pass );
  }
}

void drawLitSpheres(const float projectionMatrix[16], const float viewMatrix[16])
{
  int i;

  // For each light, convert its world-space position to eye-space...
  for (i=0; i<2; i++) {
    // Le[i] = V * Lw[i]
    transformVector(lightSetPerView_eye.source[i].position, viewMatrix,
      lightSetPerView_world.source[i].position);
  }

  // Update light set per-view buffer
  cgSetBufferSubData(lightSetPerView_buffer, lightSetPerView_offset,
    sizeof(lightSetPerView_eye), &lightSetPerView_eye);

  // Draw two sphers
  drawLitSphere(projectionMatrix, viewMatrix, 0, 3.2);
  drawLitSphere(projectionMatrix, viewMatrix, 1, -3.2);
}

void drawScene()
{
  float eyePosition[4];
  float viewMatrix[16];

  // Update latest eye position.
  eyePosition[0] = 8*cos(eyeAngle);
  eyePosition[1] = 0;
  eyePosition[2] = 8*sin(eyeAngle);
  eyePosition[3] = 1;

  // Compute current view matrix.
  makeLookAtMatrix(eyePosition[0], eyePosition[1], eyePosition[2],  /* eye position */
                   0, 0, 0, /* view center */
                   0, 1, 0, /* up vector */
                   viewMatrix);

  drawLitSpheres(myProjectionMatrix, viewMatrix);
  renderLightPositions(myProjectionMatrix, viewMatrix);
}