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
3f20261e
Commit
3f20261e
authored
Dec 01, 2021
by
Paul Gofman
Committed by
Alexandre Julliard
Dec 03, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
ntdll: Set rcx on exit from syscall dispatcher on x64.
Signed-off-by:
Paul Gofman
<
pgofman@codeweavers.com
>
Signed-off-by:
Alexandre Julliard
<
julliard@winehq.org
>
parent
ab4edfd8
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
83 additions
and
4 deletions
+83
-4
exception.c
dlls/ntdll/tests/exception.c
+76
-0
signal_x86_64.c
dlls/ntdll/unix/signal_x86_64.c
+7
-4
No files found.
dlls/ntdll/tests/exception.c
View file @
3f20261e
...
@@ -4916,6 +4916,81 @@ static void test_unwind_from_apc(void)
...
@@ -4916,6 +4916,81 @@ static void test_unwind_from_apc(void)
ok
(
pass
==
4
,
"Got unexpected pass %d.
\n
"
,
pass
);
ok
(
pass
==
4
,
"Got unexpected pass %d.
\n
"
,
pass
);
ok
(
test_unwind_apc_called
,
"Test user APC was not called.
\n
"
);
ok
(
test_unwind_apc_called
,
"Test user APC was not called.
\n
"
);
}
}
static
void
test_syscall_clobbered_regs
(
void
)
{
struct
regs
{
UINT64
rcx
;
};
static
const
BYTE
code
[]
=
{
0x48
,
0x8d
,
0x05
,
0x00
,
0x10
,
0x00
,
0x00
,
/* leaq 0x1000(%rip),%rax */
0x48
,
0x25
,
0x00
,
0xf0
,
0xff
,
0xff
,
/* andq $~0xfff,%rax */
0x48
,
0x83
,
0xe8
,
0x08
,
/* subq $8,%rax */
0x48
,
0x89
,
0x20
,
/* movq %rsp,0(%rax) */
0x48
,
0x89
,
0xc4
,
/* movq %rax,%rsp */
0x41
,
0x50
,
/* push %r8 */
0x53
,
0x55
,
0x57
,
0x56
,
0x41
,
0x54
,
0x41
,
0x55
,
0x41
,
0x56
,
0x41
,
0x57
,
/* push %rbx, %rbp, %rdi, %rsi, %r12, %r13, %r14, %r15 */
0x41
,
0xff
,
0xd1
,
/* callq *r9 */
0x41
,
0x5f
,
0x41
,
0x5e
,
0x41
,
0x5d
,
0x41
,
0x5c
,
0x5e
,
0x5f
,
0x5d
,
0x5b
,
/* pop %r15, %r14, %r13, %r12, %rsi, %rdi, %rbp, %rbx */
0x41
,
0x58
,
/* pop %r8 */
0x49
,
0x89
,
0x48
,
0x00
,
/* mov %rcx,(%r8) */
0x5c
,
/* pop %rsp */
0xc3
,
/* ret */
};
NTSTATUS
(
WINAPI
*
func
)(
void
*
arg1
,
void
*
arg2
,
struct
regs
*
,
void
*
call_addr
);
NTSTATUS
(
WINAPI
*
pNtCancelTimer
)(
HANDLE
,
BOOLEAN
*
);
HMODULE
hntdll
=
GetModuleHandleA
(
"ntdll.dll"
);
struct
regs
regs
;
CONTEXT
context
;
NTSTATUS
status
;
pNtCancelTimer
=
(
void
*
)
GetProcAddress
(
hntdll
,
"NtCancelTimer"
);
ok
(
!!
pNtCancelTimer
,
"NtCancelTimer not found.
\n
"
);
memcpy
(
code_mem
,
code
,
sizeof
(
code
));
func
=
code_mem
;
memset
(
&
regs
,
0
,
sizeof
(
regs
));
status
=
func
((
HANDLE
)
0xdeadbeef
,
NULL
,
&
regs
,
pNtCancelTimer
);
ok
(
status
==
STATUS_INVALID_HANDLE
,
"Got unexpected status %#x.
\n
"
,
status
);
/* After the syscall instruction rcx contains the address of the instruction next after syscall. */
ok
((
BYTE
*
)
regs
.
rcx
>
(
BYTE
*
)
pNtCancelTimer
&&
(
BYTE
*
)
regs
.
rcx
<
(
BYTE
*
)
pNtCancelTimer
+
0x20
,
"Got unexpected rcx %s, pNtCancelTimer %p.
\n
"
,
wine_dbgstr_longlong
(
regs
.
rcx
),
pNtCancelTimer
);
status
=
func
((
HANDLE
)
0xdeadbeef
,
(
BOOLEAN
*
)
0xdeadbeef
,
&
regs
,
pNtCancelTimer
);
ok
(
status
==
STATUS_ACCESS_VIOLATION
,
"Got unexpected status %#x.
\n
"
,
status
);
ok
((
BYTE
*
)
regs
.
rcx
>
(
BYTE
*
)
pNtCancelTimer
&&
(
BYTE
*
)
regs
.
rcx
<
(
BYTE
*
)
pNtCancelTimer
+
0x20
,
"Got unexpected rcx %s, pNtCancelTimer %p.
\n
"
,
wine_dbgstr_longlong
(
regs
.
rcx
),
pNtCancelTimer
);
context
.
ContextFlags
=
CONTEXT_CONTROL
;
status
=
func
(
GetCurrentThread
(),
&
context
,
&
regs
,
pNtGetContextThread
);
ok
(
status
==
STATUS_SUCCESS
,
"Got unexpected status %#x.
\n
"
,
status
);
ok
((
BYTE
*
)
regs
.
rcx
>
(
BYTE
*
)
pNtGetContextThread
&&
(
BYTE
*
)
regs
.
rcx
<
(
BYTE
*
)
pNtGetContextThread
+
0x20
,
"Got unexpected rcx %s, pNtGetContextThread %p.
\n
"
,
wine_dbgstr_longlong
(
regs
.
rcx
),
pNtGetContextThread
);
status
=
func
(
GetCurrentThread
(),
&
context
,
&
regs
,
pNtSetContextThread
);
ok
(
status
==
STATUS_SUCCESS
,
"Got unexpected status %#x.
\n
"
,
status
);
ok
((
BYTE
*
)
regs
.
rcx
>
(
BYTE
*
)
pNtGetContextThread
&&
(
BYTE
*
)
regs
.
rcx
<
(
BYTE
*
)
pNtGetContextThread
+
0x20
,
"Got unexpected rcx %s, pNtGetContextThread %p.
\n
"
,
wine_dbgstr_longlong
(
regs
.
rcx
),
pNtGetContextThread
);
context
.
ContextFlags
=
CONTEXT_INTEGER
;
status
=
func
(
GetCurrentThread
(),
&
context
,
&
regs
,
pNtGetContextThread
);
ok
(
status
==
STATUS_SUCCESS
,
"Got unexpected status %#x.
\n
"
,
status
);
ok
((
BYTE
*
)
regs
.
rcx
>
(
BYTE
*
)
pNtGetContextThread
&&
(
BYTE
*
)
regs
.
rcx
<
(
BYTE
*
)
pNtGetContextThread
+
0x20
,
"Got unexpected rcx %s, pNtGetContextThread %p.
\n
"
,
wine_dbgstr_longlong
(
regs
.
rcx
),
pNtGetContextThread
);
status
=
func
(
GetCurrentThread
(),
&
context
,
&
regs
,
pNtSetContextThread
);
ok
(
status
==
STATUS_SUCCESS
,
"Got unexpected status %#x.
\n
"
,
status
);
ok
((
BYTE
*
)
regs
.
rcx
>
(
BYTE
*
)
pNtSetContextThread
&&
(
BYTE
*
)
regs
.
rcx
<
(
BYTE
*
)
pNtSetContextThread
+
0x20
,
"Got unexpected rcx %s, pNtSetContextThread %p.
\n
"
,
wine_dbgstr_longlong
(
regs
.
rcx
),
pNtSetContextThread
);
}
#elif defined(__arm__)
#elif defined(__arm__)
#define UNW_FLAG_NHANDLER 0
#define UNW_FLAG_NHANDLER 0
...
@@ -10657,6 +10732,7 @@ START_TEST(exception)
...
@@ -10657,6 +10732,7 @@ START_TEST(exception)
test_extended_context
();
test_extended_context
();
test_copy_context
();
test_copy_context
();
test_unwind_from_apc
();
test_unwind_from_apc
();
test_syscall_clobbered_regs
();
#elif defined(__aarch64__)
#elif defined(__aarch64__)
...
...
dlls/ntdll/unix/signal_x86_64.c
View file @
3f20261e
...
@@ -3226,18 +3226,21 @@ __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher,
...
@@ -3226,18 +3226,21 @@ __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher,
"testl $0x3,%edx
\n\t
"
/* CONTEXT_CONTROL | CONTEXT_INTEGER */
"testl $0x3,%edx
\n\t
"
/* CONTEXT_CONTROL | CONTEXT_INTEGER */
"jnz 1f
\n\t
"
"jnz 1f
\n\t
"
"movq 0x88(%rcx),%rsp
\n\t
"
"movq 0x88(%rcx),%rsp
\n\t
"
"jmpq *0x70(%rcx)
\n
"
/* frame->rip */
"movq 0x70(%rcx),%rcx
\n\t
"
/* frame->rip */
"jmpq *%rcx
\n\t
"
"1:
\t
leaq 0x70(%rcx),%rsp
\n\t
"
"1:
\t
leaq 0x70(%rcx),%rsp
\n\t
"
"testl $0x2,%edx
\n\t
"
/* CONTEXT_INTEGER */
"testl $0x2,%edx
\n\t
"
/* CONTEXT_INTEGER */
"jz 1f
\n\t
"
"jnz 1f
\n\t
"
"movq 0x00(%rcx),%rax
\n\t
"
"movq (%rsp),%rcx
\n\t
"
/* frame->rip */
"iretq
\n
"
"1:
\t
movq 0x00(%rcx),%rax
\n\t
"
"movq 0x18(%rcx),%rdx
\n\t
"
"movq 0x18(%rcx),%rdx
\n\t
"
"movq 0x30(%rcx),%r8
\n\t
"
"movq 0x30(%rcx),%r8
\n\t
"
"movq 0x38(%rcx),%r9
\n\t
"
"movq 0x38(%rcx),%r9
\n\t
"
"movq 0x40(%rcx),%r10
\n\t
"
"movq 0x40(%rcx),%r10
\n\t
"
"movq 0x48(%rcx),%r11
\n\t
"
"movq 0x48(%rcx),%r11
\n\t
"
"movq 0x10(%rcx),%rcx
\n
"
"movq 0x10(%rcx),%rcx
\n
"
"
1:
\t
iretq
\n
"
"iretq
\n
"
"5:
\t
movl $0xc000000d,%edx
\n\t
"
/* STATUS_INVALID_PARAMETER */
"5:
\t
movl $0xc000000d,%edx
\n\t
"
/* STATUS_INVALID_PARAMETER */
"movq %rsp,%rcx
\n
"
"movq %rsp,%rcx
\n
"
__ASM_NAME
(
"__wine_syscall_dispatcher_return"
)
":
\n\t
"
__ASM_NAME
(
"__wine_syscall_dispatcher_return"
)
":
\n\t
"
...
...
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