/* Copyright (c) 2008, Avian Contributors 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. */ #include "types.h" .text #ifdef __x86_64__ .globl vmNativeCall vmNativeCall: pushq %rbp // %rdi aka 0(%rbp): function // %rsi aka 8(%rbp): stack // %rdx aka 16(%rbp): stackSize // %rcx aka 24(%rbp): gprTable // %r8 aka 32(%rbp): sseTable // %r9 aka 40(%rbp): returnType // save our argument registers so we can clobber them pushq %r9 pushq %r8 pushq %rcx pushq %rdx pushq %rsi pushq %rdi movq %rsp,%rbp // reserve space for arguments passed via memory subq %rdx,%rsp // align to a 16 byte boundary andq $0xFFFFFFFFFFFFFFF0,%rsp // copy memory arguments into place movq $0,%rcx jmp test loop: movq %rcx,%rax movq %rcx,%rdx addq %rsp,%rdx addq 8(%rbp),%rax movq (%rax),%rax movq %rax,(%rdx) addq $8,%rcx test: cmpq 16(%rbp),%rcx jb loop // do we need to load the general-purpose registers? cmpq $0,24(%rbp) je sse // yes, we do movq 24(%rbp),%rax movq 0(%rax),%rdi movq 8(%rax),%rsi movq 16(%rax),%rdx movq 24(%rax),%rcx movq 32(%rax),%r8 movq 40(%rax),%r9 sse: // do we need to load the SSE registers? cmpq $0,32(%rbp) je call // yes, we do movq 32(%rbp),%rax movq 0(%rax),%xmm0 movq 8(%rax),%xmm1 movq 16(%rax),%xmm2 movq 24(%rax),%xmm3 movq 32(%rax),%xmm4 movq 40(%rax),%xmm5 movq 48(%rax),%xmm6 movq 64(%rax),%xmm7 call: call *0(%rbp) // handle return value based on expected type movq 40(%rbp),%rcx void: cmpq $VOID_TYPE,%rcx jne float jmp exit float: cmpq $FLOAT_TYPE,%rcx je copy cmpq $DOUBLE_TYPE,%rcx jne exit copy: movq %xmm0,%rax exit: movq %rbp,%rsp // pop our argument registers addq $48,%rsp popq %rbp ret .globl vmJump vmJump: movq %rsi,%rbp movq %rdx,%rsp movq %rcx,%rbx jmp *%rdi #elif defined __i386__ # if defined __APPLE__ || defined __MINGW32__ .globl _vmNativeCall _vmNativeCall: # else .globl vmNativeCall vmNativeCall: # endif pushl %ebp movl %esp,%ebp // 8(%ebp): function // 12(%ebp): stack // 16(%ebp): stackSize // 20(%ebp): returnType // reserve space for arguments movl 16(%ebp),%ecx subl %ecx,%esp # ifdef __APPLE__ // align to a 16 byte boundary on Darwin andl $0xFFFFFFF0,%esp # endif // copy arguments into place movl $0,%ecx jmp test loop: movl %ecx,%eax movl %ecx,%edx addl %esp,%edx addl 12(%ebp),%eax movl (%eax),%eax movl %eax,(%edx) addl $4,%ecx test: cmpl 16(%ebp),%ecx jb loop // call function call *8(%ebp) // handle return value based on expected type movl 20(%ebp),%ecx void: cmpl $VOID_TYPE,%ecx jne int64 jmp exit int64: cmpl $INT64_TYPE,%ecx jne float jmp exit float: cmpl $FLOAT_TYPE,%ecx jne double fstps 8(%ebp) movl 8(%ebp),%eax jmp exit double: cmpl $DOUBLE_TYPE,%ecx jne exit fstpl 8(%ebp) movl 8(%ebp),%eax movl 12(%ebp),%edx exit: movl %ebp,%esp popl %ebp ret # if defined __APPLE__ || defined __MINGW32__ .globl _vmJump _vmJump: # else .globl vmJump vmJump: # endif movl 4(%esp),%eax movl 8(%esp),%ebp movl 16(%esp),%ebx movl 12(%esp),%esp jmp *%eax #else # error unsupported platform #endif