implement full reference counting for vertex/fragment programs
Use _mesa_reference_vert/fragprog() wherever we assign program pointers. Fixes a memory corruption bug found with glean/api2 test.
This commit is contained in:
@@ -149,8 +149,6 @@ int MESA_DEBUG_FLAGS = 0;
|
|||||||
/* ubyte -> float conversion */
|
/* ubyte -> float conversion */
|
||||||
GLfloat _mesa_ubyte_to_float_color_tab[256];
|
GLfloat _mesa_ubyte_to_float_color_tab[256];
|
||||||
|
|
||||||
static void
|
|
||||||
free_shared_state( GLcontext *ctx, struct gl_shared_state *ss );
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -420,12 +418,14 @@ alloc_shared_state( GLcontext *ctx )
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if FEATURE_ARB_vertex_program
|
#if FEATURE_ARB_vertex_program
|
||||||
ss->DefaultVertexProgram = ctx->Driver.NewProgram(ctx, GL_VERTEX_PROGRAM_ARB, 0);
|
ss->DefaultVertexProgram = (struct gl_vertex_program *)
|
||||||
|
ctx->Driver.NewProgram(ctx, GL_VERTEX_PROGRAM_ARB, 0);
|
||||||
if (!ss->DefaultVertexProgram)
|
if (!ss->DefaultVertexProgram)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ARB_fragment_program
|
#if FEATURE_ARB_fragment_program
|
||||||
ss->DefaultFragmentProgram = ctx->Driver.NewProgram(ctx, GL_FRAGMENT_PROGRAM_ARB, 0);
|
ss->DefaultFragmentProgram = (struct gl_fragment_program *)
|
||||||
|
ctx->Driver.NewProgram(ctx, GL_FRAGMENT_PROGRAM_ARB, 0);
|
||||||
if (!ss->DefaultFragmentProgram)
|
if (!ss->DefaultFragmentProgram)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
#endif
|
#endif
|
||||||
@@ -502,12 +502,10 @@ cleanup:
|
|||||||
_mesa_DeleteHashTable(ss->Programs);
|
_mesa_DeleteHashTable(ss->Programs);
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ARB_vertex_program
|
#if FEATURE_ARB_vertex_program
|
||||||
if (ss->DefaultVertexProgram)
|
_mesa_reference_vertprog(ctx, &ss->DefaultVertexProgram, NULL);
|
||||||
ctx->Driver.DeleteProgram(ctx, ss->DefaultVertexProgram);
|
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ARB_fragment_program
|
#if FEATURE_ARB_fragment_program
|
||||||
if (ss->DefaultFragmentProgram)
|
_mesa_reference_fragprog(ctx, &ss->DefaultFragmentProgram, NULL);
|
||||||
ctx->Driver.DeleteProgram(ctx, ss->DefaultFragmentProgram);
|
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ATI_fragment_shader
|
#if FEATURE_ATI_fragment_shader
|
||||||
if (ss->DefaultFragmentShader)
|
if (ss->DefaultFragmentShader)
|
||||||
@@ -714,10 +712,10 @@ free_shared_state( GLcontext *ctx, struct gl_shared_state *ss )
|
|||||||
_mesa_DeleteHashTable(ss->Programs);
|
_mesa_DeleteHashTable(ss->Programs);
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ARB_vertex_program
|
#if FEATURE_ARB_vertex_program
|
||||||
ctx->Driver.DeleteProgram(ctx, ss->DefaultVertexProgram);
|
_mesa_reference_vertprog(ctx, &ss->DefaultVertexProgram, NULL);
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ARB_fragment_program
|
#if FEATURE_ARB_fragment_program
|
||||||
ctx->Driver.DeleteProgram(ctx, ss->DefaultFragmentProgram);
|
_mesa_reference_fragprog(ctx, &ss->DefaultFragmentProgram, NULL);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if FEATURE_ATI_fragment_shader
|
#if FEATURE_ATI_fragment_shader
|
||||||
@@ -1251,6 +1249,14 @@ _mesa_free_context_data( GLcontext *ctx )
|
|||||||
_mesa_unreference_framebuffer(&ctx->DrawBuffer);
|
_mesa_unreference_framebuffer(&ctx->DrawBuffer);
|
||||||
_mesa_unreference_framebuffer(&ctx->ReadBuffer);
|
_mesa_unreference_framebuffer(&ctx->ReadBuffer);
|
||||||
|
|
||||||
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram.Current, NULL);
|
||||||
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._Current, NULL);
|
||||||
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._TnlProgram, NULL);
|
||||||
|
|
||||||
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram.Current, NULL);
|
||||||
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._Current, NULL);
|
||||||
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._TexEnvProgram, NULL);
|
||||||
|
|
||||||
_mesa_free_attrib_data(ctx);
|
_mesa_free_attrib_data(ctx);
|
||||||
_mesa_free_lighting_data( ctx );
|
_mesa_free_lighting_data( ctx );
|
||||||
_mesa_free_eval_data( ctx );
|
_mesa_free_eval_data( ctx );
|
||||||
|
@@ -2185,10 +2185,10 @@ struct gl_shared_state
|
|||||||
/*@{*/
|
/*@{*/
|
||||||
struct _mesa_HashTable *Programs; /**< All vertex/fragment programs */
|
struct _mesa_HashTable *Programs; /**< All vertex/fragment programs */
|
||||||
#if FEATURE_ARB_vertex_program
|
#if FEATURE_ARB_vertex_program
|
||||||
struct gl_program *DefaultVertexProgram;
|
struct gl_vertex_program *DefaultVertexProgram;
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_ARB_fragment_program
|
#if FEATURE_ARB_fragment_program
|
||||||
struct gl_program *DefaultFragmentProgram;
|
struct gl_fragment_program *DefaultFragmentProgram;
|
||||||
#endif
|
#endif
|
||||||
/*@}*/
|
/*@}*/
|
||||||
|
|
||||||
|
@@ -978,50 +978,60 @@ update_program(GLcontext *ctx)
|
|||||||
* 3. Programs derived from fixed-function state.
|
* 3. Programs derived from fixed-function state.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
ctx->FragmentProgram._Current = NULL;
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._Current, NULL);
|
||||||
|
|
||||||
if (shProg && shProg->LinkStatus) {
|
if (shProg && shProg->LinkStatus) {
|
||||||
/* Use shader programs */
|
/* Use shader programs */
|
||||||
/* XXX this isn't quite right, since we may have either a vertex
|
/* XXX this isn't quite right, since we may have either a vertex
|
||||||
* _or_ fragment shader (not always both).
|
* _or_ fragment shader (not always both).
|
||||||
*/
|
*/
|
||||||
ctx->VertexProgram._Current = shProg->VertexProgram;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._Current,
|
||||||
ctx->FragmentProgram._Current = shProg->FragmentProgram;
|
shProg->VertexProgram);
|
||||||
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._Current,
|
||||||
|
shProg->FragmentProgram);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (ctx->VertexProgram._Enabled) {
|
if (ctx->VertexProgram._Enabled) {
|
||||||
/* use user-defined vertex program */
|
/* use user-defined vertex program */
|
||||||
ctx->VertexProgram._Current = ctx->VertexProgram.Current;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._Current,
|
||||||
|
ctx->VertexProgram.Current);
|
||||||
}
|
}
|
||||||
else if (ctx->VertexProgram._MaintainTnlProgram) {
|
else if (ctx->VertexProgram._MaintainTnlProgram) {
|
||||||
/* Use vertex program generated from fixed-function state.
|
/* Use vertex program generated from fixed-function state.
|
||||||
* The _Current pointer will get set in
|
* The _Current pointer will get set in
|
||||||
* _tnl_UpdateFixedFunctionProgram() later if appropriate.
|
* _tnl_UpdateFixedFunctionProgram() later if appropriate.
|
||||||
*/
|
*/
|
||||||
ctx->VertexProgram._Current = NULL;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._Current, NULL);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
/* no vertex program */
|
/* no vertex program */
|
||||||
ctx->VertexProgram._Current = NULL;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._Current, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ctx->FragmentProgram._Enabled) {
|
if (ctx->FragmentProgram._Enabled) {
|
||||||
/* use user-defined vertex program */
|
/* use user-defined vertex program */
|
||||||
ctx->FragmentProgram._Current = ctx->FragmentProgram.Current;
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._Current,
|
||||||
|
ctx->FragmentProgram.Current);
|
||||||
}
|
}
|
||||||
else if (ctx->FragmentProgram._MaintainTexEnvProgram) {
|
else if (ctx->FragmentProgram._MaintainTexEnvProgram) {
|
||||||
/* Use fragment program generated from fixed-function state.
|
/* Use fragment program generated from fixed-function state.
|
||||||
* The _Current pointer will get set in _mesa_UpdateTexEnvProgram()
|
* The _Current pointer will get set in _mesa_UpdateTexEnvProgram()
|
||||||
* later if appropriate.
|
* later if appropriate.
|
||||||
*/
|
*/
|
||||||
ctx->FragmentProgram._Current = NULL;
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._Current, NULL);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
/* no fragment program */
|
/* no fragment program */
|
||||||
ctx->FragmentProgram._Current = NULL;
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram._Current, NULL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ctx->VertexProgram._Current)
|
||||||
|
assert(ctx->VertexProgram._Current->Base.Parameters);
|
||||||
|
if (ctx->FragmentProgram._Current)
|
||||||
|
assert(ctx->FragmentProgram._Current->Base.Parameters);
|
||||||
|
|
||||||
|
|
||||||
ctx->FragmentProgram._Active = ctx->FragmentProgram._Enabled;
|
ctx->FragmentProgram._Active = ctx->FragmentProgram._Enabled;
|
||||||
if (ctx->FragmentProgram._MaintainTexEnvProgram &&
|
if (ctx->FragmentProgram._MaintainTexEnvProgram &&
|
||||||
!ctx->FragmentProgram._Enabled) {
|
!ctx->FragmentProgram._Enabled) {
|
||||||
|
@@ -59,9 +59,9 @@ _mesa_init_program(GLcontext *ctx)
|
|||||||
ctx->VertexProgram.Enabled = GL_FALSE;
|
ctx->VertexProgram.Enabled = GL_FALSE;
|
||||||
ctx->VertexProgram.PointSizeEnabled = GL_FALSE;
|
ctx->VertexProgram.PointSizeEnabled = GL_FALSE;
|
||||||
ctx->VertexProgram.TwoSideEnabled = GL_FALSE;
|
ctx->VertexProgram.TwoSideEnabled = GL_FALSE;
|
||||||
ctx->VertexProgram.Current = (struct gl_vertex_program *) ctx->Shared->DefaultVertexProgram;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram.Current,
|
||||||
|
ctx->Shared->DefaultVertexProgram);
|
||||||
assert(ctx->VertexProgram.Current);
|
assert(ctx->VertexProgram.Current);
|
||||||
ctx->VertexProgram.Current->Base.RefCount++;
|
|
||||||
for (i = 0; i < MAX_NV_VERTEX_PROGRAM_PARAMS / 4; i++) {
|
for (i = 0; i < MAX_NV_VERTEX_PROGRAM_PARAMS / 4; i++) {
|
||||||
ctx->VertexProgram.TrackMatrix[i] = GL_NONE;
|
ctx->VertexProgram.TrackMatrix[i] = GL_NONE;
|
||||||
ctx->VertexProgram.TrackMatrixTransform[i] = GL_IDENTITY_NV;
|
ctx->VertexProgram.TrackMatrixTransform[i] = GL_IDENTITY_NV;
|
||||||
@@ -70,7 +70,8 @@ _mesa_init_program(GLcontext *ctx)
|
|||||||
|
|
||||||
#if FEATURE_NV_fragment_program || FEATURE_ARB_fragment_program
|
#if FEATURE_NV_fragment_program || FEATURE_ARB_fragment_program
|
||||||
ctx->FragmentProgram.Enabled = GL_FALSE;
|
ctx->FragmentProgram.Enabled = GL_FALSE;
|
||||||
ctx->FragmentProgram.Current = (struct gl_fragment_program *) ctx->Shared->DefaultFragmentProgram;
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram.Current,
|
||||||
|
ctx->Shared->DefaultFragmentProgram);
|
||||||
assert(ctx->FragmentProgram.Current);
|
assert(ctx->FragmentProgram.Current);
|
||||||
ctx->FragmentProgram.Current->Base.RefCount++;
|
ctx->FragmentProgram.Current->Base.RefCount++;
|
||||||
#endif
|
#endif
|
||||||
@@ -92,18 +93,10 @@ void
|
|||||||
_mesa_free_program_data(GLcontext *ctx)
|
_mesa_free_program_data(GLcontext *ctx)
|
||||||
{
|
{
|
||||||
#if FEATURE_NV_vertex_program || FEATURE_ARB_vertex_program
|
#if FEATURE_NV_vertex_program || FEATURE_ARB_vertex_program
|
||||||
if (ctx->VertexProgram.Current) {
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram.Current, NULL);
|
||||||
ctx->VertexProgram.Current->Base.RefCount--;
|
|
||||||
if (ctx->VertexProgram.Current->Base.RefCount <= 0)
|
|
||||||
ctx->Driver.DeleteProgram(ctx, &(ctx->VertexProgram.Current->Base));
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#if FEATURE_NV_fragment_program || FEATURE_ARB_fragment_program
|
#if FEATURE_NV_fragment_program || FEATURE_ARB_fragment_program
|
||||||
if (ctx->FragmentProgram.Current) {
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram.Current, NULL);
|
||||||
ctx->FragmentProgram.Current->Base.RefCount--;
|
|
||||||
if (ctx->FragmentProgram.Current->Base.RefCount <= 0)
|
|
||||||
ctx->Driver.DeleteProgram(ctx, &(ctx->FragmentProgram.Current->Base));
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
/* XXX probably move this stuff */
|
/* XXX probably move this stuff */
|
||||||
#if FEATURE_ATI_fragment_shader
|
#if FEATURE_ATI_fragment_shader
|
||||||
@@ -365,6 +358,59 @@ _mesa_lookup_program(GLcontext *ctx, GLuint id)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reference counting for vertex/fragment programs
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
_mesa_reference_program(GLcontext *ctx,
|
||||||
|
struct gl_program **ptr,
|
||||||
|
struct gl_program *prog)
|
||||||
|
{
|
||||||
|
assert(ptr);
|
||||||
|
if (*ptr && prog) {
|
||||||
|
/* sanity check */
|
||||||
|
ASSERT((*ptr)->Target == prog->Target);
|
||||||
|
}
|
||||||
|
if (*ptr == prog) {
|
||||||
|
return; /* no change */
|
||||||
|
}
|
||||||
|
if (*ptr) {
|
||||||
|
GLboolean deleteFlag;
|
||||||
|
|
||||||
|
/*_glthread_LOCK_MUTEX((*ptr)->Mutex);*/
|
||||||
|
#if 0
|
||||||
|
printf("Program %p %u 0x%x Refcount-- to %d\n",
|
||||||
|
*ptr, (*ptr)->Id, (*ptr)->Target, (*ptr)->RefCount - 1);
|
||||||
|
#endif
|
||||||
|
ASSERT((*ptr)->RefCount > 0);
|
||||||
|
(*ptr)->RefCount--;
|
||||||
|
|
||||||
|
deleteFlag = ((*ptr)->RefCount == 0);
|
||||||
|
/*_glthread_UNLOCK_MUTEX((*ptr)->Mutex);*/
|
||||||
|
|
||||||
|
if (deleteFlag) {
|
||||||
|
ASSERT(ctx);
|
||||||
|
ctx->Driver.DeleteProgram(ctx, *ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
*ptr = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(!*ptr);
|
||||||
|
if (prog) {
|
||||||
|
/*_glthread_LOCK_MUTEX(prog->Mutex);*/
|
||||||
|
prog->RefCount++;
|
||||||
|
#if 0
|
||||||
|
printf("Program %p %u 0x%x Refcount++ to %d\n",
|
||||||
|
prog, prog->Id, prog->Target, prog->RefCount);
|
||||||
|
#endif
|
||||||
|
/*_glthread_UNLOCK_MUTEX(prog->Mutex);*/
|
||||||
|
}
|
||||||
|
|
||||||
|
*ptr = prog;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return a copy of a program.
|
* Return a copy of a program.
|
||||||
* XXX Problem here if the program object is actually OO-derivation
|
* XXX Problem here if the program object is actually OO-derivation
|
||||||
@@ -380,8 +426,9 @@ _mesa_clone_program(GLcontext *ctx, const struct gl_program *prog)
|
|||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
assert(clone->Target == prog->Target);
|
assert(clone->Target == prog->Target);
|
||||||
|
assert(clone->RefCount == 1);
|
||||||
|
|
||||||
clone->String = (GLubyte *) _mesa_strdup((char *) prog->String);
|
clone->String = (GLubyte *) _mesa_strdup((char *) prog->String);
|
||||||
clone->RefCount = 1;
|
|
||||||
clone->Format = prog->Format;
|
clone->Format = prog->Format;
|
||||||
clone->Instructions = _mesa_alloc_instructions(prog->NumInstructions);
|
clone->Instructions = _mesa_alloc_instructions(prog->NumInstructions);
|
||||||
if (!clone->Instructions) {
|
if (!clone->Instructions) {
|
||||||
@@ -513,9 +560,9 @@ _mesa_BindProgram(GLenum target, GLuint id)
|
|||||||
/* Bind a default program */
|
/* Bind a default program */
|
||||||
newProg = NULL;
|
newProg = NULL;
|
||||||
if (target == GL_VERTEX_PROGRAM_ARB) /* == GL_VERTEX_PROGRAM_NV */
|
if (target == GL_VERTEX_PROGRAM_ARB) /* == GL_VERTEX_PROGRAM_NV */
|
||||||
newProg = ctx->Shared->DefaultVertexProgram;
|
newProg = &ctx->Shared->DefaultVertexProgram->Base;
|
||||||
else
|
else
|
||||||
newProg = ctx->Shared->DefaultFragmentProgram;
|
newProg = &ctx->Shared->DefaultFragmentProgram->Base;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
/* Bind a user program */
|
/* Bind a user program */
|
||||||
@@ -543,26 +590,16 @@ _mesa_BindProgram(GLenum target, GLuint id)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* unbind/delete oldProg */
|
|
||||||
if (curProg->Id != 0) {
|
|
||||||
/* decrement refcount on previously bound fragment program */
|
|
||||||
curProg->RefCount--;
|
|
||||||
/* and delete if refcount goes below one */
|
|
||||||
if (curProg->RefCount <= 0) {
|
|
||||||
/* the program ID was already removed from the hash table */
|
|
||||||
ctx->Driver.DeleteProgram(ctx, curProg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* bind newProg */
|
/* bind newProg */
|
||||||
if (target == GL_VERTEX_PROGRAM_ARB) { /* == GL_VERTEX_PROGRAM_NV */
|
if (target == GL_VERTEX_PROGRAM_ARB) { /* == GL_VERTEX_PROGRAM_NV */
|
||||||
ctx->VertexProgram.Current = (struct gl_vertex_program *) newProg;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram.Current,
|
||||||
|
(struct gl_vertex_program *) newProg);
|
||||||
}
|
}
|
||||||
else if (target == GL_FRAGMENT_PROGRAM_NV ||
|
else if (target == GL_FRAGMENT_PROGRAM_NV ||
|
||||||
target == GL_FRAGMENT_PROGRAM_ARB) {
|
target == GL_FRAGMENT_PROGRAM_ARB) {
|
||||||
ctx->FragmentProgram.Current = (struct gl_fragment_program *) newProg;
|
_mesa_reference_fragprog(ctx, &ctx->FragmentProgram.Current,
|
||||||
|
(struct gl_fragment_program *) newProg);
|
||||||
}
|
}
|
||||||
newProg->RefCount++;
|
|
||||||
|
|
||||||
/* Never null pointers */
|
/* Never null pointers */
|
||||||
ASSERT(ctx->VertexProgram.Current);
|
ASSERT(ctx->VertexProgram.Current);
|
||||||
@@ -620,10 +657,7 @@ _mesa_DeletePrograms(GLsizei n, const GLuint *ids)
|
|||||||
}
|
}
|
||||||
/* The ID is immediately available for re-use now */
|
/* The ID is immediately available for re-use now */
|
||||||
_mesa_HashRemove(ctx->Shared->Programs, ids[i]);
|
_mesa_HashRemove(ctx->Shared->Programs, ids[i]);
|
||||||
prog->RefCount--;
|
_mesa_reference_program(ctx, &prog, NULL);
|
||||||
if (prog->RefCount <= 0) {
|
|
||||||
ctx->Driver.DeleteProgram(ctx, prog);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -86,6 +86,28 @@ _mesa_delete_program(GLcontext *ctx, struct gl_program *prog);
|
|||||||
extern struct gl_program *
|
extern struct gl_program *
|
||||||
_mesa_lookup_program(GLcontext *ctx, GLuint id);
|
_mesa_lookup_program(GLcontext *ctx, GLuint id);
|
||||||
|
|
||||||
|
extern void
|
||||||
|
_mesa_reference_program(GLcontext *ctx,
|
||||||
|
struct gl_program **ptr,
|
||||||
|
struct gl_program *prog);
|
||||||
|
|
||||||
|
static INLINE void
|
||||||
|
_mesa_reference_vertprog(GLcontext *ctx,
|
||||||
|
struct gl_vertex_program **ptr,
|
||||||
|
struct gl_vertex_program *prog)
|
||||||
|
{
|
||||||
|
_mesa_reference_program(ctx, (struct gl_program **) ptr,
|
||||||
|
(struct gl_program *) prog);
|
||||||
|
}
|
||||||
|
|
||||||
|
static INLINE void
|
||||||
|
_mesa_reference_fragprog(GLcontext *ctx,
|
||||||
|
struct gl_fragment_program **ptr,
|
||||||
|
struct gl_fragment_program *prog)
|
||||||
|
{
|
||||||
|
_mesa_reference_program(ctx, (struct gl_program **) ptr,
|
||||||
|
(struct gl_program *) prog);
|
||||||
|
}
|
||||||
|
|
||||||
extern struct gl_program *
|
extern struct gl_program *
|
||||||
_mesa_clone_program(GLcontext *ctx, const struct gl_program *prog);
|
_mesa_clone_program(GLcontext *ctx, const struct gl_program *prog);
|
||||||
|
@@ -79,8 +79,7 @@ _mesa_clear_shader_program_data(GLcontext *ctx,
|
|||||||
/* to prevent a double-free in the next call */
|
/* to prevent a double-free in the next call */
|
||||||
shProg->VertexProgram->Base.Parameters = NULL;
|
shProg->VertexProgram->Base.Parameters = NULL;
|
||||||
}
|
}
|
||||||
ctx->Driver.DeleteProgram(ctx, &shProg->VertexProgram->Base);
|
_mesa_reference_vertprog(ctx, &shProg->VertexProgram, NULL);
|
||||||
shProg->VertexProgram = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (shProg->FragmentProgram) {
|
if (shProg->FragmentProgram) {
|
||||||
@@ -88,8 +87,7 @@ _mesa_clear_shader_program_data(GLcontext *ctx,
|
|||||||
/* to prevent a double-free in the next call */
|
/* to prevent a double-free in the next call */
|
||||||
shProg->FragmentProgram->Base.Parameters = NULL;
|
shProg->FragmentProgram->Base.Parameters = NULL;
|
||||||
}
|
}
|
||||||
ctx->Driver.DeleteProgram(ctx, &shProg->FragmentProgram->Base);
|
_mesa_reference_fragprog(ctx, &shProg->FragmentProgram, NULL);
|
||||||
shProg->FragmentProgram = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (shProg->Uniforms) {
|
if (shProg->Uniforms) {
|
||||||
@@ -1100,6 +1098,8 @@ _mesa_link_program(GLcontext *ctx, GLuint program)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FLUSH_VERTICES(ctx, _NEW_PROGRAM);
|
||||||
|
|
||||||
_slang_link(ctx, program, shProg);
|
_slang_link(ctx, program, shProg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -516,19 +516,19 @@ _slang_link(GLcontext *ctx,
|
|||||||
* changing src/dst registers after merging the uniforms and varying vars.
|
* changing src/dst registers after merging the uniforms and varying vars.
|
||||||
*/
|
*/
|
||||||
if (vertProg) {
|
if (vertProg) {
|
||||||
shProg->VertexProgram
|
_mesa_reference_vertprog(ctx, &shProg->VertexProgram,
|
||||||
= vertex_program(_mesa_clone_program(ctx, &vertProg->Base));
|
vertex_program(_mesa_clone_program(ctx, &vertProg->Base)));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
shProg->VertexProgram = NULL;
|
_mesa_reference_vertprog(ctx, &shProg->VertexProgram, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fragProg) {
|
if (fragProg) {
|
||||||
shProg->FragmentProgram
|
_mesa_reference_fragprog(ctx, &shProg->FragmentProgram,
|
||||||
= fragment_program(_mesa_clone_program(ctx, &fragProg->Base));
|
fragment_program(_mesa_clone_program(ctx, &fragProg->Base)));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
shProg->FragmentProgram = NULL;
|
_mesa_reference_fragprog(ctx, &shProg->FragmentProgram, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (shProg->VertexProgram)
|
if (shProg->VertexProgram)
|
||||||
@@ -545,10 +545,12 @@ _slang_link(GLcontext *ctx,
|
|||||||
if (shProg->VertexProgram) {
|
if (shProg->VertexProgram) {
|
||||||
_mesa_free_parameter_list(shProg->VertexProgram->Base.Parameters);
|
_mesa_free_parameter_list(shProg->VertexProgram->Base.Parameters);
|
||||||
shProg->VertexProgram->Base.Parameters = shProg->Uniforms;
|
shProg->VertexProgram->Base.Parameters = shProg->Uniforms;
|
||||||
|
assert(shProg->Uniforms);
|
||||||
}
|
}
|
||||||
if (shProg->FragmentProgram) {
|
if (shProg->FragmentProgram) {
|
||||||
_mesa_free_parameter_list(shProg->FragmentProgram->Base.Parameters);
|
_mesa_free_parameter_list(shProg->FragmentProgram->Base.Parameters);
|
||||||
shProg->FragmentProgram->Base.Parameters = shProg->Uniforms;
|
shProg->FragmentProgram->Base.Parameters = shProg->Uniforms;
|
||||||
|
assert(shProg->Uniforms);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (shProg->VertexProgram) {
|
if (shProg->VertexProgram) {
|
||||||
|
@@ -1573,7 +1573,8 @@ void _tnl_UpdateFixedFunctionProgram( GLcontext *ctx )
|
|||||||
if (0)
|
if (0)
|
||||||
_mesa_printf("Found existing TNL program for key %x\n", hash);
|
_mesa_printf("Found existing TNL program for key %x\n", hash);
|
||||||
}
|
}
|
||||||
ctx->VertexProgram._Current = ctx->VertexProgram._TnlProgram;
|
_mesa_reference_vertprog(ctx, &ctx->VertexProgram._Current,
|
||||||
|
ctx->VertexProgram._TnlProgram);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Tell the driver about the change. Could define a new target for
|
/* Tell the driver about the change. Could define a new target for
|
||||||
|
Reference in New Issue
Block a user