/*
* 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.
*/
/*
* This uses the FFI (Foreign Function Interface) library to abstract away
* the system-dependent stuff. The FFI code is slower than a custom
* assembly version, but has the distinct advantage of having been
* written already for several platforms.
*/
#include "Dalvik.h"
#include "ffi.h"
/*
* Convert a signature type character to an FFI type.
*/
static ffi_type* getFfiType(char sigType)
{
switch (sigType) {
case 'V': return &ffi_type_void;
case 'Z': return &ffi_type_uint8;
case 'B': return &ffi_type_sint8;
case 'C': return &ffi_type_uint16;
case 'S': return &ffi_type_sint16;
case 'I': return &ffi_type_sint32;
case 'F': return &ffi_type_float;
case 'J': return &ffi_type_sint64;
case 'D': return &ffi_type_double;
case '[':
case 'L': return &ffi_type_pointer;
default:
ALOGE("bad ffitype 0x%02x", sigType);
dvmAbort();
return NULL;
}
}
/*
* Call "func" with the specified arguments.
*
* The second argument to JNI native functions is either the object (the
* "this" pointer) or, for static functions, a pointer to the class object.
* The Dalvik instructions will push "this" into argv[0], but it's up to
* us to insert the class object.
*
* Because there is no such thing in as a null "this" pointer, we use
* the non-NULL state of "clazz" to determine whether or not it's static.
*
* For maximum efficiency we should compute the CIF once and save it with
* the method. However, this requires storing the data with every native
* method. Since the goal is to have custom assembly versions of this
* on the platforms where performance matters, I'm recomputing the CIF on
* every call.
*/
void dvmPlatformInvoke(void* pEnv, ClassObject* clazz, int argInfo, int argc,
const u4* argv, const char* shorty, void* func, JValue* pReturn)
{
const int kMaxArgs = argc+2; /* +1 for env, maybe +1 for clazz */
ffi_cif cif;
ffi_type* types[kMaxArgs];
void* values[kMaxArgs];
ffi_type* retType;
char sigByte;
int srcArg, dstArg;
types[0] = &ffi_type_pointer;
values[0] = &pEnv;
types[1] = &ffi_type_pointer;
if (clazz != NULL) {
values[1] = &clazz;
srcArg = 0;
} else {
values[1] = (void*) argv++;
srcArg = 1;
}
dstArg = 2;
/*
* Scan the types out of the short signature. Use them to fill out the
* "types" array. Store the start address of the argument in "values".
*/
retType = getFfiType(*shorty);
while ((sigByte = *++shorty) != '\0') {
types[dstArg] = getFfiType(sigByte);
values[dstArg++] = (void*) argv++;
if (sigByte == 'D' || sigByte == 'J')
argv++;
}
/*
* Prep the CIF (Call InterFace object).
*/
if (ffi_prep_cif(&cif, FFI_DEFAULT_ABI, dstArg, retType, types) != FFI_OK) {
ALOGE("ffi_prep_cif failed");
dvmAbort();
}
ffi_call(&cif, FFI_FN(func), pReturn, values);
}