2 * 16-bit and 32-bit mode stack frame layout
4 * Copyright 1995, 1998 Alexandre Julliard
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.
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.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21 #ifndef __WINE_STACKFRAME_H
22 #define __WINE_STACKFRAME_H
28 #include "wine/winbase16.h"
32 /* 32-bit stack layout after CallTo16() */
33 typedef struct _STACK32FRAME
35 SEGPTR frame16
; /* 00 16-bit frame from last CallFrom16() */
36 DWORD restore_addr
; /* 04 return address for restoring code selector */
37 DWORD codeselector
; /* 08 code selector to restore */
38 DWORD edi
; /* 0c saved registers */
43 DWORD ebp
; /* 20 saved 32-bit frame pointer */
44 DWORD retaddr
; /* 24 return address */
45 DWORD target
; /* 28 target address / CONTEXT86 pointer */
46 DWORD nb_args
; /* 2c number of 16-bit argument bytes */
49 /* 16-bit stack layout after CallFrom16() */
50 typedef struct _STACK16FRAME
52 STACK32FRAME
*frame32
; /* 00 32-bit frame from last CallTo16() */
53 DWORD edx
; /* 04 saved registers */
60 DWORD callfrom_ip
; /* 18 callfrom tail IP */
61 DWORD module_cs
; /* 1c module code segment */
62 DWORD relay
; /* 20 relay function address */
63 WORD entry_ip
; /* 22 entry point IP */
64 DWORD entry_point
; /* 26 API entry point to call, reused as mutex count */
65 WORD bp
; /* 2a 16-bit stack frame chain */
66 WORD ip
; /* 2c return address */
72 #define THREAD_STACK16(teb) ((STACK16FRAME*)MapSL((teb)->cur_stack))
73 #define CURRENT_STACK16 (THREAD_STACK16(NtCurrentTeb()))
74 #define CURRENT_DS (CURRENT_STACK16->ds)
76 /* varargs lists on the 16-bit stack */
78 typedef void *VA_LIST16
;
80 #define __VA_ROUNDED16(type) \
81 ((sizeof(type) + sizeof(WORD) - 1) / sizeof(WORD) * sizeof(WORD))
82 #define VA_START16(list) ((list) = (VA_LIST16)(CURRENT_STACK16 + 1))
83 #define VA_ARG16(list,type) \
84 (((list) = (VA_LIST16)((char *)(list) + __VA_ROUNDED16(type))), \
85 *((type *)(void *)((char *)(list) - __VA_ROUNDED16(type))))
86 #define VA_END16(list) ((void)0)
89 /* Push bytes on the 16-bit stack of a thread;
90 * return a segptr to the first pushed byte
92 static inline SEGPTR WINE_UNUSED
stack16_push( int size
)
94 TEB
*teb
= NtCurrentTeb();
95 STACK16FRAME
*frame
= THREAD_STACK16(teb
);
96 memmove( (char*)frame
- size
, frame
, sizeof(*frame
) );
97 teb
->cur_stack
-= size
;
98 return (SEGPTR
)(teb
->cur_stack
+ sizeof(*frame
));
101 /* Pop bytes from the 16-bit stack of a thread */
102 static inline void WINE_UNUSED
stack16_pop( int size
)
104 TEB
*teb
= NtCurrentTeb();
105 STACK16FRAME
*frame
= THREAD_STACK16(teb
);
106 memmove( (char*)frame
+ size
, frame
, sizeof(*frame
) );
107 teb
->cur_stack
+= size
;
110 /* Push a DWORD on the 32-bit stack */
111 static inline void WINE_UNUSED
stack32_push( CONTEXT86
*context
, DWORD val
)
113 context
->Esp
-= sizeof(DWORD
);
114 *(DWORD
*)context
->Esp
= val
;
117 /* Pop a DWORD from the 32-bit stack */
118 static inline DWORD WINE_UNUSED
stack32_pop( CONTEXT86
*context
)
120 DWORD ret
= *(DWORD
*)context
->Esp
;
121 context
->Esp
+= sizeof(DWORD
);
125 #endif /* __WINE_STACKFRAME_H */