/* Copyright (C) 2008 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* * File: CallABI.S * * Code: facilitates call to native code C and C++ routines. * */ /* * Function prototype: * * void dvmPlatformInvoke(void* pEnv, ClassObject* clazz, int argInfo, int argc, * const u4* argv, const char* signature, void* func, JValue* pReturn) * * The method we are calling has the form: * * return_type func(JNIEnv* pEnv, ClassObject* clazz, ...) * -or- * return_type func(JNIEnv* pEnv, Object* this, ...) * * We receive a collection of 32-bit values which correspond to arguments from * the interpreter (e.g. float occupies one, double occupies two). It's up to * us to convert these into local calling conventions. */ /* * On entry: * 4(%sp) JNIEnv (can be left alone) * 8(%esp) clazz (NULL for virtual method calls, non-NULL for static) * 12(%esp) arg info * 16(%esp) argc (number of 32-bit values in argv) * 20(%esp) argv * 24(%esp) short signature * 28(%esp) func * 32(%esp) pReturn * * For a virtual method call, the "this" reference is in argv[0]. * * argInfo (32-bit int) layout: * * SRRRHHHH HHHHHHHH HHHHHHHH HHHHHHHH * * S - if set, argInfo hints are invalid * R - return type enumeration (see jniInternal.h) * VOID -> 0 * FLOAT -> 1 * DOUBLE -> 2 * S8 -> 3 * S4 -> 4 * H - target-specific hints (see below for details) * * IA32 ABI JNI hint format * * ZZZZ ZZZZZZZZ AAAAAAAA AAAAAAAA * * Z - reserved * A - size of the variable argument block in 32-bit words */ .text .align 4 .global dvmPlatformInvoke .type dvmPlatformInvoke, %function dvmPlatformInvoke: CallABI_ENTER: /* * Save registers. */ movl %ebp, -4(%esp) movl %ebx, -8(%esp) # save %ebx movl %esi, -12(%esp) # save %esi movl %edi, -16(%esp) # save %edi lea (%esp), %ebp /* * Update and align (16 bytes) stack pointer */ lea -32(%esp), %esp /* * Check if argInfo is valid. Is always valid so should remove this check? */ movzwl 12(%ebp), %ecx # %ecx<- argsize in words movl 12(%ebp), %ebx # %ebx<- argInfo shl $2, %ecx # %ecx<- argsize in bytes subl %ecx, %esp # %esp<- expanded for arg region /* * Is the alignment right? */ #if 1 test $4, %esp jnz 1f subl $4, %esp 1: test $8, %esp jnz 1f subl $8, %esp 1: #endif movl 8(%ebp), %eax # %eax<- clazz cmpl $0, %eax # Check virtual or static movl 4(%ebp), %ecx # %ecx<- JNIEnv movl 20(%ebp), %esi # %esi<- argV jne 1f # Branch if static movl (%esi), %eax # get the this pointer addl $4, %esi # %esi<- update past this 1: movl %ecx, -8(%esp) # push JNIEnv as arg #1 movl %eax, -4(%esp) # push clazz or this as arg #2 lea -8(%esp), %esp /* * Copy arguments */ movzwl %bx, %ecx # %ecx<- %bx; argsize in words lea 8(%esp), %edi # %edi<- stack location for arguments cld rep movsl # move %ecx arguments to 8(%esp) call *28(%ebp) sarl $28, %ebx # %ebx<- SRRR (low 4 bits) je CallABI_EXIT # exit call cmpl $2, %ebx movl 32(%ebp), %ecx # %ecx<- return pointer je 2f # handle double return jl 1f # handle float return /* * If a native function returns a result smaller than 8-bytes * then higher bytes may contain garbage. * This code does type-checking based on size of return result. * We zero higher bytes instead of allowing the garbage to go through. */ cmpl $3,%ebx je S8 cmpl $4,%ebx je S4 cmpl $7,%ebx je S1 cmpl $6,%ebx jne S2 U2: movzwl %ax, %eax movl %eax, (%ecx) # save 32-bit return jmp CallABI_EXIT # exit call S1: movsbl %al, %eax movl %eax, (%ecx) # save 32-bit return jmp CallABI_EXIT # exit call S2: movswl %ax, %eax movl %eax, (%ecx) # save 32-bit return jmp CallABI_EXIT # exit call S4: cltd movl %eax, (%ecx) # save 32-bit return jmp CallABI_EXIT # exit call S8: movl %edx, 4(%ecx) # save 64-bit return movl %eax, (%ecx) # save 32-bit return jmp CallABI_EXIT # exit call 2: fstpl (%ecx) # save double return jmp CallABI_EXIT # exit call 1: fstps (%ecx) # save float return CallABI_EXIT: lea (%ebp), %esp movl -16(%ebp), %edi # restore %edi movl -12(%ebp), %esi # restore %esi movl -8(%ebp), %ebx # restore %ebx movl -4(%ebp), %ebp # restore caller base pointer ret # return