Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-winehq
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-winehq
Commits
bbcfa6b4
Commit
bbcfa6b4
authored
Jun 05, 2015
by
Ken Thomases
Committed by
Alexandre Julliard
Jun 05, 2015
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
ntdll: Add support for using libunwind to unwind the stack for x86_64.
On OS X, Apple uses compact unwind info in preference to DWARF unwind info.
parent
260832c5
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
215 additions
and
1 deletion
+215
-1
signal_x86_64.c
dlls/ntdll/signal_x86_64.c
+215
-1
No files found.
dlls/ntdll/signal_x86_64.c
View file @
bbcfa6b4
...
...
@@ -51,6 +51,10 @@
#ifdef HAVE_SYS_UCONTEXT_H
# include <sys/ucontext.h>
#endif
#ifdef HAVE_LIBUNWIND_H
# define UNW_LOCAL_ONLY
# include <libunwind.h>
#endif
#define NONAMELESSUNION
#define NONAMELESSSTRUCT
...
...
@@ -1424,6 +1428,188 @@ static NTSTATUS dwarf_virtual_unwind( ULONG64 ip, ULONG64 *frame,CONTEXT *contex
}
#if HAVE_LIBUNWIND_H
/***********************************************************************
* libunwind_set_cursor_from_context
*/
static
int
libunwind_set_cursor_from_context
(
unw_cursor_t
*
cursor
,
CONTEXT
*
context
,
ULONG64
ip
)
{
int
rc
;
rc
=
unw_set_reg
(
cursor
,
UNW_REG_IP
,
ip
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_REG_SP
,
context
->
Rsp
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RAX
,
context
->
Rax
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RDX
,
context
->
Rdx
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RCX
,
context
->
Rcx
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RBX
,
context
->
Rbx
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RSI
,
context
->
Rsi
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RDI
,
context
->
Rdi
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_RBP
,
context
->
Rbp
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R8
,
context
->
R8
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R9
,
context
->
R9
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R10
,
context
->
R10
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R11
,
context
->
R11
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R12
,
context
->
R12
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R13
,
context
->
R13
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R14
,
context
->
R14
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_set_reg
(
cursor
,
UNW_X86_64_R15
,
context
->
R15
);
return
rc
;
}
/***********************************************************************
* libunwind_get_reg
*/
static
int
libunwind_get_reg
(
unw_cursor_t
*
cursor
,
unw_regnum_t
reg
,
ULONG64
*
val
)
{
int
rc
;
unw_word_t
word
;
rc
=
unw_get_reg
(
cursor
,
reg
,
&
word
);
if
(
rc
==
UNW_ESUCCESS
)
*
val
=
word
;
return
rc
;
}
/***********************************************************************
* libunwind_set_context_from_cursor
*/
static
BOOL
libunwind_set_context_from_cursor
(
CONTEXT
*
context
,
unw_cursor_t
*
cursor
)
{
int
rc
;
rc
=
libunwind_get_reg
(
cursor
,
UNW_REG_IP
,
&
context
->
Rip
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_REG_SP
,
&
context
->
Rsp
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RAX
,
&
context
->
Rax
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RDX
,
&
context
->
Rdx
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RCX
,
&
context
->
Rcx
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RBX
,
&
context
->
Rbx
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RSI
,
&
context
->
Rsi
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RDI
,
&
context
->
Rdi
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_RBP
,
&
context
->
Rbp
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R8
,
&
context
->
R8
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R9
,
&
context
->
R9
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R10
,
&
context
->
R10
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R11
,
&
context
->
R11
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R12
,
&
context
->
R12
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R13
,
&
context
->
R13
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R14
,
&
context
->
R14
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_get_reg
(
cursor
,
UNW_X86_64_R15
,
&
context
->
R15
);
return
rc
;
}
/***********************************************************************
* libunwind_virtual_unwind
*
* Equivalent of RtlVirtualUnwind for builtin modules.
*/
static
NTSTATUS
libunwind_virtual_unwind
(
ULONG64
ip
,
BOOL
*
got_info
,
ULONG64
*
frame
,
CONTEXT
*
context
,
PEXCEPTION_ROUTINE
*
handler
,
void
**
handler_data
)
{
unw_context_t
unw_context
;
unw_cursor_t
cursor
;
unw_proc_info_t
info
;
int
rc
;
rc
=
unw_getcontext
(
&
unw_context
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
unw_init_local
(
&
cursor
,
&
unw_context
);
if
(
rc
==
UNW_ESUCCESS
)
rc
=
libunwind_set_cursor_from_context
(
&
cursor
,
context
,
ip
-
1
);
if
(
rc
!=
UNW_ESUCCESS
)
{
WARN
(
"setup failed: %d
\n
"
,
rc
);
return
STATUS_INVALID_DISPOSITION
;
}
rc
=
unw_get_proc_info
(
&
cursor
,
&
info
);
if
(
rc
!=
UNW_ESUCCESS
&&
rc
!=
UNW_ENOINFO
)
{
WARN
(
"failed to get info: %d
\n
"
,
rc
);
return
STATUS_INVALID_DISPOSITION
;
}
if
(
rc
==
UNW_ENOINFO
||
ip
<
info
.
start_ip
||
info
.
end_ip
<=
ip
||
!
info
.
format
)
{
*
got_info
=
FALSE
;
return
STATUS_SUCCESS
;
}
TRACE
(
"ip %#lx function %#lx-%#lx personality %#lx lsda %#lx fde %#lx
\n
"
,
ip
,
(
unsigned
long
)
info
.
start_ip
,
(
unsigned
long
)
info
.
end_ip
,
(
unsigned
long
)
info
.
handler
,
(
unsigned
long
)
info
.
lsda
,
(
unsigned
long
)
info
.
unwind_info
);
rc
=
unw_step
(
&
cursor
);
if
(
rc
<
0
)
{
WARN
(
"failed to unwind: %d
\n
"
,
rc
);
return
STATUS_INVALID_DISPOSITION
;
}
*
frame
=
context
->
Rsp
;
rc
=
libunwind_set_context_from_cursor
(
context
,
&
cursor
);
if
(
rc
!=
UNW_ESUCCESS
)
{
WARN
(
"failed to update context after unwind: %d
\n
"
,
rc
);
return
STATUS_INVALID_DISPOSITION
;
}
*
handler
=
(
void
*
)
info
.
handler
;
*
handler_data
=
(
void
*
)
info
.
lsda
;
*
got_info
=
TRUE
;
TRACE
(
"next function rip=%016lx
\n
"
,
context
->
Rip
);
TRACE
(
" rax=%016lx rbx=%016lx rcx=%016lx rdx=%016lx
\n
"
,
context
->
Rax
,
context
->
Rbx
,
context
->
Rcx
,
context
->
Rdx
);
TRACE
(
" rsi=%016lx rdi=%016lx rbp=%016lx rsp=%016lx
\n
"
,
context
->
Rsi
,
context
->
Rdi
,
context
->
Rbp
,
context
->
Rsp
);
TRACE
(
" r8=%016lx r9=%016lx r10=%016lx r11=%016lx
\n
"
,
context
->
R8
,
context
->
R9
,
context
->
R10
,
context
->
R11
);
TRACE
(
" r12=%016lx r13=%016lx r14=%016lx r15=%016lx
\n
"
,
context
->
R12
,
context
->
R13
,
context
->
R14
,
context
->
R15
);
return
STATUS_SUCCESS
;
}
#endif
/***********************************************************************
* dispatch_signal
*/
...
...
@@ -2141,6 +2327,7 @@ static NTSTATUS call_stack_handlers( EXCEPTION_RECORD *rec, CONTEXT *orig_contex
if
(
!
module
||
(
module
->
Flags
&
LDR_WINE_INTERNAL
))
{
BOOL
got_info
=
FALSE
;
struct
dwarf_eh_bases
bases
;
const
struct
dwarf_fde
*
fde
=
_Unwind_Find_FDE
(
(
void
*
)(
context
.
Rip
-
1
),
&
bases
);
...
...
@@ -2149,6 +2336,19 @@ static NTSTATUS call_stack_handlers( EXCEPTION_RECORD *rec, CONTEXT *orig_contex
status
=
dwarf_virtual_unwind
(
context
.
Rip
,
&
dispatch
.
EstablisherFrame
,
&
new_context
,
fde
,
&
bases
,
&
dispatch
.
LanguageHandler
,
&
dispatch
.
HandlerData
);
if
(
status
!=
STATUS_SUCCESS
)
return
status
;
got_info
=
TRUE
;
}
#if HAVE_LIBUNWIND_H
else
{
status
=
libunwind_virtual_unwind
(
context
.
Rip
,
&
got_info
,
&
dispatch
.
EstablisherFrame
,
&
new_context
,
&
dispatch
.
LanguageHandler
,
&
dispatch
.
HandlerData
);
if
(
status
!=
STATUS_SUCCESS
)
return
status
;
}
#endif
if
(
got_info
)
{
dispatch
.
FunctionEntry
=
NULL
;
if
(
dispatch
.
LanguageHandler
&&
!
module
)
{
...
...
@@ -3155,15 +3355,29 @@ void WINAPI RtlUnwindEx( PVOID end_frame, PVOID target_ip, EXCEPTION_RECORD *rec
if
(
!
module
||
(
module
->
Flags
&
LDR_WINE_INTERNAL
))
{
BOOL
got_info
=
FALSE
;
struct
dwarf_eh_bases
bases
;
const
struct
dwarf_fde
*
fde
=
_Unwind_Find_FDE
(
(
void
*
)(
context
->
Rip
-
1
),
&
bases
);
if
(
fde
)
{
dispatch
.
FunctionEntry
=
NULL
;
status
=
dwarf_virtual_unwind
(
context
->
Rip
,
&
dispatch
.
EstablisherFrame
,
&
new_context
,
fde
,
&
bases
,
&
dispatch
.
LanguageHandler
,
&
dispatch
.
HandlerData
);
if
(
status
!=
STATUS_SUCCESS
)
raise_status
(
status
,
rec
);
got_info
=
TRUE
;
}
#if HAVE_LIBUNWIND_H
else
{
status
=
libunwind_virtual_unwind
(
context
->
Rip
,
&
got_info
,
&
dispatch
.
EstablisherFrame
,
&
new_context
,
&
dispatch
.
LanguageHandler
,
&
dispatch
.
HandlerData
);
if
(
status
!=
STATUS_SUCCESS
)
raise_status
(
status
,
rec
);
}
#endif
if
(
got_info
)
{
dispatch
.
FunctionEntry
=
NULL
;
if
(
dispatch
.
LanguageHandler
&&
!
module
)
{
FIXME
(
"calling personality routine in system library not supported yet
\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