grSstWinOpen((FxU32) NULL, GR_RESOLUTION_800x600, GR_REFRESH_60Hz, GR_COLORFORMAT_RGBA,

The Voodoo cards expect to receive pixel drawing requests that are within their device coordinates. Normally the x, y of these coordinates are set using the grSstWinOpen() command, so in our example above, your coordinates should be within x = 0…800 and y = 0…600. Drawing outside of these coordinates can cause some very unusual behavior in most cases.

## Rendering Primitives for Voodoo

Working with the device coordinate range of the Voodoo card is extremely prohibiting, and not a desirable way to write a graphics program. For this reason we would like to develop some rendering primitives for the Voodoo card that will allow us to work with a more flexible “world” coordinate system. When we are done working with the world system we can project the world onto the appropriate device coordinates. The pipeline for this method is as follows:

world

view
transformation

perspective
transformation

project and
clip

The first step in the pipeline is converting world coordinates into viewing coordinates. The view coordinate transformation moves all objects in the world that should be “in front” of the camera behind the z=0 plane, the goal being that we will later project objects onto the plane and use that plane as what the user will see.

To move objects behind the z=0 plane we will need to define where the imaginary “camera” or “eye” should be placed, what direction the camera will be looking, and what the camera calls “up.” After we have these three vectors defined, we can construct the following view transformation matrix:

(insert Minoura View-16 with correction: Oz = |E – VRP|)
(insert Minoura View-23)

(insert Minoura View-18)

This algorithm is implemented as the GrlLookAt() routine, which loads the view information into a view transformation matrix ‘q’:

void GrlLookAt(float eyex, float eyey, float eyez, float vx, float vy, float vz, float upx, float upy, float upz)

{

GrVertex e_vrp;

float mag_e_vrp, dot_vrpz, mag_vpp;
GrVertex vp, vpp;

// Z' = (E - VRP)/|(E - VRP)|
e_vrp.x = eyex - vx;

e_vrp.y = eyey - vy;

e_vrp.z = eyez - vz;

mag_e_vrp = (float) sqrt(e_vrp.x * e_vrp.x + e_vrp.y * e_vrp.y + e_vrp.z * e_vrp.z);
q[2][0] = e_vrp.x / mag_e_vrp, q[2][1] = e_vrp.y / mag_e_vrp, q[2][2] = e_vrp.z / mag_e_vrp;

// V' = (V.Z') Z'
dot_vrpz = upx*q[2][0] + upy*q[2][1] + upz*q[2][2];

vp.x = dot_vrpz*q[2][0];

vp.y = dot_vrpz*q[2][1];

vp.z = dot_vrpz*q[2][2];

// V'' = V - V'

vpp.x = upx - vp.x;

vpp.y = upy - vp.y;

vpp.z = upz - vp.z;

// Y' = V'' / |V''|
mag_vpp = (float) sqrt(vpp.x * vpp.x + vpp.y * vpp.y + vpp.z * vpp.z);

q[1][0] = vpp.x / mag_vpp, q[1][1] = vpp.y / mag_vpp, q[1][2] = vpp.z / mag_vpp;

// X' = Y' x Z'
q[0][0] = q[1][1] * q[2][2] - q[1][2] * q[2][1];

q[0][1] = -1.0f * (q[1][0] * q[2][2] - q[1][2] * q[2][0]);

q[0][2] = q[1][0] * q[2][1] - q[1][1] * q[2][0];

q[0][3] = 0.0f;

// View Translation

q[0][3] = -vx;

q[1][3] = -vy;

q[2][3] = (float) - sqrt((eyex - vx)*(eyex - vx) + (eyey - vy)*(eyey - vy) + (eyez - vz)*(eyez - vz));

}

Once we have the view transformation matrix we need to perform the perspective transformation, which will make objects that are farther away appear farther away. A typical method for doing this is explained in the text “Mathematical Elements of Computer Graphics,” Rogers and Adams, 1976:

(insert Rogers and Adams Figure 3-8)

Using their method, objects that are behind the z=0 plane are projected onto the z=0 plane as if they are seen from an eye sitting a distance ‘k’ in front of the z=0 plane. This can be implemented mathematically using the theory of right triangles, which simplifies to:

x* = x / ((z / k) + 1)

y* = y / ((z / k) + 1)

Where x* and y* are the “perspective coordinates” of the original x and y, and z is the distance of the original coordinate from the z=0 plane.

The final step in the pipeline is to take what we have behind the z=0 plane and convert it into device coordinates usable by Glide:

(Insert Minoura WinView-2)

The algorithm for performing the last steps is given below.

void GrlFrustum(float itop, float ileft, float ibottom, float iright, float fnear, float ffar)
{

top = itop;

left = ileft;
bottom = ibottom;

right = iright;

fn = fnear;

ff = ffar;

}

// GrlProject – takes source vertex and save projection

// vertex in destination vertex

void GrlProject(GrVertex *dstVert, GrVertex *sv)

{

GrVertex vr;

// copy original to save color information
memcpy(dstVert, sv, sizeof(GrVertex));

// perform view transformation (vr = [q][sv])
vr.x = q[0][0] * sv->x + q[0][1] * sv->y + q[0][2] * sv->z + q[0][3];

vr.y = q[1][0] * sv->x + q[1][1] * sv->y + q[1][2] * sv->z + q[1][3];

vr.z = q[2][0] * sv->x + q[2][1] * sv->y + q[2][2] * sv->z + q[2][3];

// perspective transformation

vr.x = vr.x / ((vr.z / -fn) + 1);

vr.y = vr.y / ((vr.z / -fn) + 1);

// project into device coordinates

dstVert->x = screenx * (vr.x - left) / (right - left);

dstVert->y = screeny * (vr.y - bottom) / (top - bottom);

dstVert->z = 65535.0f * (vr.z - 0.0001f) / (ff - 0.0001f);
dstVert->ooz = -65535.0f / dstVert->z;

// handle clipping here
}

## More Rendering Primitives

More rendering primitives are given in the attached example code.

## References

Minoura, Toshimi. __Introduction to Computer Graphics__. OSU Printing, 1998.

Rogers, D. and Adams, J. __Mathematical Elements of Computer Graphics__. McGraw-Hill, 1976.

**Share with your friends:**