2014-04-21 02:14:48 +00:00
|
|
|
/* Copyright (c) 2008-2014, Avian Contributors
|
2009-12-03 02:08:29 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2009-07-26 02:48:36 +00:00
|
|
|
#ifdef __x86_64__
|
|
|
|
|
|
|
|
#define CONTINUATION_NEXT 8
|
|
|
|
#define CONTINUATION_ADDRESS 32
|
|
|
|
#define CONTINUATION_RETURN_ADDRESS_OFFSET 40
|
|
|
|
#define CONTINUATION_FRAME_POINTER_OFFSET 48
|
|
|
|
#define CONTINUATION_LENGTH 56
|
|
|
|
#define CONTINUATION_BODY 64
|
|
|
|
|
|
|
|
// call the next continuation, if any
|
2014-07-01 16:18:45 +00:00
|
|
|
movq TARGET_THREAD_CONTINUATION(%rbx),%rcx
|
2009-07-26 02:48:36 +00:00
|
|
|
cmpq $0,%rcx
|
|
|
|
je LOCAL(vmInvoke_exit)
|
|
|
|
|
|
|
|
// allocate a frame of size (continuation.length * BYTES_PER_WORD)
|
|
|
|
// + CALLEE_SAVED_REGISTER_FOOTPRINT
|
|
|
|
movq CONTINUATION_LENGTH(%rcx),%rsi
|
|
|
|
shlq $3,%rsi
|
|
|
|
subq %rsi,%rsp
|
|
|
|
subq $CALLEE_SAVED_REGISTER_FOOTPRINT,%rsp
|
|
|
|
|
|
|
|
// copy the continuation body into the frame
|
|
|
|
leaq CONTINUATION_BODY(%rcx),%rdi
|
|
|
|
|
|
|
|
movq $0,%r9
|
|
|
|
jmp LOCAL(vmInvoke_continuationTest)
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_continuationLoop):
|
|
|
|
movq (%rdi,%r9,1),%r8
|
|
|
|
movq %r8,(%rsp,%r9,1)
|
|
|
|
addq $8,%r9
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_continuationTest):
|
|
|
|
cmpq %rsi,%r9
|
|
|
|
jb LOCAL(vmInvoke_continuationLoop)
|
|
|
|
|
|
|
|
// set the return address to vmInvoke_returnAddress
|
|
|
|
movq CONTINUATION_RETURN_ADDRESS_OFFSET(%rcx),%rdi
|
2009-12-03 19:46:29 +00:00
|
|
|
#if defined __MINGW32__ || defined __CYGWIN32__
|
|
|
|
leaq GLOBAL(vmInvoke_returnAddress)(%rip),%r10
|
|
|
|
#else
|
2009-12-03 06:15:27 +00:00
|
|
|
movq GLOBAL(vmInvoke_returnAddress)@GOTPCREL(%rip),%r10
|
2009-12-03 19:46:29 +00:00
|
|
|
#endif
|
2009-07-26 02:48:36 +00:00
|
|
|
movq %r10,(%rsp,%rdi,1)
|
|
|
|
|
2011-01-27 18:54:41 +00:00
|
|
|
#ifdef AVIAN_USE_FRAME_POINTER
|
2009-07-26 02:48:36 +00:00
|
|
|
// save the current base pointer in the frame and update it
|
|
|
|
movq CONTINUATION_FRAME_POINTER_OFFSET(%rcx),%rdi
|
|
|
|
movq %rbp,(%rsp,%rdi,1)
|
|
|
|
addq %rsp,%rdi
|
|
|
|
movq %rdi,%rbp
|
2011-01-27 18:54:41 +00:00
|
|
|
#endif
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
// consume the continuation
|
|
|
|
movq CONTINUATION_NEXT(%rcx),%rdi
|
2014-07-01 16:18:45 +00:00
|
|
|
movq %rdi,TARGET_THREAD_CONTINUATION(%rbx)
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
// call the continuation unless we're handling an exception
|
2014-07-01 16:18:45 +00:00
|
|
|
movq TARGET_THREAD_EXCEPTION(%rbx),%rsi
|
2009-07-26 02:48:36 +00:00
|
|
|
cmpq $0,%rsi
|
|
|
|
jne LOCAL(vmInvoke_handleException)
|
|
|
|
jmp *CONTINUATION_ADDRESS(%rcx)
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_handleException):
|
|
|
|
// we're handling an exception - call the exception handler instead
|
2014-07-01 16:18:45 +00:00
|
|
|
movq $0,TARGET_THREAD_EXCEPTION(%rbx)
|
|
|
|
movq TARGET_THREAD_EXCEPTIONSTACKADJUSTMENT(%rbx),%rdi
|
2009-07-26 02:48:36 +00:00
|
|
|
subq %rdi,%rsp
|
2014-07-01 16:18:45 +00:00
|
|
|
movq TARGET_THREAD_EXCEPTIONOFFSET(%rbx),%rdi
|
2009-07-26 02:48:36 +00:00
|
|
|
movq %rsi,(%rsp,%rdi,1)
|
|
|
|
|
2014-07-01 16:18:45 +00:00
|
|
|
jmp *TARGET_THREAD_EXCEPTIONHANDLER(%rbx)
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
LOCAL(vmInvoke_exit):
|
|
|
|
|
|
|
|
#elif defined __i386__
|
2012-07-10 23:22:49 +00:00
|
|
|
|
2009-07-26 02:48:36 +00:00
|
|
|
#define CONTINUATION_NEXT 4
|
|
|
|
#define CONTINUATION_ADDRESS 16
|
|
|
|
#define CONTINUATION_RETURN_ADDRESS_OFFSET 20
|
|
|
|
#define CONTINUATION_FRAME_POINTER_OFFSET 24
|
|
|
|
#define CONTINUATION_LENGTH 28
|
|
|
|
#define CONTINUATION_BODY 32
|
2012-07-10 23:22:49 +00:00
|
|
|
|
|
|
|
#ifdef AVIAN_USE_FRAME_POINTER
|
|
|
|
# define CONTINUATION_ALIGNMENT_PADDING 8
|
|
|
|
#else
|
|
|
|
# define CONTINUATION_ALIGNMENT_PADDING 12
|
|
|
|
#endif
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
// call the next continuation, if any
|
2014-07-01 16:18:45 +00:00
|
|
|
movl TARGET_THREAD_CONTINUATION(%ebx),%ecx
|
2009-07-26 02:48:36 +00:00
|
|
|
cmpl $0,%ecx
|
|
|
|
je LOCAL(vmInvoke_exit)
|
|
|
|
|
2009-12-03 06:15:27 +00:00
|
|
|
// allocate a frame of size (continuation.length * BYTES_PER_WORD),
|
|
|
|
// plus stack alignment padding
|
2009-07-26 02:48:36 +00:00
|
|
|
movl CONTINUATION_LENGTH(%ecx),%esi
|
|
|
|
shll $2,%esi
|
2012-07-10 23:22:49 +00:00
|
|
|
leal CONTINUATION_ALIGNMENT_PADDING(%esi),%esi
|
2009-07-26 02:48:36 +00:00
|
|
|
subl %esi,%esp
|
|
|
|
|
|
|
|
// copy the continuation body into the frame
|
|
|
|
leal CONTINUATION_BODY(%ecx),%edi
|
|
|
|
|
|
|
|
push %eax
|
|
|
|
push %edx
|
|
|
|
|
|
|
|
movl $0,%edx
|
|
|
|
jmp LOCAL(vmInvoke_continuationTest)
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_continuationLoop):
|
|
|
|
movl (%edi,%edx,1),%eax
|
|
|
|
movl %eax,8(%esp,%edx,1)
|
|
|
|
addl $4,%edx
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_continuationTest):
|
|
|
|
cmpl %esi,%edx
|
|
|
|
jb LOCAL(vmInvoke_continuationLoop)
|
|
|
|
|
|
|
|
pop %edx
|
|
|
|
pop %eax
|
|
|
|
|
|
|
|
// set the return address to vmInvoke_returnAddress
|
|
|
|
movl CONTINUATION_RETURN_ADDRESS_OFFSET(%ecx),%edi
|
2009-12-03 02:37:22 +00:00
|
|
|
#if defined __MINGW32__ || defined __CYGWIN32__
|
2010-06-25 01:12:15 +00:00
|
|
|
movl $GLOBAL(vmInvoke_returnAddress),%esi
|
2009-12-03 02:29:57 +00:00
|
|
|
#else
|
2009-07-26 02:48:36 +00:00
|
|
|
call LOCAL(getPC)
|
2009-12-03 06:15:27 +00:00
|
|
|
# if defined __APPLE__
|
|
|
|
LOCAL(vmInvoke_offset):
|
2010-06-25 01:12:15 +00:00
|
|
|
leal GLOBAL(vmInvoke_returnAddress)-LOCAL(vmInvoke_offset)(%esi),%esi
|
2009-12-03 06:15:27 +00:00
|
|
|
# else
|
2009-07-26 02:48:36 +00:00
|
|
|
addl $_GLOBAL_OFFSET_TABLE_,%esi
|
2010-06-25 01:12:15 +00:00
|
|
|
movl GLOBAL(vmInvoke_returnAddress)@GOT(%esi),%esi
|
2009-12-03 06:15:27 +00:00
|
|
|
# endif
|
2009-12-03 02:29:57 +00:00
|
|
|
#endif
|
2009-07-26 02:48:36 +00:00
|
|
|
movl %esi,(%esp,%edi,1)
|
|
|
|
|
2011-01-27 18:54:41 +00:00
|
|
|
#ifdef AVIAN_USE_FRAME_POINTER
|
2009-07-26 02:48:36 +00:00
|
|
|
// save the current base pointer in the frame and update it
|
|
|
|
movl CONTINUATION_FRAME_POINTER_OFFSET(%ecx),%edi
|
|
|
|
movl %ebp,(%esp,%edi,1)
|
|
|
|
addl %esp,%edi
|
|
|
|
movl %edi,%ebp
|
2011-01-27 18:54:41 +00:00
|
|
|
#endif
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
// consume the continuation
|
|
|
|
movl CONTINUATION_NEXT(%ecx),%edi
|
2014-07-01 16:18:45 +00:00
|
|
|
movl %edi,TARGET_THREAD_CONTINUATION(%ebx)
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
// call the continuation unless we're handling an exception
|
2014-07-01 16:18:45 +00:00
|
|
|
movl TARGET_THREAD_EXCEPTION(%ebx),%esi
|
2009-07-26 02:48:36 +00:00
|
|
|
cmpl $0,%esi
|
|
|
|
jne LOCAL(vmInvoke_handleException)
|
|
|
|
|
|
|
|
jmp *CONTINUATION_ADDRESS(%ecx)
|
|
|
|
|
|
|
|
LOCAL(vmInvoke_handleException):
|
|
|
|
// we're handling an exception - call the exception handler instead
|
2014-07-01 16:18:45 +00:00
|
|
|
movl $0,TARGET_THREAD_EXCEPTION(%ebx)
|
|
|
|
movl TARGET_THREAD_EXCEPTIONSTACKADJUSTMENT(%ebx),%edi
|
2009-07-26 02:48:36 +00:00
|
|
|
subl %edi,%esp
|
2014-07-01 16:18:45 +00:00
|
|
|
movl TARGET_THREAD_EXCEPTIONOFFSET(%ebx),%edi
|
2009-07-26 02:48:36 +00:00
|
|
|
movl %esi,(%esp,%edi,1)
|
|
|
|
|
2014-07-01 16:18:45 +00:00
|
|
|
jmp *TARGET_THREAD_EXCEPTIONHANDLER(%ebx)
|
2009-07-26 02:48:36 +00:00
|
|
|
|
|
|
|
LOCAL(vmInvoke_exit):
|
|
|
|
|
|
|
|
#else
|
|
|
|
# error unsupported architecture
|
|
|
|
#endif
|
|
|
|
|