Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-cw
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-cw
Commits
4cb00196
Commit
4cb00196
authored
Oct 28, 2003
by
Alexandre Julliard
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Switch back to the main process stack before calling exception
handlers.
parent
8f1b861e
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
321 additions
and
190 deletions
+321
-190
signal_i386.c
dlls/ntdll/signal_i386.c
+296
-164
relay.c
tools/winebuild/relay.c
+25
-26
No files found.
dlls/ntdll/signal_i386.c
View file @
4cb00196
...
@@ -416,6 +416,7 @@ typedef int (*wine_signal_handler)(unsigned int sig);
...
@@ -416,6 +416,7 @@ typedef int (*wine_signal_handler)(unsigned int sig);
static
wine_signal_handler
handlers
[
256
];
static
wine_signal_handler
handlers
[
256
];
extern
void
WINAPI
EXC_RtlRaiseException
(
PEXCEPTION_RECORD
,
PCONTEXT
);
extern
void
WINAPI
EXC_RtlRaiseException
(
PEXCEPTION_RECORD
,
PCONTEXT
);
extern
void
DECLSPEC_NORETURN
__wine_call_from_32_restore_regs
(
CONTEXT
context
);
/* Global variable to save the thread %fs register while in 16-bit code (FIXME) */
/* Global variable to save the thread %fs register while in 16-bit code (FIXME) */
static
unsigned
int
signal_fs
;
static
unsigned
int
signal_fs
;
...
@@ -512,6 +513,8 @@ static void restore_vm86_context( const CONTEXT *context, struct vm86plus_struct
...
@@ -512,6 +513,8 @@ static void restore_vm86_context( const CONTEXT *context, struct vm86plus_struct
#endif
/* __HAVE_VM86 */
#endif
/* __HAVE_VM86 */
typedef
void
(
WINAPI
*
raise_func
)(
EXCEPTION_RECORD
*
rec
,
CONTEXT
*
context
);
/***********************************************************************
/***********************************************************************
* save_context
* save_context
*
*
...
@@ -737,17 +740,10 @@ inline static void save_fpu( CONTEXT *context, const SIGCONTEXT *sigcontext )
...
@@ -737,17 +740,10 @@ inline static void save_fpu( CONTEXT *context, const SIGCONTEXT *sigcontext )
*
*
* Restore the FPU context to a sigcontext.
* Restore the FPU context to a sigcontext.
*/
*/
inline
static
void
restore_fpu
(
CONTEXT
*
context
,
const
SIGCONTEXT
*
sigcontext
)
inline
static
void
restore_fpu
(
CONTEXT
*
context
)
{
{
/* reset the current interrupt status */
/* reset the current interrupt status */
context
->
FloatSave
.
StatusWord
&=
context
->
FloatSave
.
ControlWord
|
0xffffff80
;
context
->
FloatSave
.
StatusWord
&=
context
->
FloatSave
.
ControlWord
|
0xffffff80
;
#ifdef FPU_sig
if
(
FPU_sig
(
sigcontext
))
{
*
FPU_sig
(
sigcontext
)
=
context
->
FloatSave
;
return
;
}
#endif
/* FPU_sig */
#ifdef __GNUC__
#ifdef __GNUC__
/* avoid nested exceptions */
/* avoid nested exceptions */
__asm__
__volatile__
(
"frstor %0; fwait"
:
:
"m"
(
context
->
FloatSave
)
);
__asm__
__volatile__
(
"frstor %0; fwait"
:
:
"m"
(
context
->
FloatSave
)
);
...
@@ -755,6 +751,158 @@ inline static void restore_fpu( CONTEXT *context, const SIGCONTEXT *sigcontext )
...
@@ -755,6 +751,158 @@ inline static void restore_fpu( CONTEXT *context, const SIGCONTEXT *sigcontext )
}
}
/***********************************************************************
* setup_exception
*
* Setup a proper stack frame for the raise function, and modify the
* sigcontext so that the return from the signal handler will call
* the raise function.
*/
static
EXCEPTION_RECORD
*
setup_exception
(
SIGCONTEXT
*
sigcontext
,
raise_func
func
)
{
struct
stack_layout
{
void
*
ret_addr
;
/* return address from raise_func */
EXCEPTION_RECORD
*
rec_ptr
;
/* first arg for raise_func */
CONTEXT
*
context_ptr
;
/* second arg for raise_func */
void
*
dummy
;
/* dummy ret addr for __wine_call_from_32_restore_regs */
CONTEXT
context
;
EXCEPTION_RECORD
rec
;
DWORD
ebp
;
DWORD
eip
;
}
*
stack
;
WORD
fs
,
gs
;
/* get %fs and %gs at time of the fault */
#ifdef FS_sig
fs
=
LOWORD
(
FS_sig
(
sigcontext
));
#else
fs
=
wine_get_fs
();
#endif
#ifdef GS_sig
gs
=
LOWORD
(
GS_sig
(
sigcontext
));
#else
gs
=
wine_get_gs
();
#endif
/* now restore a proper %fs for the fault handler */
if
(
!
IS_SELECTOR_SYSTEM
(
CS_sig
(
sigcontext
))
||
!
IS_SELECTOR_SYSTEM
(
SS_sig
(
sigcontext
)))
/* 16-bit mode */
{
/*
* Win16 or DOS protected mode. Note that during switch
* from 16-bit mode to linear mode, CS may be set to system
* segment before FS is restored. Fortunately, in this case
* SS is still non-system segment. This is why both CS and SS
* are checked.
*/
wine_set_fs
(
signal_fs
);
wine_set_gs
(
NtCurrentTeb
()
->
gs_sel
);
stack
=
(
struct
stack_layout
*
)
NtCurrentTeb
()
->
cur_stack
;
}
else
/* 32-bit mode */
{
stack
=
(
struct
stack_layout
*
)
ESP_sig
(
sigcontext
);
#ifdef __HAVE_VM86
if
((
void
*
)
EIP_sig
(
sigcontext
)
==
vm86_return
)
/* vm86 mode */
{
unsigned
int
*
int_stack
=
(
unsigned
int
*
)
stack
;
/* fetch the saved %fs and %gs from the stack */
wine_set_fs
(
int_stack
[
0
]
);
wine_set_gs
(
int_stack
[
1
]
);
}
else
#endif
{
#ifdef FS_sig
wine_set_fs
(
FS_sig
(
sigcontext
)
);
#endif
#ifdef GS_sig
wine_set_gs
(
GS_sig
(
sigcontext
)
);
#endif
}
}
/* stack sanity checks */
if
((
char
*
)
stack
>=
(
char
*
)
NtCurrentTeb
()
->
DeallocationStack
&&
(
char
*
)
stack
<
(
char
*
)
NtCurrentTeb
()
->
DeallocationStack
+
SIGNAL_STACK_SIZE
)
{
ERR
(
"nested exception on signal stack in thread %04lx eip %08lx esp %08lx stack %p-%p
\n
"
,
GetCurrentThreadId
(),
EIP_sig
(
sigcontext
),
ESP_sig
(
sigcontext
),
NtCurrentTeb
()
->
Tib
.
StackLimit
,
NtCurrentTeb
()
->
Tib
.
StackBase
);
SYSDEPS_AbortThread
(
1
);
}
if
((
char
*
)(
stack
-
1
)
<
(
char
*
)
NtCurrentTeb
()
->
Tib
.
StackLimit
+
SIGNAL_STACK_SIZE
+
4096
||
(
char
*
)
stack
>
(
char
*
)
NtCurrentTeb
()
->
Tib
.
StackBase
)
{
UINT
diff
=
(
char
*
)
NtCurrentTeb
()
->
Tib
.
StackLimit
+
SIGNAL_STACK_SIZE
+
4096
-
(
char
*
)
stack
;
if
(
diff
<
4096
)
ERR
(
"stack overflow %u bytes in thread %04lx eip %08lx esp %08lx stack %p-%p
\n
"
,
diff
,
GetCurrentThreadId
(),
EIP_sig
(
sigcontext
),
ESP_sig
(
sigcontext
),
NtCurrentTeb
()
->
Tib
.
StackLimit
,
NtCurrentTeb
()
->
Tib
.
StackBase
);
else
ERR
(
"exception outside of stack limits in thread %04lx eip %08lx esp %08lx stack %p-%p
\n
"
,
GetCurrentThreadId
(),
EIP_sig
(
sigcontext
),
ESP_sig
(
sigcontext
),
NtCurrentTeb
()
->
Tib
.
StackLimit
,
NtCurrentTeb
()
->
Tib
.
StackBase
);
SYSDEPS_AbortThread
(
1
);
}
stack
--
;
/* push the stack_layout structure */
stack
->
ret_addr
=
__wine_call_from_32_restore_regs
;
stack
->
rec_ptr
=
&
stack
->
rec
;
stack
->
context_ptr
=
&
stack
->
context
;
stack
->
rec
.
ExceptionRecord
=
NULL
;
stack
->
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
stack
->
rec
.
ExceptionAddress
=
(
LPVOID
)
EIP_sig
(
sigcontext
);
stack
->
rec
.
NumberParameters
=
0
;
stack
->
context
.
ContextFlags
=
CONTEXT_FULL
;
stack
->
context
.
Eax
=
EAX_sig
(
sigcontext
);
stack
->
context
.
Ebx
=
EBX_sig
(
sigcontext
);
stack
->
context
.
Ecx
=
ECX_sig
(
sigcontext
);
stack
->
context
.
Edx
=
EDX_sig
(
sigcontext
);
stack
->
context
.
Esi
=
ESI_sig
(
sigcontext
);
stack
->
context
.
Edi
=
EDI_sig
(
sigcontext
);
stack
->
context
.
Ebp
=
EBP_sig
(
sigcontext
);
stack
->
context
.
EFlags
=
EFL_sig
(
sigcontext
);
stack
->
context
.
Eip
=
EIP_sig
(
sigcontext
);
stack
->
context
.
Esp
=
ESP_sig
(
sigcontext
);
stack
->
context
.
SegCs
=
LOWORD
(
CS_sig
(
sigcontext
));
stack
->
context
.
SegDs
=
LOWORD
(
DS_sig
(
sigcontext
));
stack
->
context
.
SegEs
=
LOWORD
(
ES_sig
(
sigcontext
));
stack
->
context
.
SegFs
=
fs
;
stack
->
context
.
SegGs
=
gs
;
stack
->
context
.
SegSs
=
LOWORD
(
SS_sig
(
sigcontext
));
/* now modify the sigcontext to return to the raise function */
ESP_sig
(
sigcontext
)
=
(
DWORD
)
stack
;
EIP_sig
(
sigcontext
)
=
(
DWORD
)
func
;
CS_sig
(
sigcontext
)
=
wine_get_cs
();
DS_sig
(
sigcontext
)
=
wine_get_ds
();
ES_sig
(
sigcontext
)
=
wine_get_es
();
FS_sig
(
sigcontext
)
=
wine_get_fs
();
GS_sig
(
sigcontext
)
=
wine_get_gs
();
SS_sig
(
sigcontext
)
=
wine_get_ss
();
return
stack
->
rec_ptr
;
}
/***********************************************************************
* get_exception_context
*
* Get a pointer to the context built by setup_exception.
*/
static
inline
CONTEXT
*
get_exception_context
(
EXCEPTION_RECORD
*
rec
)
{
return
(
CONTEXT
*
)
rec
-
1
;
/* cf. stack_layout structure */
}
/**********************************************************************
/**********************************************************************
* get_fpu_code
* get_fpu_code
*
*
...
@@ -781,49 +929,20 @@ static inline DWORD get_fpu_code( const CONTEXT *context )
...
@@ -781,49 +929,20 @@ static inline DWORD get_fpu_code( const CONTEXT *context )
/**********************************************************************
/**********************************************************************
* do_segv
* raise_segv_exception
*
* Implementation of SIGSEGV handler.
*/
*/
static
void
do_segv
(
CONTEXT
*
context
,
int
trap_code
,
void
*
cr2
,
int
err_code
)
static
void
WINAPI
raise_segv_exception
(
EXCEPTION_RECORD
*
rec
,
CONTEXT
*
context
)
{
{
EXCEPTION_RECORD
rec
;
switch
(
rec
->
ExceptionCode
)
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
->
Eip
;
rec
.
NumberParameters
=
0
;
switch
(
trap_code
)
{
{
case
T_OFLOW
:
/* Overflow exception */
case
EXCEPTION_ACCESS_VIOLATION
:
rec
.
ExceptionCode
=
EXCEPTION_INT_OVERFLOW
;
if
(
rec
->
NumberParameters
==
2
)
break
;
{
case
T_BOUND
:
/* Bound range exception */
if
(
!
(
rec
->
ExceptionCode
=
VIRTUAL_HandleFault
(
(
void
*
)
rec
->
ExceptionInformation
[
1
]
)))
rec
.
ExceptionCode
=
EXCEPTION_ARRAY_BOUNDS_EXCEEDED
;
return
;
break
;
}
case
T_PRIVINFLT
:
/* Invalid opcode exception */
rec
.
ExceptionCode
=
EXCEPTION_ILLEGAL_INSTRUCTION
;
break
;
case
T_STKFLT
:
/* Stack fault */
rec
.
ExceptionCode
=
EXCEPTION_STACK_OVERFLOW
;
break
;
case
T_SEGNPFLT
:
/* Segment not present exception */
case
T_PROTFLT
:
/* General protection fault */
case
T_UNKNOWN
:
/* Unknown fault code */
rec
.
ExceptionCode
=
err_code
?
EXCEPTION_ACCESS_VIOLATION
:
EXCEPTION_PRIV_INSTRUCTION
;
break
;
case
T_PAGEFLT
:
/* Page fault */
#ifdef FAULT_ADDRESS
if
(
!
(
rec
.
ExceptionCode
=
VIRTUAL_HandleFault
(
cr2
)))
return
;
rec
.
NumberParameters
=
2
;
rec
.
ExceptionInformation
[
0
]
=
(
err_code
&
2
)
!=
0
;
rec
.
ExceptionInformation
[
1
]
=
(
DWORD
)
cr2
;
#else
rec
.
ExceptionCode
=
EXCEPTION_ACCESS_VIOLATION
;
#endif
break
;
break
;
case
T_ALIGNFLT
:
/* Alignment check exception */
case
EXCEPTION_DATATYPE_MISALIGNMENT
:
/* FIXME: pass through exception handler first? */
/* FIXME: pass through exception handler first? */
if
(
context
->
EFlags
&
0x00040000
)
if
(
context
->
EFlags
&
0x00040000
)
{
{
...
@@ -831,46 +950,21 @@ static void do_segv( CONTEXT *context, int trap_code, void *cr2, int err_code )
...
@@ -831,46 +950,21 @@ static void do_segv( CONTEXT *context, int trap_code, void *cr2, int err_code )
context
->
EFlags
&=
~
0x00040000
;
context
->
EFlags
&=
~
0x00040000
;
return
;
return
;
}
}
rec
.
ExceptionCode
=
EXCEPTION_DATATYPE_MISALIGNMENT
;
break
;
default:
ERR
(
"Got unexpected trap %d
\n
"
,
trap_code
);
/* fall through */
case
T_NMI
:
/* NMI interrupt */
case
T_DNA
:
/* Device not available exception */
case
T_DOUBLEFLT
:
/* Double fault exception */
case
T_TSSFLT
:
/* Invalid TSS exception */
case
T_RESERVED
:
/* Unknown exception */
case
T_MCHK
:
/* Machine check exception */
#ifdef T_CACHEFLT
case
T_CACHEFLT
:
/* Cache flush exception */
#endif
rec
.
ExceptionCode
=
EXCEPTION_ILLEGAL_INSTRUCTION
;
break
;
break
;
}
}
EXC_RtlRaiseException
(
&
rec
,
context
);
EXC_RtlRaiseException
(
rec
,
context
);
}
}
/**********************************************************************
/**********************************************************************
* do_trap
* raise_trap_exception
*
* Implementation of SIGTRAP handler.
*/
*/
static
void
do_trap
(
CONTEXT
*
context
,
int
trap_code
)
static
void
WINAPI
raise_trap_exception
(
EXCEPTION_RECORD
*
rec
,
CONTEXT
*
context
)
{
{
EXCEPTION_RECORD
rec
;
DWORD
dr0
,
dr1
,
dr2
,
dr3
,
dr6
,
dr7
;
DWORD
dr0
,
dr1
,
dr2
,
dr3
,
dr6
,
dr7
;
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
if
(
rec
->
ExceptionCode
==
EXCEPTION_SINGLE_STEP
)
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
->
Eip
;
rec
.
NumberParameters
=
0
;
switch
(
trap_code
)
{
{
case
T_TRCTRAP
:
/* Single-step exception */
rec
.
ExceptionCode
=
EXCEPTION_SINGLE_STEP
;
if
(
context
->
EFlags
&
0x100
)
if
(
context
->
EFlags
&
0x100
)
{
{
context
->
EFlags
&=
~
0x100
;
/* clear single-step flag */
context
->
EFlags
&=
~
0x100
;
/* clear single-step flag */
...
@@ -883,17 +977,10 @@ static void do_trap( CONTEXT *context, int trap_code )
...
@@ -883,17 +977,10 @@ static void do_trap( CONTEXT *context, int trap_code )
* if not, then someone did a kill(SIGTRAP) on us, and we
* if not, then someone did a kill(SIGTRAP) on us, and we
* shall return a breakpoint, not a single step exception
* shall return a breakpoint, not a single step exception
*/
*/
if
(
!
(
context
->
Dr6
&
0xf
))
if
(
!
(
context
->
Dr6
&
0xf
))
rec
->
ExceptionCode
=
EXCEPTION_BREAKPOINT
;
rec
.
ExceptionCode
=
EXCEPTION_BREAKPOINT
;
}
}
break
;
case
T_BPTFLT
:
/* Breakpoint exception */
rec
.
ExceptionAddress
=
(
char
*
)
rec
.
ExceptionAddress
-
1
;
/* back up over the int3 instruction */
/* fall through */
default:
rec
.
ExceptionCode
=
EXCEPTION_BREAKPOINT
;
break
;
}
}
dr0
=
context
->
Dr0
;
dr0
=
context
->
Dr0
;
dr1
=
context
->
Dr1
;
dr1
=
context
->
Dr1
;
dr2
=
context
->
Dr2
;
dr2
=
context
->
Dr2
;
...
@@ -901,7 +988,7 @@ static void do_trap( CONTEXT *context, int trap_code )
...
@@ -901,7 +988,7 @@ static void do_trap( CONTEXT *context, int trap_code )
dr6
=
context
->
Dr6
;
dr6
=
context
->
Dr6
;
dr7
=
context
->
Dr7
;
dr7
=
context
->
Dr7
;
EXC_RtlRaiseException
(
&
rec
,
context
);
EXC_RtlRaiseException
(
rec
,
context
);
if
(
dr0
!=
context
->
Dr0
||
dr1
!=
context
->
Dr1
||
dr2
!=
context
->
Dr2
||
if
(
dr0
!=
context
->
Dr0
||
dr1
!=
context
->
Dr1
||
dr2
!=
context
->
Dr2
||
dr3
!=
context
->
Dr3
||
dr6
!=
context
->
Dr6
||
dr7
!=
context
->
Dr7
)
dr3
!=
context
->
Dr3
||
dr6
!=
context
->
Dr6
||
dr7
!=
context
->
Dr7
)
...
@@ -914,36 +1001,12 @@ static void do_trap( CONTEXT *context, int trap_code )
...
@@ -914,36 +1001,12 @@ static void do_trap( CONTEXT *context, int trap_code )
/**********************************************************************
/**********************************************************************
* do_fpe
* raise_fpu_exception
*
* Implementation of SIGFPE handler
*/
*/
static
void
do_fpe
(
CONTEXT
*
context
,
int
trap_code
)
static
void
WINAPI
raise_fpu_exception
(
EXCEPTION_RECORD
*
rec
,
CONTEXT
*
context
)
{
{
EXCEPTION_RECORD
rec
;
EXC_RtlRaiseException
(
rec
,
context
);
restore_fpu
(
context
);
switch
(
trap_code
)
{
case
T_DIVIDE
:
/* Division by zero exception */
rec
.
ExceptionCode
=
EXCEPTION_INT_DIVIDE_BY_ZERO
;
break
;
case
T_FPOPFLT
:
/* Coprocessor segment overrun */
rec
.
ExceptionCode
=
EXCEPTION_FLT_INVALID_OPERATION
;
break
;
case
T_ARITHTRAP
:
/* Floating point exception */
case
T_UNKNOWN
:
/* Unknown fault code */
rec
.
ExceptionCode
=
get_fpu_code
(
context
);
break
;
default:
ERR
(
"Got unexpected trap %d
\n
"
,
trap_code
);
rec
.
ExceptionCode
=
EXCEPTION_FLT_INVALID_OPERATION
;
break
;
}
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
->
Eip
;
rec
.
NumberParameters
=
0
;
EXC_RtlRaiseException
(
&
rec
,
context
);
}
}
...
@@ -1031,6 +1094,7 @@ static HANDLER_DEF(usr2_handler)
...
@@ -1031,6 +1094,7 @@ static HANDLER_DEF(usr2_handler)
{
{
CONTEXT
context
;
CONTEXT
context
;
/* FIXME: should use setup_exception here */
save_context
(
&
context
,
HANDLER_CONTEXT
);
save_context
(
&
context
,
HANDLER_CONTEXT
);
set_vm86_pend
(
&
context
);
set_vm86_pend
(
&
context
);
restore_context
(
&
context
,
HANDLER_CONTEXT
);
restore_context
(
&
context
,
HANDLER_CONTEXT
);
...
@@ -1045,17 +1109,54 @@ static HANDLER_DEF(usr2_handler)
...
@@ -1045,17 +1109,54 @@ static HANDLER_DEF(usr2_handler)
*/
*/
static
HANDLER_DEF
(
segv_handler
)
static
HANDLER_DEF
(
segv_handler
)
{
{
CONTEXT
context
;
EXCEPTION_RECORD
*
rec
=
setup_exception
(
HANDLER_CONTEXT
,
raise_segv_exception
);
void
*
cr2
;
save_context
(
&
context
,
HANDLER_CONTEXT
);
switch
(
get_trap_code
(
HANDLER_CONTEXT
))
{
case
T_OFLOW
:
/* Overflow exception */
rec
->
ExceptionCode
=
EXCEPTION_INT_OVERFLOW
;
break
;
case
T_BOUND
:
/* Bound range exception */
rec
->
ExceptionCode
=
EXCEPTION_ARRAY_BOUNDS_EXCEEDED
;
break
;
case
T_PRIVINFLT
:
/* Invalid opcode exception */
rec
->
ExceptionCode
=
EXCEPTION_ILLEGAL_INSTRUCTION
;
break
;
case
T_STKFLT
:
/* Stack fault */
rec
->
ExceptionCode
=
EXCEPTION_STACK_OVERFLOW
;
break
;
case
T_SEGNPFLT
:
/* Segment not present exception */
case
T_PROTFLT
:
/* General protection fault */
case
T_UNKNOWN
:
/* Unknown fault code */
rec
->
ExceptionCode
=
get_error_code
(
HANDLER_CONTEXT
)
?
EXCEPTION_ACCESS_VIOLATION
:
EXCEPTION_PRIV_INSTRUCTION
;
break
;
case
T_PAGEFLT
:
/* Page fault */
rec
->
ExceptionCode
=
EXCEPTION_ACCESS_VIOLATION
;
#ifdef FAULT_ADDRESS
#ifdef FAULT_ADDRESS
cr2
=
FAULT_ADDRESS
;
rec
->
NumberParameters
=
2
;
#else
rec
->
ExceptionInformation
[
0
]
=
(
get_error_code
(
HANDLER_CONTEXT
)
&
2
)
!=
0
;
cr2
=
NULL
;
rec
->
ExceptionInformation
[
1
]
=
(
DWORD
)
FAULT_ADDRESS
;
#endif
#endif
do_segv
(
&
context
,
get_trap_code
(
HANDLER_CONTEXT
),
cr2
,
get_error_code
(
HANDLER_CONTEXT
)
);
break
;
restore_context
(
&
context
,
HANDLER_CONTEXT
);
case
T_ALIGNFLT
:
/* Alignment check exception */
rec
->
ExceptionCode
=
EXCEPTION_DATATYPE_MISALIGNMENT
;
break
;
default:
ERR
(
"Got unexpected trap %d
\n
"
,
get_trap_code
(
HANDLER_CONTEXT
)
);
/* fall through */
case
T_NMI
:
/* NMI interrupt */
case
T_DNA
:
/* Device not available exception */
case
T_DOUBLEFLT
:
/* Double fault exception */
case
T_TSSFLT
:
/* Invalid TSS exception */
case
T_RESERVED
:
/* Unknown exception */
case
T_MCHK
:
/* Machine check exception */
#ifdef T_CACHEFLT
case
T_CACHEFLT
:
/* Cache flush exception */
#endif
rec
->
ExceptionCode
=
EXCEPTION_ILLEGAL_INSTRUCTION
;
break
;
}
}
}
...
@@ -1066,10 +1167,21 @@ static HANDLER_DEF(segv_handler)
...
@@ -1066,10 +1167,21 @@ static HANDLER_DEF(segv_handler)
*/
*/
static
HANDLER_DEF
(
trap_handler
)
static
HANDLER_DEF
(
trap_handler
)
{
{
CONTEXT
context
;
EXCEPTION_RECORD
*
rec
=
setup_exception
(
HANDLER_CONTEXT
,
raise_trap_exception
);
save_context
(
&
context
,
HANDLER_CONTEXT
);
do_trap
(
&
context
,
get_trap_code
(
HANDLER_CONTEXT
)
);
switch
(
get_trap_code
(
HANDLER_CONTEXT
))
restore_context
(
&
context
,
HANDLER_CONTEXT
);
{
case
T_TRCTRAP
:
/* Single-step exception */
rec
->
ExceptionCode
=
EXCEPTION_SINGLE_STEP
;
EFL_sig
(
HANDLER_CONTEXT
)
&=
~
0x100
;
/* clear single-step flag */
break
;
case
T_BPTFLT
:
/* Breakpoint exception */
rec
->
ExceptionAddress
=
(
char
*
)
rec
->
ExceptionAddress
-
1
;
/* back up over the int3 instruction */
/* fall through */
default:
rec
->
ExceptionCode
=
EXCEPTION_BREAKPOINT
;
break
;
}
}
}
...
@@ -1080,12 +1192,29 @@ static HANDLER_DEF(trap_handler)
...
@@ -1080,12 +1192,29 @@ static HANDLER_DEF(trap_handler)
*/
*/
static
HANDLER_DEF
(
fpe_handler
)
static
HANDLER_DEF
(
fpe_handler
)
{
{
CONTEXT
context
;
EXCEPTION_RECORD
*
rec
=
setup_exception
(
HANDLER_CONTEXT
,
raise_fpu_exception
);
save_fpu
(
&
context
,
HANDLER_CONTEXT
);
CONTEXT
*
context
;
save_context
(
&
context
,
HANDLER_CONTEXT
);
do_fpe
(
&
context
,
get_trap_code
(
HANDLER_CONTEXT
)
);
context
=
get_exception_context
(
rec
);
restore_context
(
&
context
,
HANDLER_CONTEXT
);
save_fpu
(
context
,
HANDLER_CONTEXT
);
restore_fpu
(
&
context
,
HANDLER_CONTEXT
);
switch
(
get_trap_code
(
HANDLER_CONTEXT
))
{
case
T_DIVIDE
:
/* Division by zero exception */
rec
->
ExceptionCode
=
EXCEPTION_INT_DIVIDE_BY_ZERO
;
break
;
case
T_FPOPFLT
:
/* Coprocessor segment overrun */
rec
->
ExceptionCode
=
EXCEPTION_FLT_INVALID_OPERATION
;
break
;
case
T_ARITHTRAP
:
/* Floating point exception */
case
T_UNKNOWN
:
/* Unknown fault code */
rec
->
ExceptionCode
=
get_fpu_code
(
context
);
break
;
default:
ERR
(
"Got unexpected trap %d
\n
"
,
get_trap_code
(
HANDLER_CONTEXT
)
);
rec
->
ExceptionCode
=
EXCEPTION_FLT_INVALID_OPERATION
;
break
;
}
}
}
...
@@ -1099,17 +1228,8 @@ static HANDLER_DEF(int_handler)
...
@@ -1099,17 +1228,8 @@ static HANDLER_DEF(int_handler)
init_handler
(
HANDLER_CONTEXT
);
init_handler
(
HANDLER_CONTEXT
);
if
(
!
dispatch_signal
(
SIGINT
))
if
(
!
dispatch_signal
(
SIGINT
))
{
{
EXCEPTION_RECORD
rec
;
EXCEPTION_RECORD
*
rec
=
setup_exception
(
HANDLER_CONTEXT
,
EXC_RtlRaiseException
);
CONTEXT
context
;
rec
->
ExceptionCode
=
CONTROL_C_EXIT
;
save_context
(
&
context
,
HANDLER_CONTEXT
);
rec
.
ExceptionCode
=
CONTROL_C_EXIT
;
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
.
Eip
;
rec
.
NumberParameters
=
0
;
EXC_RtlRaiseException
(
&
rec
,
&
context
);
restore_context
(
&
context
,
HANDLER_CONTEXT
);
}
}
}
}
...
@@ -1120,17 +1240,9 @@ static HANDLER_DEF(int_handler)
...
@@ -1120,17 +1240,9 @@ static HANDLER_DEF(int_handler)
*/
*/
static
HANDLER_DEF
(
abrt_handler
)
static
HANDLER_DEF
(
abrt_handler
)
{
{
EXCEPTION_RECORD
rec
;
EXCEPTION_RECORD
*
rec
=
setup_exception
(
HANDLER_CONTEXT
,
EXC_RtlRaiseException
);
CONTEXT
context
;
rec
->
ExceptionCode
=
EXCEPTION_WINE_ASSERTION
;
rec
->
ExceptionFlags
=
EH_NONCONTINUABLE
;
save_context
(
&
context
,
HANDLER_CONTEXT
);
rec
.
ExceptionCode
=
EXCEPTION_WINE_ASSERTION
;
rec
.
ExceptionFlags
=
EH_NONCONTINUABLE
;
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
.
Eip
;
rec
.
NumberParameters
=
0
;
EXC_RtlRaiseException
(
&
rec
,
&
context
);
/* Should never return.. */
restore_context
(
&
context
,
HANDLER_CONTEXT
);
}
}
...
@@ -1350,14 +1462,19 @@ void __wine_enter_vm86( CONTEXT *context )
...
@@ -1350,14 +1462,19 @@ void __wine_enter_vm86( CONTEXT *context )
* (avoiding sigprocmask for performance reasons) */
* (avoiding sigprocmask for performance reasons) */
teb
->
vm86_ptr
=
&
vm86
;
teb
->
vm86_ptr
=
&
vm86
;
vm86
.
regs
.
eflags
|=
teb
->
vm86_pending
;
vm86
.
regs
.
eflags
|=
teb
->
vm86_pending
;
/* Check for VIF|VIP here, since vm86_enter doesn't */
/* Check for VIF|VIP here, since vm86_enter doesn't */
if
((
vm86
.
regs
.
eflags
&
(
VIF_MASK
|
VIP_MASK
))
==
(
VIF_MASK
|
VIP_MASK
))
{
if
((
vm86
.
regs
.
eflags
&
(
VIF_MASK
|
VIP_MASK
))
==
(
VIF_MASK
|
VIP_MASK
))
{
teb
->
vm86_ptr
=
NULL
;
teb
->
vm86_ptr
=
NULL
;
teb
->
vm86_pending
=
0
;
teb
->
vm86_pending
=
0
;
context
->
EFlags
|=
VIP_MASK
;
context
->
EFlags
|=
VIP_MASK
;
rec
.
ExceptionCode
=
EXCEPTION_VM86_STI
;
rec
.
ExceptionCode
=
EXCEPTION_VM86_STI
;
rec
.
ExceptionInformation
[
0
]
=
0
;
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
goto
cancel_vm86
;
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
->
Eip
;
rec
.
NumberParameters
=
0
;
EXC_RtlRaiseException
(
&
rec
,
context
);
continue
;
}
}
do
do
...
@@ -1373,16 +1490,37 @@ void __wine_enter_vm86( CONTEXT *context )
...
@@ -1373,16 +1490,37 @@ void __wine_enter_vm86( CONTEXT *context )
save_vm86_context
(
context
,
&
vm86
);
save_vm86_context
(
context
,
&
vm86
);
context
->
EFlags
|=
teb
->
vm86_pending
;
context
->
EFlags
|=
teb
->
vm86_pending
;
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
->
Eip
;
rec
.
NumberParameters
=
0
;
switch
(
VM86_TYPE
(
res
))
switch
(
VM86_TYPE
(
res
))
{
{
case
VM86_UNKNOWN
:
/* unhandled GP fault - IO-instruction or similar */
case
VM86_UNKNOWN
:
/* unhandled GP fault - IO-instruction or similar */
do_segv
(
context
,
T_PROTFLT
,
0
,
0
);
rec
.
ExceptionCode
=
EXCEPTION_PRIV_INSTRUCTION
;
raise_segv_exception
(
&
rec
,
context
);
continue
;
continue
;
case
VM86_TRAP
:
/* return due to DOS-debugger request */
case
VM86_TRAP
:
/* return due to DOS-debugger request */
do_trap
(
context
,
VM86_ARG
(
res
)
);
switch
(
VM86_ARG
(
res
))
{
case
T_TRCTRAP
:
/* Single-step exception */
rec
.
ExceptionCode
=
EXCEPTION_SINGLE_STEP
;
context
->
EFlags
&=
~
0x100
;
/* clear single-step flag */
break
;
case
T_BPTFLT
:
/* Breakpoint exception */
rec
.
ExceptionAddress
=
(
char
*
)
rec
.
ExceptionAddress
-
1
;
/* back up over the int3 instruction */
/* fall through */
default:
rec
.
ExceptionCode
=
EXCEPTION_BREAKPOINT
;
break
;
}
raise_trap_exception
(
&
rec
,
context
);
continue
;
continue
;
case
VM86_INTx
:
/* int3/int x instruction (ARG = x) */
case
VM86_INTx
:
/* int3/int x instruction (ARG = x) */
rec
.
ExceptionCode
=
EXCEPTION_VM86_INTx
;
rec
.
ExceptionCode
=
EXCEPTION_VM86_INTx
;
rec
.
NumberParameters
=
1
;
rec
.
ExceptionInformation
[
0
]
=
VM86_ARG
(
res
);
break
;
break
;
case
VM86_STI
:
/* sti/popf/iret instruction enabled virtual interrupts */
case
VM86_STI
:
/* sti/popf/iret instruction enabled virtual interrupts */
teb
->
vm86_pending
=
0
;
teb
->
vm86_pending
=
0
;
...
@@ -1395,12 +1533,6 @@ void __wine_enter_vm86( CONTEXT *context )
...
@@ -1395,12 +1533,6 @@ void __wine_enter_vm86( CONTEXT *context )
ERR
(
"unhandled result from vm86 mode %x
\n
"
,
res
);
ERR
(
"unhandled result from vm86 mode %x
\n
"
,
res
);
continue
;
continue
;
}
}
rec
.
ExceptionInformation
[
0
]
=
VM86_ARG
(
res
);
cancel_vm86:
rec
.
ExceptionFlags
=
EXCEPTION_CONTINUABLE
;
rec
.
ExceptionRecord
=
NULL
;
rec
.
ExceptionAddress
=
(
LPVOID
)
context
->
Eip
;
rec
.
NumberParameters
=
1
;
EXC_RtlRaiseException
(
&
rec
,
context
);
EXC_RtlRaiseException
(
&
rec
,
context
);
}
}
}
}
...
...
tools/winebuild/relay.c
View file @
4cb00196
...
@@ -1006,41 +1006,40 @@ static void BuildCallFrom32Regs( FILE *outfile )
...
@@ -1006,41 +1006,40 @@ static void BuildCallFrom32Regs( FILE *outfile )
/* Call the entry point */
/* Call the entry point */
fprintf
(
outfile
,
"
\t
call *0(%%ebx)
\n
"
);
fprintf
(
outfile
,
"
\t
call *0(%%ebx)
\n
"
);
fprintf
(
outfile
,
"
\t
leal -%d(%%ebp),%%ecx
\n
"
,
STACK_SPACE
);
/* Store %eip and %ebp onto the new stack */
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%edx
\n
"
,
CONTEXTOFFSET
(
Esp
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%eax
\n
"
,
CONTEXTOFFSET
(
Eip
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %%eax,-4(%%edx)
\n
"
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%eax
\n
"
,
CONTEXTOFFSET
(
Ebp
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %%eax,-8(%%edx)
\n
"
);
/* Restore the context structure */
/* Restore the context structure */
/* Note: we don't bother to restore %cs, %ds and %ss
fprintf
(
outfile
,
"2:
\t
pushl %d(%%ecx)
\n
"
,
CONTEXTOFFSET
(
SegEs
)
);
* changing them in 32-bit code is a recipe for disaster anyway
*/
fprintf
(
outfile
,
"
\t
pushl %d(%%ebp)
\n
"
,
CONTEXTOFFSET
(
SegEs
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
popl %%es
\n
"
);
fprintf
(
outfile
,
"
\t
popl %%es
\n
"
);
fprintf
(
outfile
,
"
\t
pushl %d(%%e
bp)
\n
"
,
CONTEXTOFFSET
(
SegFs
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
pushl %d(%%e
cx)
\n
"
,
CONTEXTOFFSET
(
SegFs
)
);
fprintf
(
outfile
,
"
\t
popl %%fs
\n
"
);
fprintf
(
outfile
,
"
\t
popl %%fs
\n
"
);
fprintf
(
outfile
,
"
\t
pushl %d(%%e
bp)
\n
"
,
CONTEXTOFFSET
(
SegGs
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
pushl %d(%%e
cx)
\n
"
,
CONTEXTOFFSET
(
SegGs
)
);
fprintf
(
outfile
,
"
\t
popl %%gs
\n
"
);
fprintf
(
outfile
,
"
\t
popl %%gs
\n
"
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%edi
\n
"
,
CONTEXTOFFSET
(
Edi
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%edi
\n
"
,
CONTEXTOFFSET
(
Edi
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%esi
\n
"
,
CONTEXTOFFSET
(
Esi
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%esi
\n
"
,
CONTEXTOFFSET
(
Esi
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%edx
\n
"
,
CONTEXTOFFSET
(
Edx
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%edx
\n
"
,
CONTEXTOFFSET
(
Edx
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%ecx
\n
"
,
CONTEXTOFFSET
(
Ecx
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%ebx
\n
"
,
CONTEXTOFFSET
(
Ebx
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%ebx
\n
"
,
CONTEXTOFFSET
(
Ebx
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%eax
\n
"
,
CONTEXTOFFSET
(
Eax
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%ebp
\n
"
,
CONTEXTOFFSET
(
Ebp
)
);
fprintf
(
outfile
,
"
\t
pushl %d(%%e
bp)
\n
"
,
CONTEXTOFFSET
(
EFlags
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
pushl %d(%%e
cx)
\n
"
,
CONTEXTOFFSET
(
SegSs
)
);
fprintf
(
outfile
,
"
\t
pop
fl
\n
"
);
fprintf
(
outfile
,
"
\t
pop
l %%ss
\n
"
);
fprintf
(
outfile
,
"
\t
movl %d(%%e
bp),%%eax
\n
"
,
CONTEXTOFFSET
(
Eax
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
movl %d(%%e
cx),%%esp
\n
"
,
CONTEXTOFFSET
(
Esp
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ebp),%%ebp
\n
"
,
CONTEXTOFFSET
(
Esp
)
-
STACK_SPACE
);
fprintf
(
outfile
,
"
\t
pushl %d(%%ecx)
\n
"
,
CONTEXTOFFSET
(
EFlags
)
);
fprintf
(
outfile
,
"
\t
leal -8(%%ebp),%%esp
\n
"
);
fprintf
(
outfile
,
"
\t
pushl %d(%%ecx)
\n
"
,
CONTEXTOFFSET
(
SegCs
)
);
fprintf
(
outfile
,
"
\t
popl %%ebp
\n
"
);
fprintf
(
outfile
,
"
\t
pushl %d(%%ecx)
\n
"
,
CONTEXTOFFSET
(
Eip
)
);
fprintf
(
outfile
,
"
\t
ret
\n
"
);
fprintf
(
outfile
,
"
\t
pushl %d(%%ecx)
\n
"
,
CONTEXTOFFSET
(
SegDs
)
);
fprintf
(
outfile
,
"
\t
movl %d(%%ecx),%%ecx
\n
"
,
CONTEXTOFFSET
(
Ecx
)
);
fprintf
(
outfile
,
"
\t
popl %%ds
\n
"
);
fprintf
(
outfile
,
"
\t
iret
\n
"
);
function_header
(
outfile
,
"__wine_call_from_32_restore_regs"
);
fprintf
(
outfile
,
"
\t
leal 4(%%esp),%%ecx
\n
"
);
fprintf
(
outfile
,
"
\t
jmp 2b
\n
"
);
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment