DC hook proc thunk management simplified.

diff --git a/objects/dc.c b/objects/dc.c
index 7d074bc..1dfb530 100644
--- a/objects/dc.c
+++ b/objects/dc.c
@@ -99,6 +99,7 @@
     dc->saveLevel  = 0;
     dc->dwHookData = 0L;
     dc->hookProc   = NULL;
+    dc->hookThunk  = NULL;
     dc->wndOrgX    = 0;
     dc->wndOrgY    = 0;
     dc->wndExtX    = 1;
@@ -680,8 +681,8 @@
     TRACE("%04x\n", hdc );
 
     /* Call hook procedure to check whether is it OK to delete this DC */
-    if (    dc->hookProc && !(dc->w.flags & (DC_SAVED | DC_MEMORY))
-         && dc->hookProc( hdc, DCHC_DELETEDC, dc->dwHookData, 0 ) == FALSE )
+    if (    dc->hookThunk && !(dc->w.flags & (DC_SAVED | DC_MEMORY))
+         && dc->hookThunk( hdc, DCHC_DELETEDC, dc->dwHookData, 0 ) == FALSE )
     {
         GDI_HEAP_UNLOCK( hdc );
         return FALSE;
@@ -710,6 +711,7 @@
     if (dc->w.hVisRgn) DeleteObject( dc->w.hVisRgn );
     if (dc->w.hGCClipRgn) DeleteObject( dc->w.hGCClipRgn );
     if (dc->w.pAbortProc) THUNK_Free( (FARPROC)dc->w.pAbortProc );
+    if (dc->hookThunk) THUNK_Free( (FARPROC)dc->hookThunk );
     PATH_DestroyGdiPath(&dc->w.path);
     
     return GDI_FreeObject( hdc );
@@ -1201,16 +1203,35 @@
 /***********************************************************************
  *           SetDCHook   (GDI.190)
  */
+/* ### start build ### */
+extern WORD CALLBACK GDI_CallTo16_word_wwll(FARPROC16,WORD,WORD,LONG,LONG);
+/* ### stop build ### */
 BOOL16 WINAPI SetDCHook( HDC16 hdc, FARPROC16 hookProc, DWORD dwHookData )
 {
     DC *dc = (DC *)GDI_GetObjPtr( hdc, DC_MAGIC );
-
-    TRACE("hookProc %08x, default is %08x\n",
-                (UINT)hookProc, (UINT)DCHook16 );
-
     if (!dc) return FALSE;
+
+    /*
+     * Note: We store the original SEGPTR 'hookProc' as we need to be
+     *       able to return it verbatim in GetDCHook,
+     *
+     *       On the other hand, we still call THUNK_Alloc and store the
+     *       32-bit thunk into another DC member, because THUNK_Alloc
+     *       recognizes the (typical) case that the 'hookProc' is indeed
+     *       the 16-bit API entry point of a built-in routine (e.g. DCHook16)
+     *
+     *       We could perform that test every time the hook is about to
+     *       be called (or else we could live with the 32->16->32 detour),
+     *       but this way is the most efficient ...
+     */
+
     dc->hookProc = hookProc;
     dc->dwHookData = dwHookData;
+
+    THUNK_Free( (FARPROC)dc->hookThunk );
+    dc->hookThunk = (DCHOOKPROC)
+                    THUNK_Alloc( hookProc, (RELAY)GDI_CallTo16_word_wwll );
+
     GDI_HEAP_UNLOCK( hdc );
     return TRUE;
 }