Commit 73593cbf authored by Alexandre Julliard's avatar Alexandre Julliard

gdi32: Store a separate flag to mark a GDI object for delayed destruction.

parent 827e1f1d
...@@ -586,7 +586,7 @@ static HGDIOBJ BITMAP_SelectObject( HGDIOBJ handle, HDC hdc ) ...@@ -586,7 +586,7 @@ static HGDIOBJ BITMAP_SelectObject( HGDIOBJ handle, HDC hdc )
goto done; goto done;
} }
if (bitmap->header.dwCount && (handle != GetStockObject(DEFAULT_BITMAP))) if (bitmap->header.selcount && (handle != GetStockObject(DEFAULT_BITMAP)))
{ {
WARN( "Bitmap already selected in another DC\n" ); WARN( "Bitmap already selected in another DC\n" );
GDI_ReleaseObj( handle ); GDI_ReleaseObj( handle );
......
...@@ -59,9 +59,10 @@ struct hdc_list ...@@ -59,9 +59,10 @@ struct hdc_list
typedef struct tagGDIOBJHDR typedef struct tagGDIOBJHDR
{ {
WORD type; WORD type; /* object type (one of the OBJ_* constants) */
WORD system : 1; WORD system : 1; /* system object flag */
DWORD dwCount; WORD deleted : 1; /* whether DeleteObject has been called on this object */
DWORD selcount; /* number of times the object is selected in a DC */
const struct gdi_obj_funcs *funcs; const struct gdi_obj_funcs *funcs;
struct hdc_list *hdcs; struct hdc_list *hdcs;
} GDIOBJHDR; } GDIOBJHDR;
......
...@@ -528,7 +528,7 @@ HGDIOBJ GDI_inc_ref_count( HGDIOBJ handle ) ...@@ -528,7 +528,7 @@ HGDIOBJ GDI_inc_ref_count( HGDIOBJ handle )
if ((header = GDI_GetObjPtr( handle, 0 ))) if ((header = GDI_GetObjPtr( handle, 0 )))
{ {
header->dwCount++; header->selcount++;
GDI_ReleaseObj( handle ); GDI_ReleaseObj( handle );
} }
else handle = 0; else handle = 0;
...@@ -548,16 +548,16 @@ BOOL GDI_dec_ref_count( HGDIOBJ handle ) ...@@ -548,16 +548,16 @@ BOOL GDI_dec_ref_count( HGDIOBJ handle )
if ((header = GDI_GetObjPtr( handle, 0 ))) if ((header = GDI_GetObjPtr( handle, 0 )))
{ {
if (header->dwCount) header->dwCount--; assert( header->selcount );
if (header->dwCount != 0x80000000) GDI_ReleaseObj( handle ); if (!--header->selcount && header->deleted)
else
{ {
/* handle delayed DeleteObject*/ /* handle delayed DeleteObject*/
header->dwCount = 0; header->deleted = 0;
GDI_ReleaseObj( handle ); GDI_ReleaseObj( handle );
TRACE( "executing delayed DeleteObject for %p\n", handle ); TRACE( "executing delayed DeleteObject for %p\n", handle );
DeleteObject( handle ); DeleteObject( handle );
} }
else GDI_ReleaseObj( handle );
} }
return header != NULL; return header != NULL;
} }
...@@ -641,11 +641,12 @@ HGDIOBJ alloc_gdi_handle( GDIOBJHDR *obj, WORD type, const struct gdi_obj_funcs ...@@ -641,11 +641,12 @@ HGDIOBJ alloc_gdi_handle( GDIOBJHDR *obj, WORD type, const struct gdi_obj_funcs
int i; int i;
/* initialize the object header */ /* initialize the object header */
obj->type = type; obj->type = type;
obj->system = 0; obj->system = 0;
obj->dwCount = 0; obj->deleted = 0;
obj->funcs = funcs; obj->selcount = 0;
obj->hdcs = NULL; obj->funcs = funcs;
obj->hdcs = NULL;
_EnterSysLevel( &GDI_level ); _EnterSysLevel( &GDI_level );
for (i = next_large_handle + 1; i < MAX_LARGE_HANDLES; i++) for (i = next_large_handle + 1; i < MAX_LARGE_HANDLES; i++)
...@@ -796,10 +797,10 @@ BOOL WINAPI DeleteObject( HGDIOBJ obj ) ...@@ -796,10 +797,10 @@ BOOL WINAPI DeleteObject( HGDIOBJ obj )
if (!header) return FALSE; if (!header) return FALSE;
} }
if (header->dwCount) if (header->selcount)
{ {
TRACE("delayed for %p because object in use, count %d\n", obj, header->dwCount ); TRACE("delayed for %p because object in use, count %u\n", obj, header->selcount );
header->dwCount |= 0x80000000; /* mark for delete */ header->deleted = 1; /* mark for delete */
GDI_ReleaseObj( obj ); GDI_ReleaseObj( obj );
return TRUE; return TRUE;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment