Commit 5277c754 authored by Alexandre Julliard's avatar Alexandre Julliard

ntdll: Implement a faster Unix call dispatcher on x86_64.

Inspired by a patch from Rémi Bernon.
parent 911789e7
...@@ -2190,7 +2190,7 @@ static void start_main_thread(void) ...@@ -2190,7 +2190,7 @@ static void start_main_thread(void)
if (main_image_info.Machine != current_machine) load_wow64_ntdll( main_image_info.Machine ); if (main_image_info.Machine != current_machine) load_wow64_ntdll( main_image_info.Machine );
load_apiset_dll(); load_apiset_dll();
ntdll_init_syscalls( 0, &syscall_table, p__wine_syscall_dispatcher ); ntdll_init_syscalls( 0, &syscall_table, p__wine_syscall_dispatcher );
#if defined(__i386__) #if defined(__i386__) || defined(__x86_64__)
*p__wine_unix_call_dispatcher = __wine_unix_call_dispatcher; *p__wine_unix_call_dispatcher = __wine_unix_call_dispatcher;
#endif #endif
server_init_process_done(); server_init_process_done();
......
...@@ -1921,12 +1921,24 @@ static BOOL handle_syscall_fault( ucontext_t *sigcontext, EXCEPTION_RECORD *rec, ...@@ -1921,12 +1921,24 @@ static BOOL handle_syscall_fault( ucontext_t *sigcontext, EXCEPTION_RECORD *rec,
*/ */
static BOOL handle_syscall_trap( ucontext_t *sigcontext ) static BOOL handle_syscall_trap( ucontext_t *sigcontext )
{ {
extern void __wine_syscall_dispatcher_prolog_end(void);
struct syscall_frame *frame = amd64_thread_data()->syscall_frame; struct syscall_frame *frame = amd64_thread_data()->syscall_frame;
/* disallow single-stepping through a syscall */ /* disallow single-stepping through a syscall */
if ((void *)RIP_sig( sigcontext ) != __wine_syscall_dispatcher) return FALSE; if ((void *)RIP_sig( sigcontext ) == __wine_syscall_dispatcher)
{
extern void __wine_syscall_dispatcher_prolog_end(void);
RIP_sig( sigcontext ) = (ULONG64)__wine_syscall_dispatcher_prolog_end;
}
else if ((void *)RIP_sig( sigcontext ) == __wine_unix_call_dispatcher)
{
extern void __wine_unix_call_dispatcher_prolog_end(void);
RIP_sig( sigcontext ) = (ULONG64)__wine_unix_call_dispatcher_prolog_end;
R10_sig( sigcontext ) = RCX_sig( sigcontext );
}
else return FALSE;
TRACE_(seh)( "ignoring trap in syscall rip=%p eflags=%08x\n", TRACE_(seh)( "ignoring trap in syscall rip=%p eflags=%08x\n",
(void *)RIP_sig(sigcontext), (ULONG)EFL_sig(sigcontext) ); (void *)RIP_sig(sigcontext), (ULONG)EFL_sig(sigcontext) );
...@@ -1935,7 +1947,6 @@ static BOOL handle_syscall_trap( ucontext_t *sigcontext ) ...@@ -1935,7 +1947,6 @@ static BOOL handle_syscall_trap( ucontext_t *sigcontext )
frame->eflags = EFL_sig(sigcontext); frame->eflags = EFL_sig(sigcontext);
frame->restore_flags = CONTEXT_CONTROL; frame->restore_flags = CONTEXT_CONTROL;
RIP_sig( sigcontext ) = (ULONG64)__wine_syscall_dispatcher_prolog_end;
RCX_sig( sigcontext ) = (ULONG64)frame; RCX_sig( sigcontext ) = (ULONG64)frame;
RSP_sig( sigcontext ) += sizeof(ULONG64); RSP_sig( sigcontext ) += sizeof(ULONG64);
EFL_sig( sigcontext ) &= ~0x100; /* clear single-step flag */ EFL_sig( sigcontext ) &= ~0x100; /* clear single-step flag */
...@@ -2617,8 +2628,12 @@ __ASM_GLOBAL_FUNC( signal_exit_thread, ...@@ -2617,8 +2628,12 @@ __ASM_GLOBAL_FUNC( signal_exit_thread,
* __wine_syscall_dispatcher * __wine_syscall_dispatcher
*/ */
__ASM_GLOBAL_FUNC( __wine_syscall_dispatcher, __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher,
#ifdef __APPLE__
"movq %gs:0x30,%rcx\n\t" "movq %gs:0x30,%rcx\n\t"
"movq 0x328(%rcx),%rcx\n\t" /* amd64_thread_data()->syscall_frame */ "movq 0x328(%rcx),%rcx\n\t"
#else
"movq %gs:0x328,%rcx\n\t" /* amd64_thread_data()->syscall_frame */
#endif
"popq 0x70(%rcx)\n\t" /* frame->rip */ "popq 0x70(%rcx)\n\t" /* frame->rip */
__ASM_CFI(".cfi_adjust_cfa_offset -8\n\t") __ASM_CFI(".cfi_adjust_cfa_offset -8\n\t")
__ASM_CFI_REG_IS_AT2(rip, rcx, 0xf0,0x00) __ASM_CFI_REG_IS_AT2(rip, rcx, 0xf0,0x00)
...@@ -2733,10 +2748,11 @@ __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher, ...@@ -2733,10 +2748,11 @@ __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher,
"subq $0x20,%rsp\n\t" "subq $0x20,%rsp\n\t"
"movq (%rbx),%r10\n\t" /* table->ServiceTable */ "movq (%rbx),%r10\n\t" /* table->ServiceTable */
"callq *(%r10,%rax,8)\n\t" "callq *(%r10,%rax,8)\n\t"
"leaq -0x98(%rbp),%rcx\n" "leaq -0x98(%rbp),%rcx\n\t"
/* $rcx is now pointing to "frame" again */ /* $rcx is now pointing to "frame" again */
__ASM_CFI(".cfi_restore_state\n\t") __ASM_CFI(".cfi_restore_state\n")
"2:\tmovl 0x94(%rcx),%edx\n\t" /* frame->restore_flags */ ".L__wine_syscall_dispatcher_return:\n\t"
"movl 0x94(%rcx),%edx\n\t" /* frame->restore_flags */
#ifdef __linux__ #ifdef __linux__
"testl $12,%r14d\n\t" /* SYSCALL_HAVE_PTHREAD_TEB | SYSCALL_HAVE_WRFSGSBASE */ "testl $12,%r14d\n\t" /* SYSCALL_HAVE_PTHREAD_TEB | SYSCALL_HAVE_WRFSGSBASE */
"jz 1f\n\t" "jz 1f\n\t"
...@@ -2825,7 +2841,104 @@ __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher, ...@@ -2825,7 +2841,104 @@ __ASM_GLOBAL_FUNC( __wine_syscall_dispatcher,
__ASM_NAME("__wine_syscall_dispatcher_return") ":\n\t" __ASM_NAME("__wine_syscall_dispatcher_return") ":\n\t"
"movl 0xb0(%rcx),%r14d\n\t" /* frame->syscall_flags */ "movl 0xb0(%rcx),%r14d\n\t" /* frame->syscall_flags */
"movq %rdx,%rax\n\t" "movq %rdx,%rax\n\t"
"jmp 2b" ) "jmp .L__wine_syscall_dispatcher_return" )
/***********************************************************************
* __wine_unix_call_dispatcher
*/
__ASM_GLOBAL_FUNC( __wine_unix_call_dispatcher,
"movq %rcx,%r10\n\t"
#ifdef __APPLE__
"movq %gs:0x30,%rcx\n\t"
"movq 0x328(%rcx),%rcx\n\t"
#else
"movq %gs:0x328,%rcx\n\t" /* amd64_thread_data()->syscall_frame */
#endif
"popq 0x70(%rcx)\n\t" /* frame->rip */
__ASM_CFI(".cfi_adjust_cfa_offset -8\n\t")
__ASM_CFI_REG_IS_AT2(rip, rcx, 0xf0,0x00)
"movl $0,0x94(%rcx)\n\t" /* frame->restore_flags */
".globl " __ASM_NAME("__wine_unix_call_dispatcher_prolog_end") "\n"
__ASM_NAME("__wine_unix_call_dispatcher_prolog_end") ":\n\t"
"movq %rbx,0x08(%rcx)\n\t"
__ASM_CFI_REG_IS_AT1(rbx, rcx, 0x08)
"movq %rsi,0x20(%rcx)\n\t"
__ASM_CFI_REG_IS_AT1(rsi, rcx, 0x20)
"movq %rdi,0x28(%rcx)\n\t"
__ASM_CFI_REG_IS_AT1(rdi, rcx, 0x28)
"movq %r12,0x50(%rcx)\n\t"
__ASM_CFI_REG_IS_AT2(r12, rcx, 0xd0, 0x00)
"movq %r13,0x58(%rcx)\n\t"
__ASM_CFI_REG_IS_AT2(r13, rcx, 0xd8, 0x00)
"movq %r14,0x60(%rcx)\n\t"
__ASM_CFI_REG_IS_AT2(r14, rcx, 0xe0, 0x00)
"movq %r15,0x68(%rcx)\n\t"
__ASM_CFI_REG_IS_AT2(r15, rcx, 0xe8, 0x00)
"movq %rsp,0x88(%rcx)\n\t"
__ASM_CFI_CFA_IS_AT2(rcx, 0x88, 0x01)
__ASM_CFI_REG_IS_AT2(rsp, rcx, 0x88, 0x01)
"movq %rbp,0x98(%rcx)\n\t"
__ASM_CFI_REG_IS_AT2(rbp, rcx, 0x98, 0x01)
"movdqa %xmm6,0x1c0(%rcx)\n\t"
"movdqa %xmm7,0x1d0(%rcx)\n\t"
"movdqa %xmm8,0x1e0(%rcx)\n\t"
"movdqa %xmm9,0x1f0(%rcx)\n\t"
"movdqa %xmm10,0x200(%rcx)\n\t"
"movdqa %xmm11,0x210(%rcx)\n\t"
"movdqa %xmm12,0x220(%rcx)\n\t"
"movdqa %xmm13,0x230(%rcx)\n\t"
"movdqa %xmm14,0x240(%rcx)\n\t"
"movdqa %xmm15,0x250(%rcx)\n\t"
"movl 0xb0(%rcx),%r14d\n\t" /* frame->syscall_flags */
#ifdef __linux__
"testl $12,%r14d\n\t" /* SYSCALL_HAVE_PTHREAD_TEB | SYSCALL_HAVE_WRFSGSBASE */
"jz 2f\n\t"
"movw %fs,0x7e(%rcx)\n\t"
"movq %gs:0x330,%rsi\n\t" /* amd64_thread_data()->pthread_teb */
"testl $8,%r14d\n\t" /* SYSCALL_HAVE_WRFSGSBASE */
"jz 1f\n\t"
"wrfsbase %rsi\n\t"
"jmp 2f\n"
"1:\tmov $0x1002,%edi\n\t" /* ARCH_SET_FS */
"mov $158,%eax\n\t" /* SYS_arch_prctl */
"mov %rcx,%r9\n\t"
"syscall\n\t"
"mov %r9,%rcx\n\t"
"2:\n\t"
#endif
"movq %rcx,%rsp\n"
"movq %r8,%rdi\n\t" /* args */
"callq *(%r10,%rdx,8)\n\t"
"movq %rsp,%rcx\n"
"testl $0xffff,0x94(%rcx)\n\t" /* frame->restore_flags */
"jnz .L__wine_syscall_dispatcher_return\n\t"
#ifdef __linux__
"testl $12,%r14d\n\t" /* SYSCALL_HAVE_PTHREAD_TEB | SYSCALL_HAVE_WRFSGSBASE */
"jz 1f\n\t"
"movw 0x7e(%rcx),%fs\n"
"1:\n\t"
#endif
"movdqa 0x1c0(%rcx),%xmm6\n\t"
"movdqa 0x1d0(%rcx),%xmm7\n\t"
"movdqa 0x1e0(%rcx),%xmm8\n\t"
"movdqa 0x1f0(%rcx),%xmm9\n\t"
"movdqa 0x200(%rcx),%xmm10\n\t"
"movdqa 0x210(%rcx),%xmm11\n\t"
"movdqa 0x220(%rcx),%xmm12\n\t"
"movdqa 0x230(%rcx),%xmm13\n\t"
"movdqa 0x240(%rcx),%xmm14\n\t"
"movdqa 0x250(%rcx),%xmm15\n\t"
"movq 0x60(%rcx),%r14\n\t"
__ASM_CFI(".cfi_same_value r14\n\t")
"movq 0x28(%rcx),%rdi\n\t"
__ASM_CFI(".cfi_same_value rdi\n\t")
"movq 0x20(%rcx),%rsi\n\t"
__ASM_CFI(".cfi_same_value rsi\n\t")
"movq 0x88(%rcx),%rsp\n\t"
__ASM_CFI(".cfi_def_cfa rsp, 0\n\t")
__ASM_CFI(".cfi_same_value rsp\n\t")
"jmpq *0x70(%rcx)" ) /* frame->rip */
/*********************************************************************** /***********************************************************************
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment