wintrust: Return error directly from SOFTPUB_LoadFileMessage.
[wine] / dlls / user.exe16 / message.c
1 /*
2  * 16-bit messaging support
3  *
4  * Copyright 2001 Alexandre Julliard
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20
21 #include "config.h"
22 #include "wine/port.h"
23
24 #include <assert.h>
25 #include <stdarg.h>
26 #include <string.h>
27
28 #include "wine/winuser16.h"
29 #include "wownt32.h"
30 #include "winerror.h"
31 #include "dde.h"
32 #include "user_private.h"
33 #include "wine/debug.h"
34
35 WINE_DEFAULT_DEBUG_CHANNEL(msg);
36 WINE_DECLARE_DEBUG_CHANNEL(message);
37
38 DWORD USER16_AlertableWait = 0;
39
40 struct wow_handlers32 wow_handlers32;
41
42 static LRESULT cwp_hook_callback( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
43                                   LRESULT *result, void *arg )
44 {
45     CWPSTRUCT cwp;
46
47     cwp.hwnd    = hwnd;
48     cwp.message = msg;
49     cwp.wParam  = wp;
50     cwp.lParam  = lp;
51     *result = 0;
52     return HOOK_CallHooks( WH_CALLWNDPROC, HC_ACTION, 1, (LPARAM)&cwp, FALSE );
53 }
54
55 static LRESULT send_message_callback( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
56                                       LRESULT *result, void *arg )
57 {
58     *result = SendMessageA( hwnd, msg, wp, lp );
59     return *result;
60 }
61
62 static LRESULT post_message_callback( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
63                                       LRESULT *result, void *arg )
64 {
65     *result = 0;
66     return PostMessageA( hwnd, msg, wp, lp );
67 }
68
69 static LRESULT post_thread_message_callback( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
70                                              LRESULT *result, void *arg )
71 {
72     DWORD_PTR tid = (DWORD_PTR)arg;
73     *result = 0;
74     return PostThreadMessageA( tid, msg, wp, lp );
75 }
76
77 static LRESULT get_message_callback( HWND16 hwnd, UINT16 msg, WPARAM16 wp, LPARAM lp,
78                                      LRESULT *result, void *arg )
79 {
80     MSG16 *msg16 = arg;
81
82     msg16->hwnd    = hwnd;
83     msg16->message = msg;
84     msg16->wParam  = wp;
85     msg16->lParam  = lp;
86     *result = 0;
87     return 0;
88 }
89
90 static LRESULT defdlg_proc_callback( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
91                                      LRESULT *result, void *arg )
92 {
93     *result = DefDlgProcA( hwnd, msg, wp, lp );
94     return *result;
95 }
96
97 static LRESULT call_window_proc_callback( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
98                                           LRESULT *result, void *arg )
99 {
100     WNDPROC proc = arg;
101     *result = CallWindowProcA( proc, hwnd, msg, wp, lp );
102     return *result;
103 }
104
105
106 /**********************************************************************
107  * Support for window procedure thunks
108  */
109
110 #include "pshpack1.h"
111 typedef struct
112 {
113     BYTE        popl_eax;        /* popl  %eax (return address) */
114     BYTE        pushl_func;      /* pushl $proc */
115     WNDPROC     proc;
116     BYTE        pushl_eax;       /* pushl %eax */
117     BYTE        ljmp;            /* ljmp relay*/
118     DWORD       relay_offset;    /* __wine_call_wndproc */
119     WORD        relay_sel;
120 } WINPROC_THUNK;
121 #include "poppack.h"
122
123 #define WINPROC_HANDLE (~0u >> 16)
124 #define MAX_WINPROCS32 4096
125 #define MAX_WINPROCS16 1024
126
127 static WNDPROC16 winproc16_array[MAX_WINPROCS16];
128 static unsigned int winproc16_used;
129
130 static WINPROC_THUNK *thunk_array;
131 static UINT thunk_selector;
132
133 /* return the window proc index for a given handle, or -1 for an invalid handle
134  * indices 0 .. MAX_WINPROCS32-1 are for 32-bit procs,
135  * indices MAX_WINPROCS32 .. MAX_WINPROCS32+MAX_WINPROCS16-1 for 16-bit procs */
136 static int winproc_to_index( WNDPROC16 handle )
137 {
138     unsigned int index;
139
140     if (HIWORD(handle) == thunk_selector)
141     {
142         index = LOWORD(handle) / sizeof(WINPROC_THUNK);
143         /* check alignment */
144         if (index * sizeof(WINPROC_THUNK) != LOWORD(handle)) return -1;
145         /* check array limits */
146         if (index >= MAX_WINPROCS32) return -1;
147     }
148     else
149     {
150         index = LOWORD(handle);
151         if ((ULONG_PTR)handle >> 16 != WINPROC_HANDLE) return -1;
152         /* check array limits */
153         if (index >= winproc16_used + MAX_WINPROCS32) return -1;
154     }
155     return index;
156 }
157
158 /* allocate a 16-bit thunk for an existing window proc */
159 static WNDPROC16 alloc_win16_thunk( WNDPROC handle )
160 {
161     static FARPROC16 relay;
162     WINPROC_THUNK *thunk;
163     UINT index = LOWORD( handle );
164
165     if (index >= MAX_WINPROCS32)  /* already a 16-bit proc */
166         return winproc16_array[index - MAX_WINPROCS32];
167
168     if (!thunk_array)  /* allocate the array and its selector */
169     {
170         LDT_ENTRY entry;
171
172         assert( MAX_WINPROCS16 * sizeof(WINPROC_THUNK) <= 0x10000 );
173
174         if (!(thunk_selector = wine_ldt_alloc_entries(1))) return NULL;
175         if (!(thunk_array = VirtualAlloc( NULL, MAX_WINPROCS16 * sizeof(WINPROC_THUNK), MEM_COMMIT,
176                                           PAGE_EXECUTE_READWRITE ))) return NULL;
177         wine_ldt_set_base( &entry, thunk_array );
178         wine_ldt_set_limit( &entry, MAX_WINPROCS16 * sizeof(WINPROC_THUNK) - 1 );
179         wine_ldt_set_flags( &entry, WINE_LDT_FLAGS_CODE | WINE_LDT_FLAGS_32BIT );
180         wine_ldt_set_entry( thunk_selector, &entry );
181         relay = GetProcAddress16( GetModuleHandle16("user"), "__wine_call_wndproc" );
182     }
183
184     thunk = &thunk_array[index];
185     thunk->popl_eax     = 0x58;   /* popl  %eax */
186     thunk->pushl_func   = 0x68;   /* pushl $proc */
187     thunk->proc         = handle;
188     thunk->pushl_eax    = 0x50;   /* pushl %eax */
189     thunk->ljmp         = 0xea;   /* ljmp   relay*/
190     thunk->relay_offset = OFFSETOF(relay);
191     thunk->relay_sel    = SELECTOROF(relay);
192     return (WNDPROC16)MAKESEGPTR( thunk_selector, index * sizeof(WINPROC_THUNK) );
193 }
194
195 /**********************************************************************
196  *           WINPROC_AllocProc16
197  */
198 WNDPROC WINPROC_AllocProc16( WNDPROC16 func )
199 {
200     int index;
201     WNDPROC ret;
202
203     if (!func) return NULL;
204
205     /* check if the function is already a win proc */
206     if ((index = winproc_to_index( func )) != -1)
207         return (WNDPROC)(ULONG_PTR)(index | (WINPROC_HANDLE << 16));
208
209     /* then check if we already have a winproc for that function */
210     for (index = 0; index < winproc16_used; index++)
211         if (winproc16_array[index] == func) goto done;
212
213     if (winproc16_used >= MAX_WINPROCS16)
214     {
215         FIXME( "too many winprocs, cannot allocate one for 16-bit %p\n", func );
216         return NULL;
217     }
218     winproc16_array[winproc16_used++] = func;
219
220 done:
221     ret = (WNDPROC)(ULONG_PTR)((index + MAX_WINPROCS32) | (WINPROC_HANDLE << 16));
222     TRACE( "returning %p for %p/16-bit (%d/%d used)\n",
223            ret, func, winproc16_used, MAX_WINPROCS16 );
224     return ret;
225 }
226
227 /**********************************************************************
228  *           WINPROC_GetProc16
229  *
230  * Get a window procedure pointer that can be passed to the Windows program.
231  */
232 WNDPROC16 WINPROC_GetProc16( WNDPROC proc, BOOL unicode )
233 {
234     WNDPROC winproc = wow_handlers32.alloc_winproc( proc, unicode );
235
236     if ((ULONG_PTR)winproc >> 16 != WINPROC_HANDLE) return (WNDPROC16)winproc;
237     return alloc_win16_thunk( winproc );
238 }
239
240 /* call a 16-bit window procedure */
241 static LRESULT call_window_proc16( HWND16 hwnd, UINT16 msg, WPARAM16 wParam, LPARAM lParam,
242                                    LRESULT *result, void *arg )
243 {
244     WNDPROC16 func = arg;
245     int index = winproc_to_index( func );
246     CONTEXT86 context;
247     size_t size = 0;
248     struct
249     {
250         WORD params[5];
251         union
252         {
253             CREATESTRUCT16 cs16;
254             DRAWITEMSTRUCT16 dis16;
255             COMPAREITEMSTRUCT16 cis16;
256         } u;
257     } args;
258
259     if (index >= MAX_WINPROCS32) func = winproc16_array[index - MAX_WINPROCS32];
260
261     /* Window procedures want ax = hInstance, ds = es = ss */
262
263     memset(&context, 0, sizeof(context));
264     context.SegDs = context.SegEs = SELECTOROF(NtCurrentTeb()->WOW32Reserved);
265     context.SegFs = wine_get_fs();
266     context.SegGs = wine_get_gs();
267     if (!(context.Eax = GetWindowWord( HWND_32(hwnd), GWLP_HINSTANCE ))) context.Eax = context.SegDs;
268     context.SegCs = SELECTOROF(func);
269     context.Eip   = OFFSETOF(func);
270     context.Ebp   = OFFSETOF(NtCurrentTeb()->WOW32Reserved) + FIELD_OFFSET(STACK16FRAME, bp);
271
272     if (lParam)
273     {
274         /* Some programs (eg. the "Undocumented Windows" examples, JWP) only
275            work if structures passed in lParam are placed in the stack/data
276            segment. Programmers easily make the mistake of converting lParam
277            to a near rather than a far pointer, since Windows apparently
278            allows this. We copy the structures to the 16 bit stack; this is
279            ugly but makes these programs work. */
280         switch (msg)
281         {
282           case WM_CREATE:
283           case WM_NCCREATE:
284             size = sizeof(CREATESTRUCT16); break;
285           case WM_DRAWITEM:
286             size = sizeof(DRAWITEMSTRUCT16); break;
287           case WM_COMPAREITEM:
288             size = sizeof(COMPAREITEMSTRUCT16); break;
289         }
290         if (size)
291         {
292             memcpy( &args.u, MapSL(lParam), size );
293             lParam = PtrToUlong(NtCurrentTeb()->WOW32Reserved) - size;
294         }
295     }
296
297     args.params[4] = hwnd;
298     args.params[3] = msg;
299     args.params[2] = wParam;
300     args.params[1] = HIWORD(lParam);
301     args.params[0] = LOWORD(lParam);
302     WOWCallback16Ex( 0, WCB16_REGS, sizeof(args.params) + size, &args, (DWORD *)&context );
303     *result = MAKELONG( LOWORD(context.Eax), LOWORD(context.Edx) );
304     return *result;
305 }
306
307 static LRESULT call_dialog_proc16( HWND16 hwnd, UINT16 msg, WPARAM16 wp, LPARAM lp,
308                                    LRESULT *result, void *arg )
309 {
310     LRESULT ret = call_window_proc16( hwnd, msg, wp, lp, result, arg );
311     *result = GetWindowLongPtrW( WIN_Handle32(hwnd), DWLP_MSGRESULT );
312     return LOWORD(ret);
313 }
314
315 static LRESULT call_window_proc_Ato16( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
316                                        LRESULT *result, void *arg )
317 {
318     return WINPROC_CallProc32ATo16( call_window_proc16, hwnd, msg, wp, lp, result, arg );
319 }
320
321 static LRESULT call_dialog_proc_Ato16( HWND hwnd, UINT msg, WPARAM wp, LPARAM lp,
322                                        LRESULT *result, void *arg )
323 {
324     return WINPROC_CallProc32ATo16( call_dialog_proc16, hwnd, msg, wp, lp, result, arg );
325 }
326
327
328
329 /**********************************************************************
330  * Support for Edit word break proc thunks
331  */
332
333 #define MAX_THUNKS 32
334
335 #include <pshpack1.h>
336 static struct word_break_thunk
337 {
338     BYTE                popl_eax;       /* popl  %eax (return address) */
339     BYTE                pushl_proc16;   /* pushl proc16 */
340     EDITWORDBREAKPROC16 proc16;
341     BYTE                pushl_eax;      /* pushl %eax */
342     BYTE                jmp;            /* ljmp call_word_break_proc16 */
343     DWORD               callback;
344 } *word_break_thunks;
345 #include <poppack.h>
346
347 /**********************************************************************
348  *           call_word_break_proc16
349  */
350 static INT16 CALLBACK call_word_break_proc16( SEGPTR proc16, LPSTR text, INT index, INT count, INT action )
351 {
352     SEGPTR segptr;
353     WORD args[5];
354     DWORD result;
355
356     segptr = MapLS( text );
357     args[4] = SELECTOROF(segptr);
358     args[3] = OFFSETOF(segptr);
359     args[2] = index;
360     args[1] = count;
361     args[0] = action;
362     WOWCallback16Ex( proc16, WCB16_PASCAL, sizeof(args), args, &result );
363     UnMapLS( segptr );
364     return LOWORD(result);
365 }
366
367 /******************************************************************
368  *              add_word_break_thunk
369  */
370 static struct word_break_thunk *add_word_break_thunk( EDITWORDBREAKPROC16 proc16 )
371 {
372     struct word_break_thunk *thunk;
373
374     if (!word_break_thunks)
375     {
376         word_break_thunks = VirtualAlloc( NULL, MAX_THUNKS * sizeof(*thunk),
377                                           MEM_COMMIT, PAGE_EXECUTE_READWRITE );
378         if (!word_break_thunks) return NULL;
379
380         for (thunk = word_break_thunks; thunk < &word_break_thunks[MAX_THUNKS]; thunk++)
381         {
382             thunk->popl_eax     = 0x58;   /* popl  %eax */
383             thunk->pushl_proc16 = 0x68;   /* pushl proc16 */
384             thunk->pushl_eax    = 0x50;   /* pushl %eax */
385             thunk->jmp          = 0xe9;   /* jmp call_word_break_proc16 */
386             thunk->callback     = (char *)call_word_break_proc16 - (char *)(&thunk->callback + 1);
387         }
388     }
389     for (thunk = word_break_thunks; thunk < &word_break_thunks[MAX_THUNKS]; thunk++)
390         if (thunk->proc16 == proc16) return thunk;
391
392     for (thunk = word_break_thunks; thunk < &word_break_thunks[MAX_THUNKS]; thunk++)
393     {
394         if (thunk->proc16) continue;
395         thunk->proc16 = proc16;
396         return thunk;
397     }
398     FIXME("Out of word break thunks\n");
399     return NULL;
400 }
401
402 /******************************************************************
403  *              get_word_break_thunk
404  */
405 static EDITWORDBREAKPROC16 get_word_break_thunk( EDITWORDBREAKPROCA proc )
406 {
407     struct word_break_thunk *thunk = (struct word_break_thunk *)proc;
408     if (word_break_thunks && thunk >= word_break_thunks && thunk < &word_break_thunks[MAX_THUNKS])
409         return thunk->proc16;
410     return NULL;
411 }
412
413
414 /***********************************************************************
415  * Support for 16<->32 message mapping
416  */
417
418 static inline void *get_buffer( void *static_buffer, size_t size, size_t need )
419 {
420     if (size >= need) return static_buffer;
421     return HeapAlloc( GetProcessHeap(), 0, need );
422 }
423
424 static inline void free_buffer( void *static_buffer, void *buffer )
425 {
426     if (buffer != static_buffer) HeapFree( GetProcessHeap(), 0, buffer );
427 }
428
429 static void RECT16to32( const RECT16 *from, RECT *to )
430 {
431     to->left   = from->left;
432     to->top    = from->top;
433     to->right  = from->right;
434     to->bottom = from->bottom;
435 }
436
437 static void RECT32to16( const RECT *from, RECT16 *to )
438 {
439     to->left   = from->left;
440     to->top    = from->top;
441     to->right  = from->right;
442     to->bottom = from->bottom;
443 }
444
445 static void MINMAXINFO32to16( const MINMAXINFO *from, MINMAXINFO16 *to )
446 {
447     to->ptReserved.x     = from->ptReserved.x;
448     to->ptReserved.y     = from->ptReserved.y;
449     to->ptMaxSize.x      = from->ptMaxSize.x;
450     to->ptMaxSize.y      = from->ptMaxSize.y;
451     to->ptMaxPosition.x  = from->ptMaxPosition.x;
452     to->ptMaxPosition.y  = from->ptMaxPosition.y;
453     to->ptMinTrackSize.x = from->ptMinTrackSize.x;
454     to->ptMinTrackSize.y = from->ptMinTrackSize.y;
455     to->ptMaxTrackSize.x = from->ptMaxTrackSize.x;
456     to->ptMaxTrackSize.y = from->ptMaxTrackSize.y;
457 }
458
459 static void MINMAXINFO16to32( const MINMAXINFO16 *from, MINMAXINFO *to )
460 {
461     to->ptReserved.x     = from->ptReserved.x;
462     to->ptReserved.y     = from->ptReserved.y;
463     to->ptMaxSize.x      = from->ptMaxSize.x;
464     to->ptMaxSize.y      = from->ptMaxSize.y;
465     to->ptMaxPosition.x  = from->ptMaxPosition.x;
466     to->ptMaxPosition.y  = from->ptMaxPosition.y;
467     to->ptMinTrackSize.x = from->ptMinTrackSize.x;
468     to->ptMinTrackSize.y = from->ptMinTrackSize.y;
469     to->ptMaxTrackSize.x = from->ptMaxTrackSize.x;
470     to->ptMaxTrackSize.y = from->ptMaxTrackSize.y;
471 }
472
473 static void WINDOWPOS32to16( const WINDOWPOS* from, WINDOWPOS16* to )
474 {
475     to->hwnd            = HWND_16(from->hwnd);
476     to->hwndInsertAfter = HWND_16(from->hwndInsertAfter);
477     to->x               = from->x;
478     to->y               = from->y;
479     to->cx              = from->cx;
480     to->cy              = from->cy;
481     to->flags           = from->flags;
482 }
483
484 static void WINDOWPOS16to32( const WINDOWPOS16* from, WINDOWPOS* to )
485 {
486     to->hwnd            = WIN_Handle32(from->hwnd);
487     to->hwndInsertAfter = (from->hwndInsertAfter == (HWND16)-1) ?
488                            HWND_TOPMOST : WIN_Handle32(from->hwndInsertAfter);
489     to->x               = from->x;
490     to->y               = from->y;
491     to->cx              = from->cx;
492     to->cy              = from->cy;
493     to->flags           = from->flags;
494 }
495
496 /* The strings are not copied */
497 static void CREATESTRUCT32Ato16( const CREATESTRUCTA* from, CREATESTRUCT16* to )
498 {
499     to->lpCreateParams = (SEGPTR)from->lpCreateParams;
500     to->hInstance      = HINSTANCE_16(from->hInstance);
501     to->hMenu          = HMENU_16(from->hMenu);
502     to->hwndParent     = HWND_16(from->hwndParent);
503     to->cy             = from->cy;
504     to->cx             = from->cx;
505     to->y              = from->y;
506     to->x              = from->x;
507     to->style          = from->style;
508     to->dwExStyle      = from->dwExStyle;
509 }
510
511 static void CREATESTRUCT16to32A( const CREATESTRUCT16* from, CREATESTRUCTA *to )
512
513 {
514     to->lpCreateParams = (LPVOID)from->lpCreateParams;
515     to->hInstance      = HINSTANCE_32(from->hInstance);
516     to->hMenu          = HMENU_32(from->hMenu);
517     to->hwndParent     = WIN_Handle32(from->hwndParent);
518     to->cy             = from->cy;
519     to->cx             = from->cx;
520     to->y              = from->y;
521     to->x              = from->x;
522     to->style          = from->style;
523     to->dwExStyle      = from->dwExStyle;
524     to->lpszName       = MapSL(from->lpszName);
525     to->lpszClass      = MapSL(from->lpszClass);
526 }
527
528 /* The strings are not copied */
529 static void MDICREATESTRUCT32Ato16( const MDICREATESTRUCTA* from, MDICREATESTRUCT16* to )
530 {
531     to->hOwner = HINSTANCE_16(from->hOwner);
532     to->x      = from->x;
533     to->y      = from->y;
534     to->cx     = from->cx;
535     to->cy     = from->cy;
536     to->style  = from->style;
537     to->lParam = from->lParam;
538 }
539
540 static void MDICREATESTRUCT16to32A( const MDICREATESTRUCT16* from, MDICREATESTRUCTA *to )
541 {
542     to->hOwner = HINSTANCE_32(from->hOwner);
543     to->x      = from->x;
544     to->y      = from->y;
545     to->cx     = from->cx;
546     to->cy     = from->cy;
547     to->style  = from->style;
548     to->lParam = from->lParam;
549     to->szTitle = MapSL(from->szTitle);
550     to->szClass = MapSL(from->szClass);
551 }
552
553 static UINT_PTR convert_handle_16_to_32(HANDLE16 src, unsigned int flags)
554 {
555     HANDLE      dst;
556     UINT        sz = GlobalSize16(src);
557     LPSTR       ptr16, ptr32;
558
559     if (!(dst = GlobalAlloc(flags, sz)))
560         return 0;
561     ptr16 = GlobalLock16(src);
562     ptr32 = GlobalLock(dst);
563     if (ptr16 != NULL && ptr32 != NULL) memcpy(ptr32, ptr16, sz);
564     GlobalUnlock16(src);
565     GlobalUnlock(dst);
566
567     return (UINT_PTR)dst;
568 }
569
570 static HANDLE16 convert_handle_32_to_16(UINT_PTR src, unsigned int flags)
571 {
572     HANDLE16    dst;
573     UINT        sz = GlobalSize((HANDLE)src);
574     LPSTR       ptr16, ptr32;
575
576     if (!(dst = GlobalAlloc16(flags, sz)))
577         return 0;
578     ptr32 = GlobalLock((HANDLE)src);
579     ptr16 = GlobalLock16(dst);
580     if (ptr16 != NULL && ptr32 != NULL) memcpy(ptr16, ptr32, sz);
581     GlobalUnlock((HANDLE)src);
582     GlobalUnlock16(dst);
583
584     return dst;
585 }
586
587 static BOOL is_old_app( HWND hwnd )
588 {
589     HINSTANCE inst = (HINSTANCE)GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
590     return inst && !((ULONG_PTR)inst >> 16) && (GetExpWinVer16(LOWORD(inst)) & 0xFF00) == 0x0300;
591 }
592
593 static int find_sub_menu( HMENU *hmenu, HMENU16 target )
594 {
595     int i, pos, count = GetMenuItemCount( *hmenu );
596
597     for (i = 0; i < count; i++)
598     {
599         HMENU sub = GetSubMenu( *hmenu, i );
600         if (!sub) continue;
601         if (HMENU_16(sub) == target) return i;
602         if ((pos = find_sub_menu( &sub, target )) != -1)
603         {
604             *hmenu = sub;
605             return pos;
606         }
607     }
608     return -1;
609 }
610
611 /**********************************************************************
612  *           WINPROC_CallProc16To32A
613  */
614 LRESULT WINPROC_CallProc16To32A( winproc_callback_t callback, HWND16 hwnd, UINT16 msg,
615                                  WPARAM16 wParam, LPARAM lParam, LRESULT *result, void *arg )
616 {
617     LRESULT ret = 0;
618     HWND hwnd32 = WIN_Handle32( hwnd );
619
620     switch(msg)
621     {
622     case WM_NCCREATE:
623     case WM_CREATE:
624         {
625             CREATESTRUCT16 *cs16 = MapSL(lParam);
626             CREATESTRUCTA cs;
627             MDICREATESTRUCTA mdi_cs;
628
629             CREATESTRUCT16to32A( cs16, &cs );
630             if (GetWindowLongW(hwnd32, GWL_EXSTYLE) & WS_EX_MDICHILD)
631             {
632                 MDICREATESTRUCT16 *mdi_cs16 = MapSL(cs16->lpCreateParams);
633                 MDICREATESTRUCT16to32A(mdi_cs16, &mdi_cs);
634                 cs.lpCreateParams = &mdi_cs;
635             }
636             ret = callback( hwnd32, msg, wParam, (LPARAM)&cs, result, arg );
637             CREATESTRUCT32Ato16( &cs, cs16 );
638         }
639         break;
640     case WM_MDICREATE:
641         {
642             MDICREATESTRUCT16 *cs16 = MapSL(lParam);
643             MDICREATESTRUCTA cs;
644
645             MDICREATESTRUCT16to32A( cs16, &cs );
646             ret = callback( hwnd32, msg, wParam, (LPARAM)&cs, result, arg );
647             MDICREATESTRUCT32Ato16( &cs, cs16 );
648         }
649         break;
650     case WM_MDIACTIVATE:
651         if (lParam)
652             ret = callback( hwnd32, msg, (WPARAM)WIN_Handle32( HIWORD(lParam) ),
653                             (LPARAM)WIN_Handle32( LOWORD(lParam) ), result, arg );
654         else /* message sent to MDI client */
655             ret = callback( hwnd32, msg, wParam, lParam, result, arg );
656         break;
657     case WM_MDIGETACTIVE:
658         {
659             BOOL maximized = FALSE;
660             ret = callback( hwnd32, msg, wParam, (LPARAM)&maximized, result, arg );
661             *result = MAKELRESULT( LOWORD(*result), maximized );
662         }
663         break;
664     case WM_MDISETMENU:
665         ret = callback( hwnd32, wParam ? WM_MDIREFRESHMENU : WM_MDISETMENU,
666                         (WPARAM)HMENU_32(LOWORD(lParam)), (LPARAM)HMENU_32(HIWORD(lParam)),
667                         result, arg );
668         break;
669     case WM_GETMINMAXINFO:
670         {
671             MINMAXINFO16 *mmi16 = MapSL(lParam);
672             MINMAXINFO mmi;
673
674             MINMAXINFO16to32( mmi16, &mmi );
675             ret = callback( hwnd32, msg, wParam, (LPARAM)&mmi, result, arg );
676             MINMAXINFO32to16( &mmi, mmi16 );
677         }
678         break;
679     case WM_WINDOWPOSCHANGING:
680     case WM_WINDOWPOSCHANGED:
681         {
682             WINDOWPOS16 *winpos16 = MapSL(lParam);
683             WINDOWPOS winpos;
684
685             WINDOWPOS16to32( winpos16, &winpos );
686             ret = callback( hwnd32, msg, wParam, (LPARAM)&winpos, result, arg );
687             WINDOWPOS32to16( &winpos, winpos16 );
688         }
689         break;
690     case WM_NCCALCSIZE:
691         {
692             NCCALCSIZE_PARAMS16 *nc16 = MapSL(lParam);
693             NCCALCSIZE_PARAMS nc;
694             WINDOWPOS winpos;
695
696             RECT16to32( &nc16->rgrc[0], &nc.rgrc[0] );
697             if (wParam)
698             {
699                 RECT16to32( &nc16->rgrc[1], &nc.rgrc[1] );
700                 RECT16to32( &nc16->rgrc[2], &nc.rgrc[2] );
701                 WINDOWPOS16to32( MapSL(nc16->lppos), &winpos );
702                 nc.lppos = &winpos;
703             }
704             ret = callback( hwnd32, msg, wParam, (LPARAM)&nc, result, arg );
705             RECT32to16( &nc.rgrc[0], &nc16->rgrc[0] );
706             if (wParam)
707             {
708                 RECT32to16( &nc.rgrc[1], &nc16->rgrc[1] );
709                 RECT32to16( &nc.rgrc[2], &nc16->rgrc[2] );
710                 WINDOWPOS32to16( &winpos, MapSL(nc16->lppos) );
711             }
712         }
713         break;
714     case WM_COMPAREITEM:
715         {
716             COMPAREITEMSTRUCT16* cis16 = MapSL(lParam);
717             COMPAREITEMSTRUCT cis;
718             cis.CtlType    = cis16->CtlType;
719             cis.CtlID      = cis16->CtlID;
720             cis.hwndItem   = WIN_Handle32( cis16->hwndItem );
721             cis.itemID1    = cis16->itemID1;
722             cis.itemData1  = cis16->itemData1;
723             cis.itemID2    = cis16->itemID2;
724             cis.itemData2  = cis16->itemData2;
725             cis.dwLocaleId = 0;  /* FIXME */
726             ret = callback( hwnd32, msg, wParam, (LPARAM)&cis, result, arg );
727         }
728         break;
729     case WM_DELETEITEM:
730         {
731             DELETEITEMSTRUCT16* dis16 = MapSL(lParam);
732             DELETEITEMSTRUCT dis;
733             dis.CtlType  = dis16->CtlType;
734             dis.CtlID    = dis16->CtlID;
735             dis.hwndItem = WIN_Handle32( dis16->hwndItem );
736             dis.itemData = dis16->itemData;
737             ret = callback( hwnd32, msg, wParam, (LPARAM)&dis, result, arg );
738         }
739         break;
740     case WM_MEASUREITEM:
741         {
742             MEASUREITEMSTRUCT16* mis16 = MapSL(lParam);
743             MEASUREITEMSTRUCT mis;
744             mis.CtlType    = mis16->CtlType;
745             mis.CtlID      = mis16->CtlID;
746             mis.itemID     = mis16->itemID;
747             mis.itemWidth  = mis16->itemWidth;
748             mis.itemHeight = mis16->itemHeight;
749             mis.itemData   = mis16->itemData;
750             ret = callback( hwnd32, msg, wParam, (LPARAM)&mis, result, arg );
751             mis16->itemWidth  = (UINT16)mis.itemWidth;
752             mis16->itemHeight = (UINT16)mis.itemHeight;
753         }
754         break;
755     case WM_DRAWITEM:
756         {
757             DRAWITEMSTRUCT16* dis16 = MapSL(lParam);
758             DRAWITEMSTRUCT dis;
759             dis.CtlType       = dis16->CtlType;
760             dis.CtlID         = dis16->CtlID;
761             dis.itemID        = dis16->itemID;
762             dis.itemAction    = dis16->itemAction;
763             dis.itemState     = dis16->itemState;
764             dis.hwndItem      = (dis.CtlType == ODT_MENU) ? (HWND)HMENU_32(dis16->hwndItem)
765                                                           : WIN_Handle32( dis16->hwndItem );
766             dis.hDC           = HDC_32(dis16->hDC);
767             dis.itemData      = dis16->itemData;
768             dis.rcItem.left   = dis16->rcItem.left;
769             dis.rcItem.top    = dis16->rcItem.top;
770             dis.rcItem.right  = dis16->rcItem.right;
771             dis.rcItem.bottom = dis16->rcItem.bottom;
772             ret = callback( hwnd32, msg, wParam, (LPARAM)&dis, result, arg );
773         }
774         break;
775     case WM_COPYDATA:
776         {
777             COPYDATASTRUCT16 *cds16 = MapSL(lParam);
778             COPYDATASTRUCT cds;
779             cds.dwData = cds16->dwData;
780             cds.cbData = cds16->cbData;
781             cds.lpData = MapSL(cds16->lpData);
782             ret = callback( hwnd32, msg, wParam, (LPARAM)&cds, result, arg );
783         }
784         break;
785     case WM_GETDLGCODE:
786         if (lParam)
787         {
788             MSG16 *msg16 = MapSL(lParam);
789             MSG msg32;
790             msg32.hwnd    = WIN_Handle32( msg16->hwnd );
791             msg32.message = msg16->message;
792             msg32.wParam  = msg16->wParam;
793             msg32.lParam  = msg16->lParam;
794             msg32.time    = msg16->time;
795             msg32.pt.x    = msg16->pt.x;
796             msg32.pt.y    = msg16->pt.y;
797             ret = callback( hwnd32, msg, wParam, (LPARAM)&msg32, result, arg );
798         }
799         else
800             ret = callback( hwnd32, msg, wParam, lParam, result, arg );
801         break;
802     case WM_NEXTMENU:
803         {
804             MDINEXTMENU next;
805             next.hmenuIn   = (HMENU)lParam;
806             next.hmenuNext = 0;
807             next.hwndNext  = 0;
808             ret = callback( hwnd32, msg, wParam, (LPARAM)&next, result, arg );
809             *result = MAKELONG( HMENU_16(next.hmenuNext), HWND_16(next.hwndNext) );
810         }
811         break;
812     case WM_ACTIVATE:
813     case WM_CHARTOITEM:
814     case WM_COMMAND:
815     case WM_VKEYTOITEM:
816         ret = callback( hwnd32, msg, MAKEWPARAM( wParam, HIWORD(lParam) ),
817                         (LPARAM)WIN_Handle32( LOWORD(lParam) ), result, arg );
818         break;
819     case WM_HSCROLL:
820     case WM_VSCROLL:
821         ret = callback( hwnd32, msg, MAKEWPARAM( wParam, LOWORD(lParam) ),
822                         (LPARAM)WIN_Handle32( HIWORD(lParam) ), result, arg );
823         break;
824     case WM_CTLCOLOR:
825         if (HIWORD(lParam) <= CTLCOLOR_STATIC)
826             ret = callback( hwnd32, WM_CTLCOLORMSGBOX + HIWORD(lParam),
827                             (WPARAM)HDC_32(wParam), (LPARAM)WIN_Handle32( LOWORD(lParam) ),
828                             result, arg );
829         break;
830     case WM_GETTEXT:
831     case WM_SETTEXT:
832     case WM_WININICHANGE:
833     case WM_DEVMODECHANGE:
834     case WM_ASKCBFORMATNAME:
835     case WM_NOTIFY:
836         ret = callback( hwnd32, msg, wParam, (LPARAM)MapSL(lParam), result, arg );
837         break;
838     case WM_MENUCHAR:
839         ret = callback( hwnd32, msg, MAKEWPARAM( wParam, LOWORD(lParam) ),
840                         (LPARAM)HMENU_32(HIWORD(lParam)), result, arg );
841         break;
842     case WM_MENUSELECT:
843         if((LOWORD(lParam) & MF_POPUP) && (LOWORD(lParam) != 0xFFFF))
844         {
845             HMENU hmenu = HMENU_32(HIWORD(lParam));
846             int pos = find_sub_menu( &hmenu, wParam );
847             if (pos == -1) pos = 0;
848             wParam = pos;
849         }
850         ret = callback( hwnd32, msg, MAKEWPARAM( wParam, LOWORD(lParam) ),
851                         (LPARAM)HMENU_32(HIWORD(lParam)), result, arg );
852         break;
853     case WM_PARENTNOTIFY:
854         if ((wParam == WM_CREATE) || (wParam == WM_DESTROY))
855             ret = callback( hwnd32, msg, MAKEWPARAM( wParam, HIWORD(lParam) ),
856                             (LPARAM)WIN_Handle32( LOWORD(lParam) ), result, arg );
857         else
858             ret = callback( hwnd32, msg, wParam, lParam, result, arg );
859         break;
860     case WM_ACTIVATEAPP:
861         /* We need this when SetActiveWindow sends a Sendmessage16() to
862          * a 32-bit window. Might be superfluous with 32-bit interprocess
863          * message queues. */
864         if (lParam) lParam = HTASK_32(lParam);
865         ret = callback( hwnd32, msg, wParam, lParam, result, arg );
866         break;
867     case WM_DDE_INITIATE:
868     case WM_DDE_TERMINATE:
869     case WM_DDE_UNADVISE:
870     case WM_DDE_REQUEST:
871         ret = callback( hwnd32, msg, (WPARAM)WIN_Handle32(wParam), lParam, result, arg );
872         break;
873     case WM_DDE_ADVISE:
874     case WM_DDE_DATA:
875     case WM_DDE_POKE:
876         {
877             HANDLE16 lo16 = LOWORD(lParam);
878             UINT_PTR lo32 = 0;
879             if (lo16 && !(lo32 = convert_handle_16_to_32(lo16, GMEM_DDESHARE))) break;
880             lParam = PackDDElParam( msg, lo32, HIWORD(lParam) );
881             ret = callback( hwnd32, msg, (WPARAM)WIN_Handle32(wParam), lParam, result, arg );
882         }
883         break; /* FIXME don't know how to free allocated memory (handle)  !! */
884     case WM_DDE_ACK:
885         {
886             UINT_PTR lo = LOWORD(lParam);
887             UINT_PTR hi = HIWORD(lParam);
888             int flag = 0;
889             char buf[2];
890
891             if (GlobalGetAtomNameA(hi, buf, 2) > 0) flag |= 1;
892             if (GlobalSize16(hi) != 0) flag |= 2;
893             switch (flag)
894             {
895             case 0:
896                 if (hi)
897                 {
898                     MESSAGE("DDE_ACK: neither atom nor handle!!!\n");
899                     hi = 0;
900                 }
901                 break;
902             case 1:
903                 break; /* atom, nothing to do */
904             case 3:
905                 MESSAGE("DDE_ACK: %lx both atom and handle... choosing handle\n", hi);
906                 /* fall thru */
907             case 2:
908                 hi = convert_handle_16_to_32(hi, GMEM_DDESHARE);
909                 break;
910             }
911             lParam = PackDDElParam( WM_DDE_ACK, lo, hi );
912             ret = callback( hwnd32, msg, (WPARAM)WIN_Handle32(wParam), lParam, result, arg );
913         }
914         break; /* FIXME don't know how to free allocated memory (handle) !! */
915     case WM_DDE_EXECUTE:
916         lParam = convert_handle_16_to_32( lParam, GMEM_DDESHARE );
917         ret = callback( hwnd32, msg, wParam, lParam, result, arg );
918         break; /* FIXME don't know how to free allocated memory (handle) !! */
919     case WM_PAINTCLIPBOARD:
920     case WM_SIZECLIPBOARD:
921         FIXME_(msg)( "message %04x needs translation\n", msg );
922         break;
923     default:
924         ret = callback( hwnd32, msg, wParam, lParam, result, arg );
925         break;
926     }
927     return ret;
928 }
929
930
931 /**********************************************************************
932  *           WINPROC_CallProc32ATo16
933  *
934  * Call a 16-bit window procedure, translating the 32-bit args.
935  */
936 LRESULT WINPROC_CallProc32ATo16( winproc_callback16_t callback, HWND hwnd, UINT msg,
937                                  WPARAM wParam, LPARAM lParam, LRESULT *result, void *arg )
938 {
939     LRESULT ret = 0;
940
941     switch(msg)
942     {
943     case WM_NCCREATE:
944     case WM_CREATE:
945         {
946             CREATESTRUCTA *cs32 = (CREATESTRUCTA *)lParam;
947             CREATESTRUCT16 cs;
948             MDICREATESTRUCT16 mdi_cs16;
949             BOOL mdi_child = (GetWindowLongW(hwnd, GWL_EXSTYLE) & WS_EX_MDICHILD);
950
951             CREATESTRUCT32Ato16( cs32, &cs );
952             cs.lpszName  = MapLS( cs32->lpszName );
953             cs.lpszClass = MapLS( cs32->lpszClass );
954
955             if (mdi_child)
956             {
957                 MDICREATESTRUCTA *mdi_cs = cs32->lpCreateParams;
958                 MDICREATESTRUCT32Ato16( mdi_cs, &mdi_cs16 );
959                 mdi_cs16.szTitle = MapLS( mdi_cs->szTitle );
960                 mdi_cs16.szClass = MapLS( mdi_cs->szClass );
961                 cs.lpCreateParams = MapLS( &mdi_cs16 );
962             }
963             lParam = MapLS( &cs );
964             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
965             UnMapLS( lParam );
966             UnMapLS( cs.lpszName );
967             UnMapLS( cs.lpszClass );
968             if (mdi_child)
969             {
970                 UnMapLS( cs.lpCreateParams );
971                 UnMapLS( mdi_cs16.szTitle );
972                 UnMapLS( mdi_cs16.szClass );
973             }
974         }
975         break;
976     case WM_MDICREATE:
977         {
978             MDICREATESTRUCTA *cs32 = (MDICREATESTRUCTA *)lParam;
979             MDICREATESTRUCT16 cs;
980
981             MDICREATESTRUCT32Ato16( cs32, &cs );
982             cs.szTitle = MapLS( cs32->szTitle );
983             cs.szClass = MapLS( cs32->szClass );
984             lParam = MapLS( &cs );
985             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
986             UnMapLS( lParam );
987             UnMapLS( cs.szTitle );
988             UnMapLS( cs.szClass );
989         }
990         break;
991     case WM_MDIACTIVATE:
992         if (GetWindowLongW( hwnd, GWL_EXSTYLE ) & WS_EX_MDICHILD)
993             ret = callback( HWND_16(hwnd), msg, ((HWND)lParam == hwnd),
994                             MAKELPARAM( LOWORD(lParam), LOWORD(wParam) ), result, arg );
995         else
996             ret = callback( HWND_16(hwnd), msg, HWND_16( wParam ), 0, result, arg );
997         break;
998     case WM_MDIGETACTIVE:
999         ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1000         if (lParam) *(BOOL *)lParam = (BOOL16)HIWORD(*result);
1001         *result = (LRESULT)WIN_Handle32( LOWORD(*result) );
1002         break;
1003     case WM_MDISETMENU:
1004         ret = callback( HWND_16(hwnd), msg, (lParam == 0),
1005                         MAKELPARAM( LOWORD(wParam), LOWORD(lParam) ), result, arg );
1006         break;
1007     case WM_GETMINMAXINFO:
1008         {
1009             MINMAXINFO *mmi32 = (MINMAXINFO *)lParam;
1010             MINMAXINFO16 mmi;
1011
1012             MINMAXINFO32to16( mmi32, &mmi );
1013             lParam = MapLS( &mmi );
1014             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1015             UnMapLS( lParam );
1016             MINMAXINFO16to32( &mmi, mmi32 );
1017         }
1018         break;
1019     case WM_NCCALCSIZE:
1020         {
1021             NCCALCSIZE_PARAMS *nc32 = (NCCALCSIZE_PARAMS *)lParam;
1022             NCCALCSIZE_PARAMS16 nc;
1023             WINDOWPOS16 winpos;
1024
1025             RECT32to16( &nc32->rgrc[0], &nc.rgrc[0] );
1026             if (wParam)
1027             {
1028                 RECT32to16( &nc32->rgrc[1], &nc.rgrc[1] );
1029                 RECT32to16( &nc32->rgrc[2], &nc.rgrc[2] );
1030                 WINDOWPOS32to16( nc32->lppos, &winpos );
1031                 nc.lppos = MapLS( &winpos );
1032             }
1033             lParam = MapLS( &nc );
1034             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1035             UnMapLS( lParam );
1036             RECT16to32( &nc.rgrc[0], &nc32->rgrc[0] );
1037             if (wParam)
1038             {
1039                 RECT16to32( &nc.rgrc[1], &nc32->rgrc[1] );
1040                 RECT16to32( &nc.rgrc[2], &nc32->rgrc[2] );
1041                 WINDOWPOS16to32( &winpos, nc32->lppos );
1042                 UnMapLS( nc.lppos );
1043             }
1044         }
1045         break;
1046     case WM_WINDOWPOSCHANGING:
1047     case WM_WINDOWPOSCHANGED:
1048         {
1049             WINDOWPOS *winpos32 = (WINDOWPOS *)lParam;
1050             WINDOWPOS16 winpos;
1051
1052             WINDOWPOS32to16( winpos32, &winpos );
1053             lParam = MapLS( &winpos );
1054             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1055             UnMapLS( lParam );
1056             WINDOWPOS16to32( &winpos, winpos32 );
1057         }
1058         break;
1059     case WM_COMPAREITEM:
1060         {
1061             COMPAREITEMSTRUCT *cis32 = (COMPAREITEMSTRUCT *)lParam;
1062             COMPAREITEMSTRUCT16 cis;
1063             cis.CtlType    = cis32->CtlType;
1064             cis.CtlID      = cis32->CtlID;
1065             cis.hwndItem   = HWND_16( cis32->hwndItem );
1066             cis.itemID1    = cis32->itemID1;
1067             cis.itemData1  = cis32->itemData1;
1068             cis.itemID2    = cis32->itemID2;
1069             cis.itemData2  = cis32->itemData2;
1070             lParam = MapLS( &cis );
1071             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1072             UnMapLS( lParam );
1073         }
1074         break;
1075     case WM_DELETEITEM:
1076         {
1077             DELETEITEMSTRUCT *dis32 = (DELETEITEMSTRUCT *)lParam;
1078             DELETEITEMSTRUCT16 dis;
1079             dis.CtlType  = dis32->CtlType;
1080             dis.CtlID    = dis32->CtlID;
1081             dis.itemID   = dis32->itemID;
1082             dis.hwndItem = (dis.CtlType == ODT_MENU) ? (HWND16)LOWORD(dis32->hwndItem)
1083                                                      : HWND_16( dis32->hwndItem );
1084             dis.itemData = dis32->itemData;
1085             lParam = MapLS( &dis );
1086             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1087             UnMapLS( lParam );
1088         }
1089         break;
1090     case WM_DRAWITEM:
1091         {
1092             DRAWITEMSTRUCT *dis32 = (DRAWITEMSTRUCT *)lParam;
1093             DRAWITEMSTRUCT16 dis;
1094             dis.CtlType       = dis32->CtlType;
1095             dis.CtlID         = dis32->CtlID;
1096             dis.itemID        = dis32->itemID;
1097             dis.itemAction    = dis32->itemAction;
1098             dis.itemState     = dis32->itemState;
1099             dis.hwndItem      = HWND_16( dis32->hwndItem );
1100             dis.hDC           = HDC_16(dis32->hDC);
1101             dis.itemData      = dis32->itemData;
1102             dis.rcItem.left   = dis32->rcItem.left;
1103             dis.rcItem.top    = dis32->rcItem.top;
1104             dis.rcItem.right  = dis32->rcItem.right;
1105             dis.rcItem.bottom = dis32->rcItem.bottom;
1106             lParam = MapLS( &dis );
1107             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1108             UnMapLS( lParam );
1109         }
1110         break;
1111     case WM_MEASUREITEM:
1112         {
1113             MEASUREITEMSTRUCT *mis32 = (MEASUREITEMSTRUCT *)lParam;
1114             MEASUREITEMSTRUCT16 mis;
1115             mis.CtlType    = mis32->CtlType;
1116             mis.CtlID      = mis32->CtlID;
1117             mis.itemID     = mis32->itemID;
1118             mis.itemWidth  = mis32->itemWidth;
1119             mis.itemHeight = mis32->itemHeight;
1120             mis.itemData   = mis32->itemData;
1121             lParam = MapLS( &mis );
1122             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1123             UnMapLS( lParam );
1124             mis32->itemWidth  = mis.itemWidth;
1125             mis32->itemHeight = mis.itemHeight;
1126         }
1127         break;
1128     case WM_COPYDATA:
1129         {
1130             COPYDATASTRUCT *cds32 = (COPYDATASTRUCT *)lParam;
1131             COPYDATASTRUCT16 cds;
1132
1133             cds.dwData = cds32->dwData;
1134             cds.cbData = cds32->cbData;
1135             cds.lpData = MapLS( cds32->lpData );
1136             lParam = MapLS( &cds );
1137             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1138             UnMapLS( lParam );
1139             UnMapLS( cds.lpData );
1140         }
1141         break;
1142     case WM_GETDLGCODE:
1143         if (lParam)
1144         {
1145             MSG *msg32 = (MSG *)lParam;
1146             MSG16 msg16;
1147
1148             msg16.hwnd    = HWND_16( msg32->hwnd );
1149             msg16.message = msg32->message;
1150             msg16.wParam  = msg32->wParam;
1151             msg16.lParam  = msg32->lParam;
1152             msg16.time    = msg32->time;
1153             msg16.pt.x    = msg32->pt.x;
1154             msg16.pt.y    = msg32->pt.y;
1155             lParam = MapLS( &msg16 );
1156             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1157             UnMapLS( lParam );
1158         }
1159         else
1160             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1161         break;
1162     case WM_NEXTMENU:
1163         {
1164             MDINEXTMENU *next = (MDINEXTMENU *)lParam;
1165             ret = callback( HWND_16(hwnd), msg, wParam, (LPARAM)next->hmenuIn, result, arg );
1166             next->hmenuNext = HMENU_32( LOWORD(*result) );
1167             next->hwndNext  = WIN_Handle32( HIWORD(*result) );
1168             *result = 0;
1169         }
1170         break;
1171     case WM_GETTEXT:
1172     case WM_ASKCBFORMATNAME:
1173         wParam = min( wParam, 0xff80 ); /* Must be < 64K */
1174         /* fall through */
1175     case WM_NOTIFY:
1176     case WM_SETTEXT:
1177     case WM_WININICHANGE:
1178     case WM_DEVMODECHANGE:
1179         lParam = MapLS( (void *)lParam );
1180         ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1181         UnMapLS( lParam );
1182         break;
1183     case WM_ACTIVATE:
1184     case WM_CHARTOITEM:
1185     case WM_COMMAND:
1186     case WM_VKEYTOITEM:
1187         ret = callback( HWND_16(hwnd), msg, wParam, MAKELPARAM( (HWND16)lParam, HIWORD(wParam) ),
1188                         result, arg );
1189         break;
1190     case WM_HSCROLL:
1191     case WM_VSCROLL:
1192         ret = callback( HWND_16(hwnd), msg, wParam, MAKELPARAM( HIWORD(wParam), (HWND16)lParam ),
1193                         result, arg );
1194         break;
1195     case WM_CTLCOLORMSGBOX:
1196     case WM_CTLCOLOREDIT:
1197     case WM_CTLCOLORLISTBOX:
1198     case WM_CTLCOLORBTN:
1199     case WM_CTLCOLORDLG:
1200     case WM_CTLCOLORSCROLLBAR:
1201     case WM_CTLCOLORSTATIC:
1202         ret = callback( HWND_16(hwnd), WM_CTLCOLOR, wParam,
1203                         MAKELPARAM( (HWND16)lParam, msg - WM_CTLCOLORMSGBOX ), result, arg );
1204         break;
1205     case WM_MENUSELECT:
1206         if(HIWORD(wParam) & MF_POPUP)
1207         {
1208             HMENU hmenu;
1209             if ((HIWORD(wParam) != 0xffff) || lParam)
1210             {
1211                 if ((hmenu = GetSubMenu( (HMENU)lParam, LOWORD(wParam) )))
1212                 {
1213                     ret = callback( HWND_16(hwnd), msg, HMENU_16(hmenu),
1214                                     MAKELPARAM( HIWORD(wParam), (HMENU16)lParam ), result, arg );
1215                     break;
1216                 }
1217             }
1218         }
1219         /* fall through */
1220     case WM_MENUCHAR:
1221         ret = callback( HWND_16(hwnd), msg, wParam,
1222                         MAKELPARAM( HIWORD(wParam), (HMENU16)lParam ), result, arg );
1223         break;
1224     case WM_PARENTNOTIFY:
1225         if ((LOWORD(wParam) == WM_CREATE) || (LOWORD(wParam) == WM_DESTROY))
1226             ret = callback( HWND_16(hwnd), msg, wParam,
1227                             MAKELPARAM( (HWND16)lParam, HIWORD(wParam) ), result, arg );
1228         else
1229             ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1230         break;
1231     case WM_ACTIVATEAPP:
1232         ret = callback( HWND_16(hwnd), msg, wParam, HTASK_16( lParam ), result, arg );
1233         break;
1234     case WM_PAINT:
1235         if (IsIconic( hwnd ) && GetClassLongPtrW( hwnd, GCLP_HICON ))
1236             ret = callback( HWND_16(hwnd), WM_PAINTICON, 1, lParam, result, arg );
1237         else
1238             ret = callback( HWND_16(hwnd), WM_PAINT, wParam, lParam, result, arg );
1239         break;
1240     case WM_ERASEBKGND:
1241         if (IsIconic( hwnd ) && GetClassLongPtrW( hwnd, GCLP_HICON )) msg = WM_ICONERASEBKGND;
1242         ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1243         break;
1244     case WM_DDE_INITIATE:
1245     case WM_DDE_TERMINATE:
1246     case WM_DDE_UNADVISE:
1247     case WM_DDE_REQUEST:
1248         ret = callback( HWND_16(hwnd), msg, HWND_16(wParam), lParam, result, arg );
1249         break;
1250     case WM_DDE_ADVISE:
1251     case WM_DDE_DATA:
1252     case WM_DDE_POKE:
1253         {
1254             UINT_PTR lo32, hi;
1255             HANDLE16 lo16 = 0;
1256
1257             UnpackDDElParam( msg, lParam, &lo32, &hi );
1258             if (lo32 && !(lo16 = convert_handle_32_to_16(lo32, GMEM_DDESHARE))) break;
1259             ret = callback( HWND_16(hwnd), msg, HWND_16(wParam),
1260                             MAKELPARAM(lo16, hi), result, arg );
1261         }
1262         break; /* FIXME don't know how to free allocated memory (handle)  !! */
1263     case WM_DDE_ACK:
1264         {
1265             UINT_PTR lo, hi;
1266             int flag = 0;
1267             char buf[2];
1268
1269             UnpackDDElParam( msg, lParam, &lo, &hi );
1270
1271             if (GlobalGetAtomNameA((ATOM)hi, buf, sizeof(buf)) > 0) flag |= 1;
1272             if (GlobalSize((HANDLE)hi) != 0) flag |= 2;
1273             switch (flag)
1274             {
1275             case 0:
1276                 if (hi)
1277                 {
1278                     MESSAGE("DDE_ACK: neither atom nor handle!!!\n");
1279                     hi = 0;
1280                 }
1281                 break;
1282             case 1:
1283                 break; /* atom, nothing to do */
1284             case 3:
1285                 MESSAGE("DDE_ACK: %lx both atom and handle... choosing handle\n", hi);
1286                 /* fall thru */
1287             case 2:
1288                 hi = convert_handle_32_to_16(hi, GMEM_DDESHARE);
1289                 break;
1290             }
1291             ret = callback( HWND_16(hwnd), msg, HWND_16(wParam),
1292                             MAKELPARAM(lo, hi), result, arg );
1293         }
1294         break; /* FIXME don't know how to free allocated memory (handle) !! */
1295     case WM_DDE_EXECUTE:
1296         lParam = convert_handle_32_to_16(lParam, GMEM_DDESHARE);
1297         ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1298         break; /* FIXME don't know how to free allocated memory (handle) !! */
1299     case SBM_SETRANGE:
1300         ret = callback( HWND_16(hwnd), SBM_SETRANGE16, 0, MAKELPARAM(wParam, lParam), result, arg );
1301         break;
1302     case SBM_GETRANGE:
1303         ret = callback( HWND_16(hwnd), SBM_GETRANGE16, wParam, lParam, result, arg );
1304         *(LPINT)wParam = LOWORD(*result);
1305         *(LPINT)lParam = HIWORD(*result);
1306         break;
1307     case BM_GETCHECK:
1308     case BM_SETCHECK:
1309     case BM_GETSTATE:
1310     case BM_SETSTATE:
1311     case BM_SETSTYLE:
1312         ret = callback( HWND_16(hwnd), msg + BM_GETCHECK16 - BM_GETCHECK, wParam, lParam, result, arg );
1313         break;
1314     case EM_GETSEL:
1315     case EM_GETRECT:
1316     case EM_SETRECT:
1317     case EM_SETRECTNP:
1318     case EM_SCROLL:
1319     case EM_LINESCROLL:
1320     case EM_SCROLLCARET:
1321     case EM_GETMODIFY:
1322     case EM_SETMODIFY:
1323     case EM_GETLINECOUNT:
1324     case EM_LINEINDEX:
1325     case EM_SETHANDLE:
1326     case EM_GETHANDLE:
1327     case EM_GETTHUMB:
1328     case EM_LINELENGTH:
1329     case EM_REPLACESEL:
1330     case EM_GETLINE:
1331     case EM_LIMITTEXT:
1332     case EM_CANUNDO:
1333     case EM_UNDO:
1334     case EM_FMTLINES:
1335     case EM_LINEFROMCHAR:
1336     case EM_SETTABSTOPS:
1337     case EM_SETPASSWORDCHAR:
1338     case EM_EMPTYUNDOBUFFER:
1339     case EM_GETFIRSTVISIBLELINE:
1340     case EM_SETREADONLY:
1341     case EM_SETWORDBREAKPROC:
1342     case EM_GETWORDBREAKPROC:
1343     case EM_GETPASSWORDCHAR:
1344         ret = callback( HWND_16(hwnd), msg + EM_GETSEL16 - EM_GETSEL, wParam, lParam, result, arg );
1345         break;
1346     case EM_SETSEL:
1347         ret = callback( HWND_16(hwnd), EM_SETSEL16, 0, MAKELPARAM( wParam, lParam ), result, arg );
1348         break;
1349     case LB_CARETOFF:
1350     case LB_CARETON:
1351     case LB_DELETESTRING:
1352     case LB_GETANCHORINDEX:
1353     case LB_GETCARETINDEX:
1354     case LB_GETCOUNT:
1355     case LB_GETCURSEL:
1356     case LB_GETHORIZONTALEXTENT:
1357     case LB_GETITEMDATA:
1358     case LB_GETITEMHEIGHT:
1359     case LB_GETSEL:
1360     case LB_GETSELCOUNT:
1361     case LB_GETTEXTLEN:
1362     case LB_GETTOPINDEX:
1363     case LB_RESETCONTENT:
1364     case LB_SELITEMRANGE:
1365     case LB_SELITEMRANGEEX:
1366     case LB_SETANCHORINDEX:
1367     case LB_SETCARETINDEX:
1368     case LB_SETCOLUMNWIDTH:
1369     case LB_SETCURSEL:
1370     case LB_SETHORIZONTALEXTENT:
1371     case LB_SETITEMDATA:
1372     case LB_SETITEMHEIGHT:
1373     case LB_SETSEL:
1374     case LB_SETTOPINDEX:
1375         ret = callback( HWND_16(hwnd), msg + LB_ADDSTRING16 - LB_ADDSTRING, wParam, lParam, result, arg );
1376         break;
1377     case LB_ADDSTRING:
1378     case LB_FINDSTRING:
1379     case LB_FINDSTRINGEXACT:
1380     case LB_INSERTSTRING:
1381     case LB_SELECTSTRING:
1382     case LB_GETTEXT:
1383     case LB_DIR:
1384     case LB_ADDFILE:
1385         lParam = MapLS( (LPSTR)lParam );
1386         ret = callback( HWND_16(hwnd), msg + LB_ADDSTRING16 - LB_ADDSTRING, wParam, lParam, result, arg );
1387         UnMapLS( lParam );
1388         break;
1389     case LB_GETSELITEMS:
1390         {
1391             INT *items32 = (INT *)lParam;
1392             INT16 *items, buffer[512];
1393             unsigned int i;
1394
1395             wParam = min( wParam, 0x7f80 ); /* Must be < 64K */
1396             if (!(items = get_buffer( buffer, sizeof(buffer), wParam * sizeof(INT16) ))) break;
1397             lParam = MapLS( items );
1398             ret = callback( HWND_16(hwnd), LB_GETSELITEMS16, wParam, lParam, result, arg );
1399             UnMapLS( lParam );
1400             for (i = 0; i < wParam; i++) items32[i] = items[i];
1401             free_buffer( buffer, items );
1402         }
1403         break;
1404     case LB_SETTABSTOPS:
1405         if (wParam)
1406         {
1407             INT *stops32 = (INT *)lParam;
1408             INT16 *stops, buffer[512];
1409             unsigned int i;
1410
1411             wParam = min( wParam, 0x7f80 ); /* Must be < 64K */
1412             if (!(stops = get_buffer( buffer, sizeof(buffer), wParam * sizeof(INT16) ))) break;
1413             for (i = 0; i < wParam; i++) stops[i] = stops32[i];
1414             lParam = MapLS( stops );
1415             ret = callback( HWND_16(hwnd), LB_SETTABSTOPS16, wParam, lParam, result, arg );
1416             UnMapLS( lParam );
1417             free_buffer( buffer, stops );
1418         }
1419         else ret = callback( HWND_16(hwnd), LB_SETTABSTOPS16, wParam, lParam, result, arg );
1420         break;
1421     case CB_DELETESTRING:
1422     case CB_GETCOUNT:
1423     case CB_GETLBTEXTLEN:
1424     case CB_LIMITTEXT:
1425     case CB_RESETCONTENT:
1426     case CB_SETEDITSEL:
1427     case CB_GETCURSEL:
1428     case CB_SETCURSEL:
1429     case CB_SHOWDROPDOWN:
1430     case CB_SETITEMDATA:
1431     case CB_SETITEMHEIGHT:
1432     case CB_GETITEMHEIGHT:
1433     case CB_SETEXTENDEDUI:
1434     case CB_GETEXTENDEDUI:
1435     case CB_GETDROPPEDSTATE:
1436         ret = callback( HWND_16(hwnd), msg + CB_GETEDITSEL16 - CB_GETEDITSEL, wParam, lParam, result, arg );
1437         break;
1438     case CB_GETEDITSEL:
1439         ret = callback( HWND_16(hwnd), CB_GETEDITSEL16, wParam, lParam, result, arg );
1440         if (wParam) *((PUINT)(wParam)) = LOWORD(*result);
1441         if (lParam) *((PUINT)(lParam)) = HIWORD(*result);  /* FIXME: substract 1? */
1442         break;
1443     case CB_ADDSTRING:
1444     case CB_FINDSTRING:
1445     case CB_FINDSTRINGEXACT:
1446     case CB_INSERTSTRING:
1447     case CB_SELECTSTRING:
1448     case CB_DIR:
1449     case CB_GETLBTEXT:
1450         lParam = MapLS( (LPSTR)lParam );
1451         ret = callback( HWND_16(hwnd), msg + CB_GETEDITSEL16 - CB_GETEDITSEL, wParam, lParam, result, arg );
1452         UnMapLS( lParam );
1453         break;
1454     case LB_GETITEMRECT:
1455     case CB_GETDROPPEDCONTROLRECT:
1456         {
1457             RECT *r32 = (RECT *)lParam;
1458             RECT16 rect;
1459             lParam = MapLS( &rect );
1460             ret = callback( HWND_16(hwnd),
1461                             (msg == LB_GETITEMRECT) ? LB_GETITEMRECT16 : CB_GETDROPPEDCONTROLRECT16,
1462                             wParam, lParam, result, arg );
1463             UnMapLS( lParam );
1464             RECT16to32( &rect, r32 );
1465         }
1466         break;
1467     case WM_PAINTCLIPBOARD:
1468     case WM_SIZECLIPBOARD:
1469         FIXME_(msg)( "message %04x needs translation\n", msg );
1470         break;
1471     /* the following messages should not be sent to 16-bit apps */
1472     case WM_SIZING:
1473     case WM_MOVING:
1474     case WM_CAPTURECHANGED:
1475     case WM_STYLECHANGING:
1476     case WM_STYLECHANGED:
1477         break;
1478     default:
1479         ret = callback( HWND_16(hwnd), msg, wParam, lParam, result, arg );
1480         break;
1481     }
1482     return ret;
1483 }
1484
1485
1486 /***********************************************************************
1487  *              SendMessage  (USER.111)
1488  */
1489 LRESULT WINAPI SendMessage16( HWND16 hwnd16, UINT16 msg, WPARAM16 wparam, LPARAM lparam )
1490 {
1491     LRESULT result;
1492     HWND hwnd = WIN_Handle32( hwnd16 );
1493
1494     if (hwnd != HWND_BROADCAST &&
1495         GetWindowThreadProcessId( hwnd, NULL ) == GetCurrentThreadId())
1496     {
1497         /* call 16-bit window proc directly */
1498         WNDPROC16 winproc;
1499
1500         /* first the WH_CALLWNDPROC hook */
1501         WINPROC_CallProc16To32A( cwp_hook_callback, hwnd16, msg, wparam, lparam, &result, NULL );
1502
1503         if (!(winproc = (WNDPROC16)GetWindowLong16( hwnd16, GWLP_WNDPROC ))) return 0;
1504
1505         TRACE_(message)("(0x%04x) [%04x] wp=%04x lp=%08lx\n", hwnd16, msg, wparam, lparam );
1506         result = CallWindowProc16( winproc, hwnd16, msg, wparam, lparam );
1507         TRACE_(message)("(0x%04x) [%04x] wp=%04x lp=%08lx returned %08lx\n",
1508                         hwnd16, msg, wparam, lparam, result );
1509     }
1510     else  /* map to 32-bit unicode for inter-thread/process message */
1511     {
1512         WINPROC_CallProc16To32A( send_message_callback, hwnd16, msg, wparam, lparam, &result, NULL );
1513     }
1514     return result;
1515 }
1516
1517
1518 /***********************************************************************
1519  *              PostMessage  (USER.110)
1520  */
1521 BOOL16 WINAPI PostMessage16( HWND16 hwnd, UINT16 msg, WPARAM16 wparam, LPARAM lparam )
1522 {
1523     LRESULT unused;
1524     return WINPROC_CallProc16To32A( post_message_callback, hwnd, msg, wparam, lparam, &unused, NULL );
1525 }
1526
1527
1528 /***********************************************************************
1529  *              PostAppMessage (USER.116)
1530  */
1531 BOOL16 WINAPI PostAppMessage16( HTASK16 hTask, UINT16 msg, WPARAM16 wparam, LPARAM lparam )
1532 {
1533     LRESULT unused;
1534     DWORD_PTR tid = HTASK_32( hTask );
1535
1536     if (!tid) return FALSE;
1537     return WINPROC_CallProc16To32A( post_thread_message_callback, 0, msg, wparam, lparam,
1538                                     &unused, (void *)tid );
1539 }
1540
1541
1542 /**********************************************************************
1543  *              CallWindowProc (USER.122)
1544  */
1545 LRESULT WINAPI CallWindowProc16( WNDPROC16 func, HWND16 hwnd, UINT16 msg,
1546                                  WPARAM16 wParam, LPARAM lParam )
1547 {
1548     int index = winproc_to_index( func );
1549     LRESULT result;
1550
1551     if (!func) return 0;
1552
1553     if (index == -1 || index >= MAX_WINPROCS32)
1554         call_window_proc16( hwnd, msg, wParam, lParam, &result, func );
1555     else
1556     {
1557         WNDPROC proc = (WNDPROC)func;
1558         if (thunk_array && thunk_array[index].proc) proc = thunk_array[index].proc;
1559         WINPROC_CallProc16To32A( call_window_proc_callback, hwnd, msg, wParam, lParam, &result, proc );
1560     }
1561     return result;
1562 }
1563
1564
1565 /**********************************************************************
1566  *           __wine_call_wndproc   (USER.1010)
1567  */
1568 LRESULT WINAPI __wine_call_wndproc( HWND16 hwnd, UINT16 msg, WPARAM16 wParam, LPARAM lParam, WNDPROC proc )
1569 {
1570     LRESULT result;
1571     WINPROC_CallProc16To32A( call_window_proc_callback, hwnd, msg, wParam, lParam, &result, proc );
1572     return result;
1573 }
1574
1575
1576 /***********************************************************************
1577  *              InSendMessage  (USER.192)
1578  */
1579 BOOL16 WINAPI InSendMessage16(void)
1580 {
1581     return InSendMessage();
1582 }
1583
1584
1585 /***********************************************************************
1586  *              ReplyMessage  (USER.115)
1587  */
1588 void WINAPI ReplyMessage16( LRESULT result )
1589 {
1590     ReplyMessage( result );
1591 }
1592
1593
1594 /***********************************************************************
1595  *              PeekMessage32 (USER.819)
1596  */
1597 BOOL16 WINAPI PeekMessage32_16( MSG32_16 *msg16, HWND16 hwnd16,
1598                                 UINT16 first, UINT16 last, UINT16 flags,
1599                                 BOOL16 wHaveParamHigh )
1600 {
1601     MSG msg;
1602     LRESULT unused;
1603     HWND hwnd = WIN_Handle32( hwnd16 );
1604
1605     if(USER16_AlertableWait)
1606         MsgWaitForMultipleObjectsEx( 0, NULL, 0, 0, MWMO_ALERTABLE );
1607     if (!PeekMessageA( &msg, hwnd, first, last, flags )) return FALSE;
1608
1609     msg16->msg.time    = msg.time;
1610     msg16->msg.pt.x    = (INT16)msg.pt.x;
1611     msg16->msg.pt.y    = (INT16)msg.pt.y;
1612     if (wHaveParamHigh) msg16->wParamHigh = HIWORD(msg.wParam);
1613     WINPROC_CallProc32ATo16( get_message_callback, msg.hwnd, msg.message, msg.wParam, msg.lParam,
1614                              &unused, &msg16->msg );
1615     return TRUE;
1616 }
1617
1618
1619 /***********************************************************************
1620  *              DefWindowProc (USER.107)
1621  */
1622 LRESULT WINAPI DefWindowProc16( HWND16 hwnd16, UINT16 msg, WPARAM16 wParam, LPARAM lParam )
1623 {
1624     LRESULT result;
1625     HWND hwnd = WIN_Handle32( hwnd16 );
1626
1627     switch(msg)
1628     {
1629     case WM_NCCREATE:
1630         {
1631             CREATESTRUCT16 *cs16 = MapSL(lParam);
1632             CREATESTRUCTA cs32;
1633
1634             cs32.lpCreateParams = ULongToPtr(cs16->lpCreateParams);
1635             cs32.hInstance      = HINSTANCE_32(cs16->hInstance);
1636             cs32.hMenu          = HMENU_32(cs16->hMenu);
1637             cs32.hwndParent     = WIN_Handle32(cs16->hwndParent);
1638             cs32.cy             = cs16->cy;
1639             cs32.cx             = cs16->cx;
1640             cs32.y              = cs16->y;
1641             cs32.x              = cs16->x;
1642             cs32.style          = cs16->style;
1643             cs32.dwExStyle      = cs16->dwExStyle;
1644             cs32.lpszName       = MapSL(cs16->lpszName);
1645             cs32.lpszClass      = MapSL(cs16->lpszClass);
1646             return DefWindowProcA( hwnd, msg, wParam, (LPARAM)&cs32 );
1647         }
1648     case WM_NCCALCSIZE:
1649         {
1650             RECT16 *rect16 = MapSL(lParam);
1651             RECT rect32;
1652
1653             rect32.left    = rect16->left;
1654             rect32.top     = rect16->top;
1655             rect32.right   = rect16->right;
1656             rect32.bottom  = rect16->bottom;
1657
1658             result = DefWindowProcA( hwnd, msg, wParam, (LPARAM)&rect32 );
1659
1660             rect16->left   = rect32.left;
1661             rect16->top    = rect32.top;
1662             rect16->right  = rect32.right;
1663             rect16->bottom = rect32.bottom;
1664             return result;
1665         }
1666     case WM_WINDOWPOSCHANGING:
1667     case WM_WINDOWPOSCHANGED:
1668         {
1669             WINDOWPOS16 *pos16 = MapSL(lParam);
1670             WINDOWPOS pos32;
1671
1672             pos32.hwnd             = WIN_Handle32(pos16->hwnd);
1673             pos32.hwndInsertAfter  = WIN_Handle32(pos16->hwndInsertAfter);
1674             pos32.x                = pos16->x;
1675             pos32.y                = pos16->y;
1676             pos32.cx               = pos16->cx;
1677             pos32.cy               = pos16->cy;
1678             pos32.flags            = pos16->flags;
1679
1680             result = DefWindowProcA( hwnd, msg, wParam, (LPARAM)&pos32 );
1681
1682             pos16->hwnd            = HWND_16(pos32.hwnd);
1683             pos16->hwndInsertAfter = HWND_16(pos32.hwndInsertAfter);
1684             pos16->x               = pos32.x;
1685             pos16->y               = pos32.y;
1686             pos16->cx              = pos32.cx;
1687             pos16->cy              = pos32.cy;
1688             pos16->flags           = pos32.flags;
1689             return result;
1690         }
1691     case WM_GETTEXT:
1692     case WM_SETTEXT:
1693         return DefWindowProcA( hwnd, msg, wParam, (LPARAM)MapSL(lParam) );
1694     default:
1695         return DefWindowProcA( hwnd, msg, wParam, lParam );
1696     }
1697 }
1698
1699
1700 /***********************************************************************
1701  *              DefDlgProc (USER.308)
1702  */
1703 LRESULT WINAPI DefDlgProc16( HWND16 hwnd, UINT16 msg, WPARAM16 wParam, LPARAM lParam )
1704 {
1705     LRESULT result;
1706     WINPROC_CallProc16To32A( defdlg_proc_callback, hwnd, msg, wParam, lParam, &result, 0 );
1707     return result;
1708 }
1709
1710
1711 /***********************************************************************
1712  *              PeekMessage  (USER.109)
1713  */
1714 BOOL16 WINAPI PeekMessage16( MSG16 *msg, HWND16 hwnd,
1715                              UINT16 first, UINT16 last, UINT16 flags )
1716 {
1717     return PeekMessage32_16( (MSG32_16 *)msg, hwnd, first, last, flags, FALSE );
1718 }
1719
1720
1721 /***********************************************************************
1722  *              GetMessage32  (USER.820)
1723  */
1724 BOOL16 WINAPI GetMessage32_16( MSG32_16 *msg16, HWND16 hwnd16, UINT16 first,
1725                                UINT16 last, BOOL16 wHaveParamHigh )
1726 {
1727     MSG msg;
1728     LRESULT unused;
1729     HWND hwnd = WIN_Handle32( hwnd16 );
1730
1731     if(USER16_AlertableWait)
1732         MsgWaitForMultipleObjectsEx( 0, NULL, INFINITE, 0, MWMO_ALERTABLE );
1733     GetMessageA( &msg, hwnd, first, last );
1734     msg16->msg.time    = msg.time;
1735     msg16->msg.pt.x    = (INT16)msg.pt.x;
1736     msg16->msg.pt.y    = (INT16)msg.pt.y;
1737     if (wHaveParamHigh) msg16->wParamHigh = HIWORD(msg.wParam);
1738     WINPROC_CallProc32ATo16( get_message_callback, msg.hwnd, msg.message, msg.wParam, msg.lParam,
1739                              &unused, &msg16->msg );
1740
1741     TRACE( "message %04x, hwnd %p, filter(%04x - %04x)\n",
1742            msg16->msg.message, hwnd, first, last );
1743
1744     return msg16->msg.message != WM_QUIT;
1745 }
1746
1747
1748 /***********************************************************************
1749  *              GetMessage  (USER.108)
1750  */
1751 BOOL16 WINAPI GetMessage16( MSG16 *msg, HWND16 hwnd, UINT16 first, UINT16 last )
1752 {
1753     return GetMessage32_16( (MSG32_16 *)msg, hwnd, first, last, FALSE );
1754 }
1755
1756
1757 /***********************************************************************
1758  *              TranslateMessage32 (USER.821)
1759  */
1760 BOOL16 WINAPI TranslateMessage32_16( const MSG32_16 *msg, BOOL16 wHaveParamHigh )
1761 {
1762     MSG msg32;
1763
1764     msg32.hwnd    = WIN_Handle32( msg->msg.hwnd );
1765     msg32.message = msg->msg.message;
1766     msg32.wParam  = MAKEWPARAM( msg->msg.wParam, wHaveParamHigh ? msg->wParamHigh : 0 );
1767     msg32.lParam  = msg->msg.lParam;
1768     return TranslateMessage( &msg32 );
1769 }
1770
1771
1772 /***********************************************************************
1773  *              TranslateMessage (USER.113)
1774  */
1775 BOOL16 WINAPI TranslateMessage16( const MSG16 *msg )
1776 {
1777     return TranslateMessage32_16( (const MSG32_16 *)msg, FALSE );
1778 }
1779
1780
1781 /***********************************************************************
1782  *              DispatchMessage (USER.114)
1783  */
1784 LONG WINAPI DispatchMessage16( const MSG16* msg )
1785 {
1786     WNDPROC16 winproc;
1787     LRESULT retval;
1788
1789       /* Process timer messages */
1790     if ((msg->message == WM_TIMER) || (msg->message == WM_SYSTIMER))
1791     {
1792         if (msg->lParam)
1793             return CallWindowProc16( (WNDPROC16)msg->lParam, msg->hwnd,
1794                                      msg->message, msg->wParam, GetTickCount() );
1795     }
1796
1797     if (!(winproc = (WNDPROC16)GetWindowLong16( msg->hwnd, GWLP_WNDPROC )))
1798     {
1799         SetLastError( ERROR_INVALID_WINDOW_HANDLE );
1800         return 0;
1801     }
1802     TRACE_(message)("(0x%04x) [%04x] wp=%04x lp=%08lx\n", msg->hwnd, msg->message, msg->wParam, msg->lParam );
1803     retval = CallWindowProc16( winproc, msg->hwnd, msg->message, msg->wParam, msg->lParam );
1804     TRACE_(message)("(0x%04x) [%04x] wp=%04x lp=%08lx returned %08lx\n",
1805                     msg->hwnd, msg->message, msg->wParam, msg->lParam, retval );
1806     return retval;
1807 }
1808
1809
1810 /***********************************************************************
1811  *              DispatchMessage32 (USER.822)
1812  */
1813 LONG WINAPI DispatchMessage32_16( const MSG32_16 *msg16, BOOL16 wHaveParamHigh )
1814 {
1815     if (wHaveParamHigh == FALSE)
1816         return DispatchMessage16( &msg16->msg );
1817     else
1818     {
1819         MSG msg;
1820
1821         msg.hwnd    = WIN_Handle32( msg16->msg.hwnd );
1822         msg.message = msg16->msg.message;
1823         msg.wParam  = MAKEWPARAM( msg16->msg.wParam, msg16->wParamHigh );
1824         msg.lParam  = msg16->msg.lParam;
1825         msg.time    = msg16->msg.time;
1826         msg.pt.x    = msg16->msg.pt.x;
1827         msg.pt.y    = msg16->msg.pt.y;
1828         return DispatchMessageA( &msg );
1829     }
1830 }
1831
1832
1833 /***********************************************************************
1834  *              IsDialogMessage (USER.90)
1835  */
1836 BOOL16 WINAPI IsDialogMessage16( HWND16 hwndDlg, MSG16 *msg16 )
1837 {
1838     MSG msg;
1839     HWND hwndDlg32;
1840
1841     msg.hwnd  = WIN_Handle32(msg16->hwnd);
1842     hwndDlg32 = WIN_Handle32(hwndDlg);
1843
1844     switch(msg16->message)
1845     {
1846     case WM_KEYDOWN:
1847     case WM_CHAR:
1848     case WM_SYSCHAR:
1849         msg.message = msg16->message;
1850         msg.wParam  = msg16->wParam;
1851         msg.lParam  = msg16->lParam;
1852         return IsDialogMessageA( hwndDlg32, &msg );
1853     }
1854
1855     if ((hwndDlg32 != msg.hwnd) && !IsChild( hwndDlg32, msg.hwnd )) return FALSE;
1856     TranslateMessage16( msg16 );
1857     DispatchMessage16( msg16 );
1858     return TRUE;
1859 }
1860
1861
1862 /***********************************************************************
1863  *              MsgWaitForMultipleObjects  (USER.640)
1864  */
1865 DWORD WINAPI MsgWaitForMultipleObjects16( DWORD count, CONST HANDLE *handles,
1866                                           BOOL wait_all, DWORD timeout, DWORD mask )
1867 {
1868     return MsgWaitForMultipleObjectsEx( count, handles, timeout, mask,
1869                                         wait_all ? MWMO_WAITALL : 0 );
1870 }
1871
1872
1873 /**********************************************************************
1874  *              SetDoubleClickTime (USER.20)
1875  */
1876 void WINAPI SetDoubleClickTime16( UINT16 interval )
1877 {
1878     SetDoubleClickTime( interval );
1879 }
1880
1881
1882 /**********************************************************************
1883  *              GetDoubleClickTime (USER.21)
1884  */
1885 UINT16 WINAPI GetDoubleClickTime16(void)
1886 {
1887     return GetDoubleClickTime();
1888 }
1889
1890
1891 /***********************************************************************
1892  *              PostQuitMessage (USER.6)
1893  */
1894 void WINAPI PostQuitMessage16( INT16 exitCode )
1895 {
1896     PostQuitMessage( exitCode );
1897 }
1898
1899
1900 /**********************************************************************
1901  *              GetKeyState (USER.106)
1902  */
1903 INT16 WINAPI GetKeyState16(INT16 vkey)
1904 {
1905     return GetKeyState(vkey);
1906 }
1907
1908
1909 /**********************************************************************
1910  *              GetKeyboardState (USER.222)
1911  */
1912 BOOL WINAPI GetKeyboardState16( LPBYTE state )
1913 {
1914     return GetKeyboardState( state );
1915 }
1916
1917
1918 /**********************************************************************
1919  *              SetKeyboardState (USER.223)
1920  */
1921 BOOL WINAPI SetKeyboardState16( LPBYTE state )
1922 {
1923     return SetKeyboardState( state );
1924 }
1925
1926
1927 /***********************************************************************
1928  *              SetMessageQueue (USER.266)
1929  */
1930 BOOL16 WINAPI SetMessageQueue16( INT16 size )
1931 {
1932     return SetMessageQueue( size );
1933 }
1934
1935
1936 /***********************************************************************
1937  *              UserYield (USER.332)
1938  */
1939 void WINAPI UserYield16(void)
1940 {
1941     MSG msg;
1942     PeekMessageW( &msg, 0, 0, 0, PM_REMOVE | PM_QS_SENDMESSAGE );
1943 }
1944
1945
1946 /***********************************************************************
1947  *              GetQueueStatus (USER.334)
1948  */
1949 DWORD WINAPI GetQueueStatus16( UINT16 flags )
1950 {
1951     return GetQueueStatus( flags );
1952 }
1953
1954
1955 /***********************************************************************
1956  *              GetInputState (USER.335)
1957  */
1958 BOOL16 WINAPI GetInputState16(void)
1959 {
1960     return GetInputState();
1961 }
1962
1963
1964 /**********************************************************************
1965  *           TranslateAccelerator      (USER.178)
1966  */
1967 INT16 WINAPI TranslateAccelerator16( HWND16 hwnd, HACCEL16 hAccel, LPMSG16 msg )
1968 {
1969     MSG msg32;
1970
1971     if (!msg) return 0;
1972     msg32.message = msg->message;
1973     /* msg32.hwnd not used */
1974     msg32.wParam  = msg->wParam;
1975     msg32.lParam  = msg->lParam;
1976     return TranslateAcceleratorW( WIN_Handle32(hwnd), HACCEL_32(hAccel), &msg32 );
1977 }
1978
1979
1980 /**********************************************************************
1981  *              TranslateMDISysAccel (USER.451)
1982  */
1983 BOOL16 WINAPI TranslateMDISysAccel16( HWND16 hwndClient, LPMSG16 msg )
1984 {
1985     if (msg->message == WM_KEYDOWN || msg->message == WM_SYSKEYDOWN)
1986     {
1987         MSG msg32;
1988         msg32.hwnd    = WIN_Handle32(msg->hwnd);
1989         msg32.message = msg->message;
1990         msg32.wParam  = msg->wParam;
1991         msg32.lParam  = msg->lParam;
1992         /* MDICLIENTINFO is still the same for win32 and win16 ... */
1993         return TranslateMDISysAccel( WIN_Handle32(hwndClient), &msg32 );
1994     }
1995     return 0;
1996 }
1997
1998
1999 /***********************************************************************
2000  *           button_proc16
2001  */
2002 static LRESULT button_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2003 {
2004     static const UINT msg16_offset = BM_GETCHECK16 - BM_GETCHECK;
2005
2006     switch (msg)
2007     {
2008     case BM_GETCHECK16:
2009     case BM_SETCHECK16:
2010     case BM_GETSTATE16:
2011     case BM_SETSTATE16:
2012     case BM_SETSTYLE16:
2013         return wow_handlers32.button_proc( hwnd, msg - msg16_offset, wParam, lParam, FALSE );
2014     default:
2015         return wow_handlers32.button_proc( hwnd, msg, wParam, lParam, unicode );
2016     }
2017 }
2018
2019
2020 /***********************************************************************
2021  *           combo_proc16
2022  */
2023 static LRESULT combo_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2024 {
2025     static const UINT msg16_offset = CB_GETEDITSEL16 - CB_GETEDITSEL;
2026
2027     switch (msg)
2028     {
2029     case CB_INSERTSTRING16:
2030     case CB_SELECTSTRING16:
2031     case CB_FINDSTRING16:
2032     case CB_FINDSTRINGEXACT16:
2033         wParam = (INT)(INT16)wParam;
2034         /* fall through */
2035     case CB_ADDSTRING16:
2036     {
2037         DWORD style = GetWindowLongW( hwnd, GWL_STYLE );
2038         if ((style & CBS_HASSTRINGS) || !(style & (CBS_OWNERDRAWFIXED | CBS_OWNERDRAWVARIABLE)))
2039            lParam = (LPARAM)MapSL(lParam);
2040         msg -= msg16_offset;
2041         break;
2042     }
2043     case CB_SETITEMHEIGHT16:
2044     case CB_GETITEMHEIGHT16:
2045     case CB_SETCURSEL16:
2046     case CB_GETLBTEXTLEN16:
2047     case CB_GETITEMDATA16:
2048     case CB_SETITEMDATA16:
2049         wParam = (INT)(INT16)wParam;    /* signed integer */
2050         msg -= msg16_offset;
2051         break;
2052     case CB_GETDROPPEDCONTROLRECT16:
2053         lParam = (LPARAM)MapSL(lParam);
2054         if (lParam)
2055         {
2056             RECT r;
2057             RECT16 *r16 = (RECT16 *)lParam;
2058             wow_handlers32.combo_proc( hwnd, CB_GETDROPPEDCONTROLRECT, wParam, (LPARAM)&r, FALSE );
2059             r16->left   = r.left;
2060             r16->top    = r.top;
2061             r16->right  = r.right;
2062             r16->bottom = r.bottom;
2063         }
2064         return CB_OKAY;
2065     case CB_DIR16:
2066         if (wParam & DDL_DRIVES) wParam |= DDL_EXCLUSIVE;
2067         lParam = (LPARAM)MapSL(lParam);
2068         msg -= msg16_offset;
2069         break;
2070     case CB_GETLBTEXT16:
2071         wParam = (INT)(INT16)wParam;
2072         lParam = (LPARAM)MapSL(lParam);
2073         msg -= msg16_offset;
2074         break;
2075     case CB_GETEDITSEL16:
2076         wParam = lParam = 0;   /* just in case */
2077         msg -= msg16_offset;
2078         break;
2079     case CB_LIMITTEXT16:
2080     case CB_SETEDITSEL16:
2081     case CB_DELETESTRING16:
2082     case CB_RESETCONTENT16:
2083     case CB_GETDROPPEDSTATE16:
2084     case CB_SHOWDROPDOWN16:
2085     case CB_GETCOUNT16:
2086     case CB_GETCURSEL16:
2087     case CB_SETEXTENDEDUI16:
2088     case CB_GETEXTENDEDUI16:
2089         msg -= msg16_offset;
2090         break;
2091     default:
2092         return wow_handlers32.combo_proc( hwnd, msg, wParam, lParam, unicode );
2093     }
2094     return wow_handlers32.combo_proc( hwnd, msg, wParam, lParam, FALSE );
2095 }
2096
2097 /*********************************************************************
2098  * edit_lock_buffer (internal)
2099  *
2100  * A 16 bit application might send an EM_GETHANDLE message and expect a HLOCAL16
2101  * (16 bit SEG:OFF handler). From that moment on we have to keep using this
2102  * 16 bit memory handler, because it is supposed to be valid at all times after
2103  * EM_GETHANDLE.
2104  * We create a HLOCAL16 buffer in edit_get_handle and copy the text from the
2105  * HLOCAL buffer, when needed
2106  *
2107  */
2108
2109 #define GWW_HANDLE16 sizeof(void*)
2110
2111 static void edit_lock_buffer( HWND hwnd )
2112 {
2113     STACK16FRAME* stack16 = MapSL(PtrToUlong(NtCurrentTeb()->WOW32Reserved));
2114     HLOCAL16 hloc16 = GetWindowWord( hwnd, GWW_HANDLE16 );
2115     HANDLE16 oldDS;
2116     HLOCAL hloc32;
2117     UINT size;
2118
2119     if (!hloc16) return;
2120     if (!(hloc32 = (HLOCAL)wow_handlers32.edit_proc( hwnd, EM_GETHANDLE, 0, 0, FALSE ))) return;
2121
2122     oldDS = stack16->ds;
2123     stack16->ds = GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
2124     size = LocalSize16(hloc16);
2125     if (LocalReAlloc( hloc32, size, LMEM_MOVEABLE ))
2126     {
2127         char *text = MapSL( LocalLock16( hloc16 ));
2128         char *dest = LocalLock( hloc32 );
2129         memcpy( dest, text, size );
2130         LocalUnlock( hloc32 );
2131         LocalUnlock16( hloc16 );
2132     }
2133     stack16->ds = oldDS;
2134
2135 }
2136
2137 static void edit_unlock_buffer( HWND hwnd )
2138 {
2139     STACK16FRAME* stack16 = MapSL(PtrToUlong(NtCurrentTeb()->WOW32Reserved));
2140     HLOCAL16 hloc16 = GetWindowWord( hwnd, GWW_HANDLE16 );
2141     HANDLE16 oldDS;
2142     HLOCAL hloc32;
2143     UINT size;
2144
2145     if (!hloc16) return;
2146     if (!(hloc32 = (HLOCAL)wow_handlers32.edit_proc( hwnd, EM_GETHANDLE, 0, 0, FALSE ))) return;
2147     size = LocalSize( hloc32 );
2148
2149     oldDS = stack16->ds;
2150     stack16->ds = GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
2151     if (LocalReAlloc16( hloc16, size, LMEM_MOVEABLE ))
2152     {
2153         char *text = LocalLock( hloc32 );
2154         char *dest = MapSL( LocalLock16( hloc16 ));
2155         memcpy( dest, text, size );
2156         LocalUnlock( hloc32 );
2157         LocalUnlock16( hloc16 );
2158     }
2159     stack16->ds = oldDS;
2160 }
2161
2162 static HLOCAL16 edit_get_handle( HWND hwnd )
2163 {
2164     CHAR *textA;
2165     UINT alloc_size;
2166     HLOCAL hloc;
2167     STACK16FRAME* stack16;
2168     HANDLE16 oldDS;
2169     HLOCAL16 hloc16 = GetWindowWord( hwnd, GWW_HANDLE16 );
2170
2171     if (hloc16) return hloc16;
2172
2173     if (!(hloc = (HLOCAL)wow_handlers32.edit_proc( hwnd, EM_GETHANDLE, 0, 0, FALSE ))) return 0;
2174     alloc_size = LocalSize( hloc );
2175
2176     stack16 = MapSL(PtrToUlong(NtCurrentTeb()->WOW32Reserved));
2177     oldDS = stack16->ds;
2178     stack16->ds = GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
2179
2180     if (!LocalHeapSize16())
2181     {
2182         if (!LocalInit16(stack16->ds, 0, GlobalSize16(stack16->ds)))
2183         {
2184             ERR("could not initialize local heap\n");
2185             goto done;
2186         }
2187     }
2188
2189     if (!(hloc16 = LocalAlloc16(LMEM_MOVEABLE | LMEM_ZEROINIT, alloc_size)))
2190     {
2191         ERR("could not allocate new 16 bit buffer\n");
2192         goto done;
2193     }
2194
2195     if (!(textA = MapSL(LocalLock16( hloc16))))
2196     {
2197         ERR("could not lock new 16 bit buffer\n");
2198         LocalFree16(hloc16);
2199         hloc16 = 0;
2200         goto done;
2201     }
2202     memcpy( textA, LocalLock( hloc ), alloc_size );
2203     LocalUnlock( hloc );
2204     LocalUnlock16( hloc16 );
2205     SetWindowWord( hwnd, GWW_HANDLE16, hloc16 );
2206
2207 done:
2208     stack16->ds = oldDS;
2209     return hloc16;
2210 }
2211
2212 static void edit_set_handle( HWND hwnd, HLOCAL16 hloc16 )
2213 {
2214     STACK16FRAME* stack16 = MapSL(PtrToUlong(NtCurrentTeb()->WOW32Reserved));
2215     HINSTANCE16 hInstance = GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
2216     HANDLE16 oldDS = stack16->ds;
2217     HLOCAL hloc32;
2218     INT count;
2219     CHAR *text;
2220
2221     if (!(GetWindowLongW( hwnd, GWL_STYLE ) & ES_MULTILINE)) return;
2222     if (!hloc16) return;
2223
2224     stack16->ds = hInstance;
2225     count = LocalSize16(hloc16);
2226     text = MapSL(LocalLock16(hloc16));
2227     if ((hloc32 = LocalAlloc(LMEM_MOVEABLE, count)))
2228     {
2229         memcpy( LocalLock(hloc32), text, count );
2230         LocalUnlock(hloc32);
2231         LocalUnlock16(hloc16);
2232         SetWindowWord( hwnd, GWW_HANDLE16, hloc16 );
2233     }
2234     stack16->ds = oldDS;
2235
2236     if (hloc32) wow_handlers32.edit_proc( hwnd, EM_SETHANDLE, (WPARAM)hloc32, 0, FALSE );
2237 }
2238
2239 static void edit_destroy_handle( HWND hwnd )
2240 {
2241     HLOCAL16 hloc16 = GetWindowWord( hwnd, GWW_HANDLE16 );
2242     if (hloc16)
2243     {
2244         STACK16FRAME* stack16 = MapSL(PtrToUlong(NtCurrentTeb()->WOW32Reserved));
2245         HANDLE16 oldDS = stack16->ds;
2246
2247         stack16->ds = GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
2248         while (LocalUnlock16(hloc16)) ;
2249         LocalFree16(hloc16);
2250         stack16->ds = oldDS;
2251         SetWindowWord( hwnd, GWW_HANDLE16, 0 );
2252     }
2253 }
2254
2255 /*********************************************************************
2256  *      edit_proc16
2257  */
2258 static LRESULT edit_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2259 {
2260     static const UINT msg16_offset = EM_GETSEL16 - EM_GETSEL;
2261     LRESULT result = 0;
2262
2263     edit_lock_buffer( hwnd );
2264     switch (msg)
2265     {
2266     case EM_SCROLL16:
2267     case EM_SCROLLCARET16:
2268     case EM_GETMODIFY16:
2269     case EM_SETMODIFY16:
2270     case EM_GETLINECOUNT16:
2271     case EM_GETTHUMB16:
2272     case EM_LINELENGTH16:
2273     case EM_LIMITTEXT16:
2274     case EM_CANUNDO16:
2275     case EM_UNDO16:
2276     case EM_FMTLINES16:
2277     case EM_LINEFROMCHAR16:
2278     case EM_SETPASSWORDCHAR16:
2279     case EM_EMPTYUNDOBUFFER16:
2280     case EM_SETREADONLY16:
2281     case EM_GETPASSWORDCHAR16:
2282         /* these messages missing from specs */
2283     case WM_USER+15:
2284     case WM_USER+16:
2285     case WM_USER+19:
2286     case WM_USER+26:
2287         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, lParam, FALSE );
2288         break;
2289     case EM_GETSEL16:
2290         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, 0, 0, FALSE );
2291         break;
2292     case EM_REPLACESEL16:
2293     case EM_GETLINE16:
2294         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, (LPARAM)MapSL(lParam), FALSE );
2295         break;
2296     case EM_LINESCROLL16:
2297         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, (INT)(SHORT)HIWORD(lParam),
2298                                            (INT)(SHORT)LOWORD(lParam), FALSE );
2299         break;
2300     case EM_LINEINDEX16:
2301         if ((INT16)wParam == -1) wParam = -1;
2302         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, lParam, FALSE );
2303         break;
2304     case EM_SETSEL16:
2305         if ((short)LOWORD(lParam) == -1)
2306         {
2307             wParam = -1;
2308             lParam = 0;
2309         }
2310         else
2311         {
2312             wParam = LOWORD(lParam);
2313             lParam = HIWORD(lParam);
2314         }
2315         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, lParam, FALSE );
2316         break;
2317     case EM_GETRECT16:
2318         if (lParam)
2319         {
2320             RECT rect;
2321             RECT16 *r16 = MapSL(lParam);
2322             wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, (LPARAM)&rect, FALSE );
2323             r16->left   = rect.left;
2324             r16->top    = rect.top;
2325             r16->right  = rect.right;
2326             r16->bottom = rect.bottom;
2327         }
2328         break;
2329     case EM_SETRECT16:
2330     case EM_SETRECTNP16:
2331         if (lParam)
2332         {
2333             RECT rect;
2334             RECT16 *r16 = MapSL(lParam);
2335             rect.left   = r16->left;
2336             rect.top    = r16->top;
2337             rect.right  = r16->right;
2338             rect.bottom = r16->bottom;
2339             wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, (LPARAM)&rect, FALSE );
2340         }
2341         break;
2342     case EM_SETHANDLE16:
2343         edit_set_handle( hwnd, (HLOCAL16)wParam );
2344         break;
2345     case EM_GETHANDLE16:
2346         result = edit_get_handle( hwnd );
2347         break;
2348     case EM_SETTABSTOPS16:
2349     {
2350         INT16 *tabs16 = MapSL(lParam);
2351         INT i, count = wParam, *tabs = NULL;
2352         if (count > 0)
2353         {
2354             if (!(tabs = HeapAlloc( GetProcessHeap(), 0, count * sizeof(*tabs) ))) return 0;
2355             for (i = 0; i < count; i++) tabs[i] = tabs16[i];
2356         }
2357         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, count, (LPARAM)tabs, FALSE );
2358         HeapFree( GetProcessHeap(), 0, tabs );
2359         break;
2360     }
2361     case EM_GETFIRSTVISIBLELINE16:
2362         if (!(GetWindowLongW( hwnd, GWL_STYLE ) & ES_MULTILINE)) break;
2363         result = wow_handlers32.edit_proc( hwnd, msg - msg16_offset, wParam, lParam, FALSE );
2364         break;
2365     case EM_SETWORDBREAKPROC16:
2366     {
2367         struct word_break_thunk *thunk = add_word_break_thunk( (EDITWORDBREAKPROC16)lParam );
2368         result = wow_handlers32.edit_proc( hwnd, EM_SETWORDBREAKPROC, wParam, (LPARAM)thunk, FALSE );
2369         break;
2370     }
2371     case EM_GETWORDBREAKPROC16:
2372         result = wow_handlers32.edit_proc( hwnd, EM_GETWORDBREAKPROC, wParam, lParam, FALSE );
2373         result = (LRESULT)get_word_break_thunk( (EDITWORDBREAKPROCA)result );
2374         break;
2375     case WM_NCDESTROY:
2376         edit_destroy_handle( hwnd );
2377         return wow_handlers32.edit_proc( hwnd, msg, wParam, lParam, unicode );  /* no unlock on destroy */
2378     case WM_HSCROLL:
2379     case WM_VSCROLL:
2380         if (LOWORD(wParam) == EM_GETTHUMB16 || LOWORD(wParam) == EM_LINESCROLL16) wParam -= msg16_offset;
2381         result = wow_handlers32.edit_proc( hwnd, msg, wParam, lParam, unicode );
2382         break;
2383     default:
2384         result = wow_handlers32.edit_proc( hwnd, msg, wParam, lParam, unicode );
2385         break;
2386     }
2387     edit_unlock_buffer( hwnd );
2388     return result;
2389 }
2390
2391
2392 /***********************************************************************
2393  *           listbox_proc16
2394  */
2395 static LRESULT listbox_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2396 {
2397     static const UINT msg16_offset = LB_ADDSTRING16 - LB_ADDSTRING;
2398     LRESULT ret;
2399
2400     switch (msg)
2401     {
2402     case WM_SIZE:
2403         if (is_old_app( hwnd ))
2404         {
2405             DWORD style = GetWindowLongW( hwnd, GWL_STYLE );
2406             int width, height, remaining, item_height;
2407             RECT rect;
2408
2409             /* give a margin for error to old 16 bits programs - if we need
2410                less than the height of the nonclient area, round to the
2411                *next* number of items */
2412
2413             if (!(style & LBS_NOINTEGRALHEIGHT) && !(style & LBS_OWNERDRAWVARIABLE))
2414             {
2415                 GetClientRect( hwnd, &rect );
2416                 width  = rect.right - rect.left;
2417                 height = rect.bottom - rect.top;
2418                 item_height = wow_handlers32.listbox_proc( hwnd, LB_GETITEMHEIGHT, 0, 0, FALSE );
2419                 remaining = item_height ? (height % item_height) : 0;
2420                 if ((height > item_height) && remaining)
2421                 {
2422                     GetWindowRect( hwnd, &rect );
2423                     if ((item_height - remaining) <= rect.bottom - rect.top - height)
2424                         remaining = remaining - item_height;
2425                     TRACE( "[%p]: changing height %d -> %d\n", hwnd, height, height - remaining );
2426                     SetWindowPos( hwnd, 0, 0, 0, rect.right - rect.left,
2427                                   rect.bottom - rect.top - remaining,
2428                                   SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOMOVE );
2429                     return 0;
2430                 }
2431             }
2432         }
2433         return wow_handlers32.listbox_proc( hwnd, msg, wParam, lParam, unicode );
2434
2435     case LB_RESETCONTENT16:
2436     case LB_DELETESTRING16:
2437     case LB_GETITEMDATA16:
2438     case LB_SETITEMDATA16:
2439     case LB_GETCOUNT16:
2440     case LB_GETTEXTLEN16:
2441     case LB_GETCURSEL16:
2442     case LB_GETTOPINDEX16:
2443     case LB_GETITEMHEIGHT16:
2444     case LB_SETCARETINDEX16:
2445     case LB_GETCARETINDEX16:
2446     case LB_SETTOPINDEX16:
2447     case LB_SETCOLUMNWIDTH16:
2448     case LB_GETSELCOUNT16:
2449     case LB_SELITEMRANGE16:
2450     case LB_SELITEMRANGEEX16:
2451     case LB_GETHORIZONTALEXTENT16:
2452     case LB_SETHORIZONTALEXTENT16:
2453     case LB_GETANCHORINDEX16:
2454     case LB_CARETON16:
2455     case LB_CARETOFF16:
2456         msg -= msg16_offset;
2457         break;
2458     case LB_GETSEL16:
2459     case LB_SETSEL16:
2460     case LB_SETCURSEL16:
2461     case LB_SETANCHORINDEX16:
2462         wParam = (INT)(INT16)wParam;
2463         msg -= msg16_offset;
2464         break;
2465     case LB_INSERTSTRING16:
2466     case LB_FINDSTRING16:
2467     case LB_FINDSTRINGEXACT16:
2468     case LB_SELECTSTRING16:
2469         wParam = (INT)(INT16)wParam;
2470         /* fall through */
2471     case LB_ADDSTRING16:
2472     case LB_ADDFILE16:
2473     {
2474         DWORD style = GetWindowLongW( hwnd, GWL_STYLE );
2475         if ((style & LBS_HASSTRINGS) || !(style & (LBS_OWNERDRAWFIXED | LBS_OWNERDRAWVARIABLE)))
2476             lParam = (LPARAM)MapSL(lParam);
2477         msg -= msg16_offset;
2478         break;
2479     }
2480     case LB_GETTEXT16:
2481         lParam = (LPARAM)MapSL(lParam);
2482         msg -= msg16_offset;
2483         break;
2484     case LB_SETITEMHEIGHT16:
2485         lParam = LOWORD(lParam);
2486         msg -= msg16_offset;
2487         break;
2488     case LB_GETITEMRECT16:
2489         {
2490             RECT rect;
2491             RECT16 *r16 = MapSL(lParam);
2492             ret = wow_handlers32.listbox_proc( hwnd, LB_GETITEMRECT, (INT16)wParam, (LPARAM)&rect, FALSE );
2493             r16->left   = rect.left;
2494             r16->top    = rect.top;
2495             r16->right  = rect.right;
2496             r16->bottom = rect.bottom;
2497             return ret;
2498         }
2499     case LB_GETSELITEMS16:
2500     {
2501         INT16 *array16 = MapSL( lParam );
2502         INT i, count = (INT16)wParam, *array;
2503         if (!(array = HeapAlloc( GetProcessHeap(), 0, wParam * sizeof(*array) ))) return LB_ERRSPACE;
2504         ret = wow_handlers32.listbox_proc( hwnd, LB_GETSELITEMS, count, (LPARAM)array, FALSE );
2505         for (i = 0; i < ret; i++) array16[i] = array[i];
2506         HeapFree( GetProcessHeap(), 0, array );
2507         return ret;
2508     }
2509     case LB_DIR16:
2510         /* according to Win16 docs, DDL_DRIVES should make DDL_EXCLUSIVE
2511          * be set automatically (this is different in Win32) */
2512         if (wParam & DDL_DRIVES) wParam |= DDL_EXCLUSIVE;
2513         lParam = (LPARAM)MapSL(lParam);
2514         msg -= msg16_offset;
2515         break;
2516     case LB_SETTABSTOPS16:
2517     {
2518         INT i, count, *tabs = NULL;
2519         INT16 *tabs16 = MapSL( lParam );
2520
2521         if ((count = (INT16)wParam) > 0)
2522         {
2523             if (!(tabs = HeapAlloc( GetProcessHeap(), 0, wParam * sizeof(*tabs) ))) return LB_ERRSPACE;
2524             for (i = 0; i < count; i++) tabs[i] = tabs16[i] << 1; /* FIXME */
2525         }
2526         ret = wow_handlers32.listbox_proc( hwnd, LB_SETTABSTOPS, count, (LPARAM)tabs, FALSE );
2527         HeapFree( GetProcessHeap(), 0, tabs );
2528         return ret;
2529     }
2530     default:
2531         return wow_handlers32.listbox_proc( hwnd, msg, wParam, lParam, unicode );
2532     }
2533     return wow_handlers32.listbox_proc( hwnd, msg, wParam, lParam, FALSE );
2534 }
2535
2536
2537 /***********************************************************************
2538  *           mdiclient_proc16
2539  */
2540 static LRESULT mdiclient_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2541 {
2542     if (msg == WM_CREATE)
2543     {
2544         LPCREATESTRUCTA cs = (LPCREATESTRUCTA)lParam;
2545         HINSTANCE instance = (HINSTANCE)GetWindowLongPtrW( hwnd, GWLP_HINSTANCE );
2546         BOOL is_win32 = !instance || ((ULONG_PTR)instance >> 16);
2547
2548         /* Translation layer doesn't know what's in the cs->lpCreateParams
2549          * so we have to keep track of what environment we're in. */
2550         if (!is_win32)
2551         {
2552             void *orig = cs->lpCreateParams;
2553             LRESULT ret;
2554             CLIENTCREATESTRUCT ccs;
2555             CLIENTCREATESTRUCT16 *ccs16 = MapSL( PtrToUlong( orig ));
2556
2557             ccs.hWindowMenu  = HMENU_32(ccs16->hWindowMenu);
2558             ccs.idFirstChild = ccs16->idFirstChild;
2559             cs->lpCreateParams = &ccs;
2560             ret = wow_handlers32.mdiclient_proc( hwnd, msg, wParam, lParam, unicode );
2561             cs->lpCreateParams = orig;
2562             return ret;
2563         }
2564     }
2565     return wow_handlers32.mdiclient_proc( hwnd, msg, wParam, lParam, unicode );
2566 }
2567
2568
2569 /***********************************************************************
2570  *           scrollbar_proc16
2571  */
2572 static LRESULT scrollbar_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2573 {
2574     static const UINT msg16_offset = SBM_SETPOS16 - SBM_SETPOS;
2575
2576     switch (msg)
2577     {
2578     case SBM_SETPOS16:
2579     case SBM_GETPOS16:
2580     case SBM_ENABLE_ARROWS16:
2581         msg -= msg16_offset;
2582         break;
2583     case SBM_SETRANGE16:
2584         msg = wParam ? SBM_SETRANGEREDRAW : SBM_SETRANGE;
2585         wParam = LOWORD(lParam);
2586         lParam = HIWORD(lParam);
2587         break;
2588     case SBM_GETRANGE16:
2589     {
2590         INT min, max;
2591         wow_handlers32.scrollbar_proc( hwnd, SBM_GETRANGE, (WPARAM)&min, (LPARAM)&max, FALSE );
2592         return MAKELRESULT(min, max);
2593     }
2594     default:
2595         return wow_handlers32.scrollbar_proc( hwnd, msg, wParam, lParam, unicode );
2596     }
2597     return wow_handlers32.scrollbar_proc( hwnd, msg, wParam, lParam, FALSE );
2598 }
2599
2600
2601 /***********************************************************************
2602  *           static_proc16
2603  */
2604 static LRESULT static_proc16( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode )
2605 {
2606     switch (msg)
2607     {
2608     case WM_NCCREATE:
2609     {
2610         CREATESTRUCTA *cs = (CREATESTRUCTA *)lParam;
2611         LRESULT ret = wow_handlers32.static_proc( hwnd, msg, wParam, lParam, unicode );
2612
2613         if (!ret) return 0;
2614         if (((ULONG_PTR)cs->hInstance >> 16)) return ret;  /* 32-bit instance, nothing to do */
2615         switch (cs->style & SS_TYPEMASK)
2616         {
2617         case SS_ICON:
2618             {
2619                 HICON16 icon = LoadIcon16( HINSTANCE_16(cs->hInstance), cs->lpszName );
2620                 if (!icon) icon = LoadCursor16( HINSTANCE_16(cs->hInstance), cs->lpszName );
2621                 if (icon) wow_handlers32.static_proc( hwnd, STM_SETIMAGE, IMAGE_ICON,
2622                                                       (LPARAM)HICON_32(icon), FALSE );
2623                 break;
2624             }
2625         case SS_BITMAP:
2626             {
2627                 HBITMAP16 bitmap = LoadBitmap16( HINSTANCE_16(cs->hInstance), cs->lpszName );
2628                 if (bitmap) wow_handlers32.static_proc( hwnd, STM_SETIMAGE, IMAGE_BITMAP,
2629                                                         (LPARAM)HBITMAP_32(bitmap), FALSE );
2630                 break;
2631             }
2632         }
2633         return ret;
2634     }
2635     case STM_SETICON16:
2636         wParam = (WPARAM)HICON_32( (HICON16)wParam );
2637         return wow_handlers32.static_proc( hwnd, STM_SETICON, wParam, lParam, FALSE );
2638     case STM_GETICON16:
2639         return HICON_16( wow_handlers32.static_proc( hwnd, STM_GETICON, wParam, lParam, FALSE ));
2640     default:
2641         return wow_handlers32.static_proc( hwnd, msg, wParam, lParam, unicode );
2642     }
2643 }
2644
2645
2646 /***********************************************************************
2647  *           wait_message16
2648  */
2649 static DWORD wait_message16( DWORD count, CONST HANDLE *handles, DWORD timeout, DWORD mask, DWORD flags )
2650 {
2651     DWORD lock, ret;
2652
2653     ReleaseThunkLock( &lock );
2654     ret = wow_handlers32.wait_message( count, handles, timeout, mask, flags );
2655     RestoreThunkLock( lock );
2656     return ret;
2657 }
2658
2659
2660 /***********************************************************************
2661  *           create_window16
2662  */
2663 HWND create_window16( CREATESTRUCTW *cs, LPCWSTR className, HINSTANCE instance, BOOL unicode )
2664 {
2665     /* map to module handle */
2666     if (instance && !((ULONG_PTR)instance >> 16))
2667         instance = HINSTANCE_32( GetExePtr( HINSTANCE_16(instance) ));
2668
2669     return wow_handlers32.create_window( cs, className, instance, unicode );
2670 }
2671
2672
2673 /***********************************************************************
2674  * cursor/icon handles
2675  */
2676
2677 static HICON alloc_icon_handle( unsigned int size )
2678 {
2679     HGLOBAL16 handle = GlobalAlloc16( GMEM_MOVEABLE, size );
2680     FarSetOwner16( handle, 0 );
2681     return HICON_32( handle );
2682 }
2683
2684 static struct tagCURSORICONINFO *get_icon_ptr( HICON handle )
2685 {
2686     return GlobalLock16( HICON_16(handle) );
2687 }
2688
2689 static void release_icon_ptr( HICON handle, struct tagCURSORICONINFO *ptr )
2690 {
2691     GlobalUnlock16( HICON_16(handle) );
2692 }
2693
2694 static int free_icon_handle( HICON handle )
2695 {
2696     return GlobalFree16( HICON_16(handle) );
2697 }
2698
2699
2700 void register_wow_handlers(void)
2701 {
2702     static const struct wow_handlers16 handlers16 =
2703     {
2704         button_proc16,
2705         combo_proc16,
2706         edit_proc16,
2707         listbox_proc16,
2708         mdiclient_proc16,
2709         scrollbar_proc16,
2710         static_proc16,
2711         wait_message16,
2712         create_window16,
2713         call_window_proc_Ato16,
2714         call_dialog_proc_Ato16,
2715         alloc_icon_handle,
2716         get_icon_ptr,
2717         release_icon_ptr,
2718         free_icon_handle
2719     };
2720
2721     UserRegisterWowHandlers( &handlers16, &wow_handlers32 );
2722 }