packspu_client.c revision 7c14a9ed66f00f1a0ed175796436ea7730db2ef3
/* Copyright (c) 2001, Stanford University
* All rights reserved
*
* See the file LICENSE.txt for information on redistributing this software.
*/
#include "packspu.h"
#include "cr_packfunctions.h"
#include "cr_glstate.h"
#include "packspu_proto.h"
void PACKSPU_APIENTRY packspu_FogCoordPointerEXT( GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackFogCoordPointerEXTSWAP( type, stride, pointer );
else
crPackFogCoordPointerEXT( type, stride, pointer );
}
#endif
crStateFogCoordPointerEXT( type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_ColorPointer( GLint size, GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackColorPointerSWAP( size, type, stride, pointer );
else
crPackColorPointer( size, type, stride, pointer );
}
#endif
crStateColorPointer( size, type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_SecondaryColorPointerEXT( GLint size, GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackSecondaryColorPointerEXTSWAP( size, type, stride, pointer );
else
crPackSecondaryColorPointerEXT( size, type, stride, pointer );
}
#endif
crStateSecondaryColorPointerEXT( size, type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_VertexPointer( GLint size, GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
CRASSERT(ctx->clientState->extensions.ARB_vertex_buffer_object);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackVertexPointerSWAP( size, type, stride, pointer );
else
crPackVertexPointer( size, type, stride, pointer );
}
#endif
crStateVertexPointer( size, type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_TexCoordPointer( GLint size, GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackTexCoordPointerSWAP( size, type, stride, pointer );
else
crPackTexCoordPointer( size, type, stride, pointer );
}
#endif
crStateTexCoordPointer( size, type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_NormalPointer( GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackNormalPointerSWAP( type, stride, pointer );
else
crPackNormalPointer( type, stride, pointer );
}
#endif
crStateNormalPointer( type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_EdgeFlagPointer( GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackEdgeFlagPointerSWAP( stride, pointer );
else
crPackEdgeFlagPointer( stride, pointer );
}
#endif
crStateEdgeFlagPointer( stride, pointer );
}
void PACKSPU_APIENTRY packspu_VertexAttribPointerARB( GLuint index, GLint size, GLenum type, GLboolean normalized, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackVertexAttribPointerARBSWAP( index, size, type, normalized, stride, pointer );
else
crPackVertexAttribPointerARB( index, size, type, normalized, stride, pointer );
}
#endif
crStateVertexAttribPointerARB( index, size, type, normalized, stride, pointer );
}
void PACKSPU_APIENTRY packspu_VertexAttribPointerNV( GLuint index, GLint size, GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackVertexAttribPointerNVSWAP( index, size, type, stride, pointer );
else
crPackVertexAttribPointerNV( index, size, type, stride, pointer );
}
#endif
crStateVertexAttribPointerNV( index, size, type, stride, pointer );
}
void PACKSPU_APIENTRY packspu_IndexPointer( GLenum type, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackIndexPointerSWAP( type, stride, pointer );
else
crPackIndexPointer( type, stride, pointer );
}
#endif
crStateIndexPointer(type, stride, pointer);
}
void PACKSPU_APIENTRY packspu_GetPointerv( GLenum pname, GLvoid **params )
{
crStateGetPointerv( pname, params );
}
void PACKSPU_APIENTRY packspu_InterleavedArrays( GLenum format, GLsizei stride, const GLvoid *pointer )
{
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
if (ctx->clientState->extensions.ARB_vertex_buffer_object) {
if (pack_spu.swap)
crPackInterleavedArraysSWAP( format, stride, pointer );
else
crPackInterleavedArrays( format, stride, pointer );
}
#endif
/*crDebug("packspu_InterleavedArrays");*/
crStateInterleavedArrays( format, stride, pointer );
}
void PACKSPU_APIENTRY
packspu_ArrayElement( GLint index )
{
/*@todo cash guest/host pointers calculation and use appropriate path here without crStateUseServerArrays call*/
#if 1
GLboolean serverArrays = GL_FALSE;
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
/*crDebug("packspu_ArrayElement index:%i", index);*/
if (ctx->clientState->extensions.ARB_vertex_buffer_object)
serverArrays = crStateUseServerArrays();
#endif
if (serverArrays) {
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
/* LockArraysEXT can not be executed between glBegin/glEnd pair, it also
* leads to vertexpointers being adjusted on the host side between glBegin/glEnd calls which
* produces unpredictable results. Locking is done before the glBegin call instead.
*/
CRASSERT(!clientState->array.locked || clientState->array.synced);
/* Send the DrawArrays command over the wire */
if (pack_spu.swap)
crPackArrayElementSWAP( index );
else
crPackArrayElement( index );
}
else {
/* evaluate locally */
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
if (pack_spu.swap)
crPackExpandArrayElementSWAP( index, clientState );
else
crPackExpandArrayElement( index, clientState );
}
#else
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
crPackExpandArrayElement(index, clientState);
#endif
}
/*#define CR_USE_LOCKARRAYS*/
void PACKSPU_APIENTRY
packspu_DrawElements( GLenum mode, GLsizei count, GLenum type, const GLvoid *indices )
{
GLboolean serverArrays = GL_FALSE;
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
GLboolean lockedArrays = GL_FALSE;
CRBufferObject *elementsBuffer = crStateGetCurrent()->bufferobject.elementsBuffer;
/*crDebug("DrawElements count=%d, indices=%p", count, indices);*/
if (ctx->clientState->extensions.ARB_vertex_buffer_object)
serverArrays = crStateUseServerArrays();
# ifdef CR_USE_LOCKARRAYS
if (!serverArrays && !ctx->clientState->client.array.locked && (count>3)
&& (!elementsBuffer || !elementsBuffer->id))
{
GLuint min, max;
GLsizei i;
switch (type)
{
case GL_UNSIGNED_BYTE:
{
GLubyte *pIdx = (GLubyte *)indices;
min = max = pIdx[0];
for (i=0; i<count; ++i)
{
if (pIdx[i]<min) min = pIdx[i];
else if (pIdx[i]>max) max = pIdx[i];
}
break;
}
case GL_UNSIGNED_SHORT:
{
GLushort *pIdx = (GLushort *)indices;
min = max = pIdx[0];
for (i=0; i<count; ++i)
{
if (pIdx[i]<min) min = pIdx[i];
else if (pIdx[i]>max) max = pIdx[i];
}
break;
}
case GL_UNSIGNED_INT:
{
GLuint *pIdx = (GLuint *)indices;
min = max = pIdx[0];
for (i=0; i<count; ++i)
{
if (pIdx[i]<min) min = pIdx[i];
else if (pIdx[i]>max) max = pIdx[i];
}
break;
}
default: crError("Unknown type 0x%x", type);
}
if ((max-min)<(GLuint)(2*count))
{
crStateLockArraysEXT(min, max-min+1);
serverArrays = crStateUseServerArrays();
if (serverArrays)
{
lockedArrays = GL_TRUE;
}
else
{
crStateUnlockArraysEXT();
}
}
}
# endif
#endif
if (serverArrays) {
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
/*Note the comment in packspu_LockArraysEXT*/
if (clientState->array.locked && !clientState->array.synced)
{
crPackLockArraysEXT(clientState->array.lockFirst, clientState->array.lockCount);
clientState->array.synced = GL_TRUE;
}
/* Send the DrawArrays command over the wire */
if (pack_spu.swap)
crPackDrawElementsSWAP( mode, count, type, indices );
else
crPackDrawElements( mode, count, type, indices );
}
else {
/* evaluate locally */
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
if (pack_spu.swap)
crPackExpandDrawElementsSWAP( mode, count, type, indices, clientState );
else
{
//packspu_Begin(mode);
crPackExpandDrawElements( mode, count, type, indices, clientState );
//packspu_End();
}
}
#if CR_ARB_vertex_buffer_object
if (lockedArrays)
{
packspu_UnlockArraysEXT();
}
#endif
}
void PACKSPU_APIENTRY
packspu_DrawRangeElements( GLenum mode, GLuint start, GLuint end, GLsizei count, GLenum type, const GLvoid *indices )
{
GLboolean serverArrays = GL_FALSE;
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
/*crDebug("DrawRangeElements count=%d", count);*/
if (ctx->clientState->extensions.ARB_vertex_buffer_object)
serverArrays = crStateUseServerArrays();
#endif
if (serverArrays) {
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
/*Note the comment in packspu_LockArraysEXT*/
if (clientState->array.locked && !clientState->array.synced)
{
crPackLockArraysEXT(clientState->array.lockFirst, clientState->array.lockCount);
clientState->array.synced = GL_TRUE;
}
/* Send the DrawRangeElements command over the wire */
if (pack_spu.swap)
crPackDrawRangeElementsSWAP( mode, start, end, count, type, indices );
else
crPackDrawRangeElements( mode, start, end, count, type, indices );
}
else {
/* evaluate locally */
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
if (pack_spu.swap)
crPackExpandDrawRangeElementsSWAP( mode, start, end, count, type, indices, clientState );
else
{
crPackExpandDrawRangeElements( mode, start, end, count, type, indices, clientState );
}
}
}
void PACKSPU_APIENTRY
packspu_DrawArrays( GLenum mode, GLint first, GLsizei count )
{
GLboolean serverArrays = GL_FALSE;
#if CR_ARB_vertex_buffer_object
GET_CONTEXT(ctx);
GLboolean lockedArrays = GL_FALSE;
/*crDebug("DrawArrays count=%d", count);*/
if (ctx->clientState->extensions.ARB_vertex_buffer_object)
serverArrays = crStateUseServerArrays();
# ifdef CR_USE_LOCKARRAYS
if (!serverArrays && !ctx->clientState->client.array.locked && (count>3))
{
crStateLockArraysEXT(first, count);
serverArrays = crStateUseServerArrays();
if (serverArrays)
{
lockedArrays = GL_TRUE;
}
else
{
crStateUnlockArraysEXT();
}
}
# endif
#endif
if (serverArrays)
{
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
/*Note the comment in packspu_LockArraysEXT*/
if (clientState->array.locked && !clientState->array.synced)
{
crPackLockArraysEXT(clientState->array.lockFirst, clientState->array.lockCount);
clientState->array.synced = GL_TRUE;
}
/* Send the DrawArrays command over the wire */
if (pack_spu.swap)
crPackDrawArraysSWAP( mode, first, count );
else
crPackDrawArrays( mode, first, count );
}
else
{
/* evaluate locally */
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
if (pack_spu.swap)
crPackExpandDrawArraysSWAP( mode, first, count, clientState );
else
crPackExpandDrawArrays( mode, first, count, clientState );
}
#if CR_ARB_vertex_buffer_object
if (lockedArrays)
{
packspu_UnlockArraysEXT();
}
#endif
}
#ifdef CR_EXT_multi_draw_arrays
void PACKSPU_APIENTRY packspu_MultiDrawArraysEXT( GLenum mode, GLint *first, GLsizei *count, GLsizei primcount )
{
GLint i;
for (i = 0; i < primcount; i++) {
if (count[i] > 0) {
packspu_DrawArrays(mode, first[i], count[i]);
}
}
}
void PACKSPU_APIENTRY packspu_MultiDrawElementsEXT( GLenum mode, const GLsizei *count, GLenum type, const GLvoid **indices, GLsizei primcount )
{
GLint i;
for (i = 0; i < primcount; i++) {
if (count[i] > 0) {
packspu_DrawElements(mode, count[i], type, indices[i]);
}
}
}
#endif
void PACKSPU_APIENTRY packspu_EnableClientState( GLenum array )
{
crStateEnableClientState(array);
crPackEnableClientState(array);
}
void PACKSPU_APIENTRY packspu_DisableClientState( GLenum array )
{
crStateDisableClientState(array);
crPackDisableClientState(array);
}
void PACKSPU_APIENTRY packspu_ClientActiveTextureARB( GLenum texUnit )
{
crStateClientActiveTextureARB(texUnit);
crPackClientActiveTextureARB(texUnit);
}
void PACKSPU_APIENTRY packspu_EnableVertexAttribArrayARB(GLuint index)
{
crStateEnableVertexAttribArrayARB(index);
crPackEnableVertexAttribArrayARB(index);
}
void PACKSPU_APIENTRY packspu_DisableVertexAttribArrayARB(GLuint index)
{
crStateDisableVertexAttribArrayARB(index);
crPackDisableVertexAttribArrayARB(index);
}
void PACKSPU_APIENTRY packspu_Enable( GLenum cap )
{
if (cap!=GL_LIGHT_MODEL_TWO_SIDE)
{
crStateEnable(cap);
if (pack_spu.swap)
crPackEnableSWAP(cap);
else
crPackEnable(cap);
}
else
{
static int g_glmts1_warn=0;
if (!g_glmts1_warn)
{
crWarning("glEnable(GL_LIGHT_MODEL_TWO_SIDE) converted to valid glLightModeli(GL_LIGHT_MODEL_TWO_SIDE,1)");
g_glmts1_warn=1;
}
crStateLightModeli(GL_LIGHT_MODEL_TWO_SIDE, 1);
crPackLightModeli(GL_LIGHT_MODEL_TWO_SIDE, 1);
}
}
void PACKSPU_APIENTRY packspu_Disable( GLenum cap )
{
if (cap!=GL_LIGHT_MODEL_TWO_SIDE)
{
crStateDisable(cap);
if (pack_spu.swap)
crPackDisableSWAP(cap);
else
crPackDisable(cap);
}
else
{
static int g_glmts0_warn=0;
if (!g_glmts0_warn)
{
crWarning("glDisable(GL_LIGHT_MODEL_TWO_SIDE) converted to valid glLightModeli(GL_LIGHT_MODEL_TWO_SIDE,0)");
g_glmts0_warn=1;
}
crStateLightModeli(GL_LIGHT_MODEL_TWO_SIDE, 0);
crPackLightModeli(GL_LIGHT_MODEL_TWO_SIDE, 0);
}
}
GLboolean PACKSPU_APIENTRY packspu_IsEnabled(GLenum cap)
{
GLboolean res = crStateIsEnabled(cap);
#ifdef DEBUG
{
GET_THREAD(thread);
int writeback = 1;
GLboolean return_val = (GLboolean) 0;
crPackIsEnabled(cap, &return_val, &writeback);
packspuFlush( (void *) thread );
while (writeback)
crNetRecv();
CRASSERT(return_val==res);
}
#endif
return res;
}
void PACKSPU_APIENTRY packspu_PushClientAttrib( GLbitfield mask )
{
crStatePushClientAttrib(mask);
crPackPushClientAttrib(mask);
}
void PACKSPU_APIENTRY packspu_PopClientAttrib( void )
{
crStatePopClientAttrib();
crPackPopClientAttrib();
}
void PACKSPU_APIENTRY packspu_LockArraysEXT(GLint first, GLint count)
{
if (first>=0 && count>0)
{
crStateLockArraysEXT(first, count);
/*Note: this is a workaround for quake3 based apps.
It's modifying vertex data between glLockArraysEXT and glDrawElements calls,
so we'd pass data to host right before the glDrawSomething or glBegin call.
*/
/*crPackLockArraysEXT(first, count);*/
}
else crDebug("Ignoring packspu_LockArraysEXT: first:%i, count:%i", first, count);
}
void PACKSPU_APIENTRY packspu_UnlockArraysEXT()
{
GET_CONTEXT(ctx);
CRClientState *clientState = &(ctx->clientState->client);
if (clientState->array.locked && clientState->array.synced)
{
crPackUnlockArraysEXT();
}
crStateUnlockArraysEXT();
}