Can anyone see any obvious errors in the code below?
I’ve been through it, but I can’t see any problems. However, I’m finding that it renders properly on some GPUs, but not on others;
Vertex Shader:
/*
Spherical Harmonics code Paul Bourke, adapted from
http://local.wasp.uwa.edu.au/%7Epbourke/surfaces_curves/sphericalh/
Normal calculation concept tonfilm
http://tonfilm.blogspot.com/2007/01/calculate-normals-in-shader.html
Lighting calculation from OpenGL Red Book, via www.lighthouse3d.com
GLSL assistance and general encouragement Memo
GLSL implementation alx @ toneburst, 2008
*/
/////////////////////
//// CONSTANTS ////
/////////////////////
#define TWOPI 6.28318531
#define PI 3.14159265
/////////////////////
//// TWEAKABLES ////
/////////////////////
// Pre-Transform controls
uniform vec4 TT_0; // TT_0(X) = Pre-Scale X (range 0.0 > 1.0)
uniform vec4 TT_1; // TT_1(Y) = Pre-Scale Y (range 0.0 > 1.0)
uniform vec4 TT_2;
uniform vec4 TT_3; // TT_3(X),(Y) = Pre-Translate X,Y (range 0.0 > 1.0)
// Spherical Harmonics controls (range 0.0 to 10.0)
uniform float M0,M1,M2,M3,M4,M5,M6,M7;
// Light position
uniform vec3 LightPosition;
/////////////////////
//// VARYINGS ////
/////////////////////
// Passes result of shading calculation to Fragment Shader
varying float colpos;
/////////////////////
//// FUNCTIONS ////
/////////////////////
// The actual Spherical Harmonics formula (operates on Spherical coordinates)
vec3 sphericalHarmonics(float theta, float phi, float m0,float m1,float m2,float m3,float m4,float m5,float m6,float m7)
{
vec3 point;
float r = 0.0;
r += pow(sin(m0*phi),m1);
r += pow(cos(m2*phi),m3);
r += pow(sin(m4*theta),m5);
r += pow(cos(m6*theta),m7);
point.x = r * sin(phi) * cos(theta);
point.y = r * cos(phi);
point.z = r * sin(phi) * sin(theta);
return point;
}
/////////////////////
//// MAIN LOOP ////
/////////////////////
void main()
{
// Create pre-transform matrix from uniform vec4s
mat4 TT = mat4(TT_0,TT_1,TT_2,TT_3);
// Get vertex coordinates (cartesian)
vec4 vertex = gl_Vertex;
// Initial vertex position pre-transformed
vertex = TT * vertex;
// Spherical coordinates to send to Spherical Harmonics function
float theta = (vertex.x + 0.5) * TWOPI; // set range to 0 > TWOPI
float phi = (vertex.y + 0.5) * PI; // set range 0 > PI
// Spherical Harmonics function
vertex.xyz = sphericalHarmonics(theta, phi, M0, M1, M2, M3, M4, M5, M6, M7);
// Shading calculation
colpos = length(vertex.xyz + LightPosition);
// Transform vertex by modelview and projection matrices
gl_Position = gl_ModelViewProjectionMatrix * vertex;
// Forward current color and texture coordinates after applying texture matrix
gl_TexCoord[0] = gl_TextureMatrix[0] * gl_MultiTexCoord0;
}
}
Fragment Shader:
/////////////////////
//// TWEAKABLES ////
/////////////////////
// Base color
uniform vec4 Color;
// Lighting range. Range 0.1 > 1.0 (use exponential control)
uniform float LightRange;
/////////////////////
//// VARYINGS ////
/////////////////////
// Shading calculation from Vertex Shader
varying float colpos;
/////////////////////
//// TEXTURES ////
/////////////////////
// Shading lookup table input
uniform sampler2D LUT;
// Lookup y-position
uniform float LUT_Y;
// Surface texture input
uniform sampler2D TileImg;
// Surface texture scale
uniform vec2 Tile;
/////////////////////
//// MAIN LOOP ////
/////////////////////
void main()
{
// Fake lighting shading with Lookup Table
float lookupX = clamp((1.0-LightRange) * colpos,0.0,0.999);
// Lookup shade across x-axis of LUT
vec4 shade = texture2D(LUT, vec2(lookupX,LUT_Y));
// Surface tiling texture
vec2 xy = gl_TexCoord[0].xy;
vec2 phase = fract(xy / Tile);
vec4 texTile = texture2D(TileImg,phase);
// Output color compute
//if (texTile.a == 0.0) {
//discard;
//} else {
gl_FragColor = Color * shade;// * texTile;
//}
}
It’s supposed to render like this (and does on the ATI Radeon X1600 on my MacBook Pro):
But on the more powerful NVIDIA GeForce 8800 GT on my MacPro, it renders like this:
As you can see, big chunks of the mesh are missing.