#include "types.h" .text #ifdef __x86_64__ .globl amd64Call amd64Call: 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 // 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 #elif defined __i386__ #ifdef __APPLE__ .globl _cdeclCall _cdeclCall: #else .globl cdeclCall cdeclCall: #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 #ifdef __APPLE__ // align to a 16 byte boundary on Darwin addl $15,%ecx andl $0xFFFFFFF0,%ecx addl $8,%ecx #endif subl %ecx,%esp // 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 int32 double: cmpl $DOUBLE_TYPE,%ecx jne int32 fstpl 8(%ebp) movl 8(%ebp),%eax movl 12(%ebp),%edx jmp exit int32: movl $0,%edx exit: movl %ebp,%esp popl %ebp ret #else # error unsupported platform #endif