2 * Copyright (C) 2003 Apple Computer, Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 #include <jni_utility.h>
30 #include <jni_runtime.h>
32 #include <runtime_array.h>
33 #include <runtime_object.h>
34 #include <runtime_root.h>
37 #define JS_LOG(formatAndArgs...) ((void)0)
39 #define JS_LOG(formatAndArgs...) { \
40 fprintf (stderr, "%s:%d -- %s: ", __FILE__, __LINE__, __FUNCTION__); \
41 fprintf(stderr, formatAndArgs); \
46 using namespace KJS::Bindings;
49 JavaParameter::JavaParameter (JNIEnv *env, jstring type)
51 _type = JavaString (env, type);
52 _JNIType = JNITypeFromClassName (_type.UTF8String());
55 JavaField::JavaField (JNIEnv *env, jobject aField)
58 jobject fieldType = callJNIObjectMethod (aField, "getType", "()Ljava/lang/Class;");
59 jstring fieldTypeName = (jstring)callJNIObjectMethod (fieldType, "getName", "()Ljava/lang/String;");
60 _type = JavaString(env, fieldTypeName);
61 _JNIType = JNITypeFromClassName (_type.UTF8String());
64 jstring fieldName = (jstring)callJNIObjectMethod (aField, "getName", "()Ljava/lang/String;");
65 _name = JavaString(env, fieldName);
67 _field = new JavaInstance(aField, 0);
70 KJS::Value JavaArray::convertJObjectToArray (KJS::ExecState *exec, jobject anObject, const char *type, const RootObject *r)
75 return KJS::Object(new RuntimeArrayImp(exec, new JavaArray ((jobject)anObject, type, r)));
78 jvalue JavaField::dispatchValueFromInstance(KJS::ExecState *exec, const JavaInstance *instance, const char *name, const char *sig, JNIType returnType) const
80 jobject jinstance = instance->javaInstance();
81 jobject fieldJInstance = _field->javaInstance();
82 JNIEnv *env = getJNIEnv();
85 bzero (&result, sizeof(jvalue));
86 jclass cls = env->GetObjectClass(fieldJInstance);
88 jmethodID mid = env->GetMethodID(cls, name, sig);
91 const RootObject *execContext = instance->executionContext();
92 if (execContext && execContext->nativeHandle()) {
93 Value exceptionDescription;
96 args[0].l = jinstance;
97 dispatchJNICall (execContext->nativeHandle(), fieldJInstance, false, returnType, mid, args, result, 0, exceptionDescription);
98 if (!exceptionDescription.isNull()) {
99 Object error = Error::create(exec, GeneralError, exceptionDescription.toString(exec).UTF8String().c_str());
100 exec->setException(error);
108 KJS::Value JavaField::valueFromInstance(KJS::ExecState *exec, const Instance *i) const
110 const JavaInstance *instance = static_cast<const JavaInstance *>(i);
112 Value jsresult = Undefined();
116 jvalue result = dispatchValueFromInstance (exec, instance, "get", "(Ljava/lang/Object;)Ljava/lang/Object;", object_type);
117 jobject anObject = result.l;
119 const char *arrayType = type();
120 if (arrayType[0] == '[') {
121 jsresult = JavaArray::convertJObjectToArray (exec, anObject, arrayType, instance->executionContext());
123 else if (anObject != 0){
124 jsresult = Instance::createRuntimeObject(Instance::JavaLanguage, anObject, instance->executionContext());
130 jvalue result = dispatchValueFromInstance (exec, instance, "getBoolean", "(Ljava/lang/Object;)Z", boolean_type);
131 jboolean value = result.z;
132 jsresult = KJS::Boolean((bool)value);
142 jvalue result = dispatchValueFromInstance (exec, instance, "getInt", "(Ljava/lang/Object;)I", int_type);
144 jsresult = Number((int)value);
152 jvalue result = dispatchValueFromInstance (exec, instance, "getDouble", "(Ljava/lang/Object;)D", double_type);
154 jsresult = Number((double)value);
161 JS_LOG ("getting %s = %s\n", name(), jsresult.toString(exec).ascii());
166 void JavaField::dispatchSetValueToInstance(KJS::ExecState *exec, const JavaInstance *instance, jvalue javaValue, const char *name, const char *sig) const
168 jobject jinstance = instance->javaInstance();
169 jobject fieldJInstance = _field->javaInstance();
170 JNIEnv *env = getJNIEnv();
172 jclass cls = env->GetObjectClass(fieldJInstance);
174 jmethodID mid = env->GetMethodID(cls, name, sig);
177 const RootObject *execContext = instance->executionContext();
178 if (execContext && execContext->nativeHandle()) {
179 Value exceptionDescription;
183 args[0].l = jinstance;
185 dispatchJNICall (execContext->nativeHandle(), fieldJInstance, false, void_type, mid, args, result, 0, exceptionDescription);
186 if (!exceptionDescription.isNull()) {
187 Object error = Error::create(exec, GeneralError, exceptionDescription.toString(exec).UTF8String().c_str());
188 exec->setException(error);
195 void JavaField::setValueToInstance(KJS::ExecState *exec, const Instance *i, const KJS::Value &aValue) const
197 const JavaInstance *instance = static_cast<const JavaInstance *>(i);
198 jvalue javaValue = convertValueToJValue (exec, aValue, _JNIType, type());
200 JS_LOG ("setting value %s to %s\n", name(), aValue.toString(exec).ascii());
204 dispatchSetValueToInstance (exec, instance, javaValue, "set", "(Ljava/lang/Object;Ljava/lang/Object;)V");
209 dispatchSetValueToInstance (exec, instance, javaValue, "setBoolean", "(Ljava/lang/Object;Z)V");
214 dispatchSetValueToInstance (exec, instance, javaValue, "setByte", "(Ljava/lang/Object;B)V");
219 dispatchSetValueToInstance (exec, instance, javaValue, "setChar", "(Ljava/lang/Object;C)V");
224 dispatchSetValueToInstance (exec, instance, javaValue, "setShort", "(Ljava/lang/Object;S)V");
229 dispatchSetValueToInstance (exec, instance, javaValue, "setInt", "(Ljava/lang/Object;I)V");
234 dispatchSetValueToInstance (exec, instance, javaValue, "setLong", "(Ljava/lang/Object;J)V");
239 dispatchSetValueToInstance (exec, instance, javaValue, "setFloat", "(Ljava/lang/Object;F)V");
244 dispatchSetValueToInstance (exec, instance, javaValue, "setDouble", "(Ljava/lang/Object;D)V");
252 JavaConstructor::JavaConstructor (JNIEnv *env, jobject aConstructor)
255 jarray jparameters = (jarray)callJNIObjectMethod (aConstructor, "getParameterTypes", "()[Ljava/lang/Class;");
256 _numParameters = env->GetArrayLength (jparameters);
257 _parameters = new JavaParameter[_numParameters];
260 for (i = 0; i < _numParameters; i++) {
261 jobject aParameter = env->GetObjectArrayElement ((jobjectArray)jparameters, i);
262 jstring parameterName = (jstring)callJNIObjectMethod (aParameter, "getName", "()Ljava/lang/String;");
263 _parameters[i] = JavaParameter(env, parameterName);
264 env->DeleteLocalRef (aParameter);
265 env->DeleteLocalRef (parameterName);
269 JavaMethod::JavaMethod (JNIEnv *env, jobject aMethod)
272 jobject returnType = callJNIObjectMethod (aMethod, "getReturnType", "()Ljava/lang/Class;");
273 jstring returnTypeName = (jstring)callJNIObjectMethod (returnType, "getName", "()Ljava/lang/String;");
274 _returnType =JavaString (env, returnTypeName);
275 _JNIReturnType = JNITypeFromClassName (_returnType.UTF8String());
276 env->DeleteLocalRef (returnType);
277 env->DeleteLocalRef (returnTypeName);
280 jstring methodName = (jstring)callJNIObjectMethod (aMethod, "getName", "()Ljava/lang/String;");
281 _name = JavaString (env, methodName);
282 env->DeleteLocalRef (methodName);
285 jarray jparameters = (jarray)callJNIObjectMethod (aMethod, "getParameterTypes", "()[Ljava/lang/Class;");
286 _numParameters = env->GetArrayLength (jparameters);
287 _parameters = new JavaParameter[_numParameters];
290 for (i = 0; i < _numParameters; i++) {
291 jobject aParameter = env->GetObjectArrayElement ((jobjectArray)jparameters, i);
292 jstring parameterName = (jstring)callJNIObjectMethod (aParameter, "getName", "()Ljava/lang/String;");
293 _parameters[i] = JavaParameter(env, parameterName);
294 env->DeleteLocalRef (aParameter);
295 env->DeleteLocalRef (parameterName);
297 env->DeleteLocalRef (jparameters);
303 jclass modifierClass = env->FindClass("java/lang/reflect/Modifier");
304 long modifiers = callJNIIntMethod (aMethod, "getModifiers", "()I");
305 _isStatic = (bool)callJNIStaticBooleanMethod (modifierClass, "isStatic", "(I)Z", modifiers);
308 // JNI method signatures use '/' between components of a class name, but
309 // we get '.' between components from the reflection API.
310 static void appendClassName (UString *aString, const char *className)
312 char *result, *cp = strdup(className);
321 aString->append(result);
326 const char *JavaMethod::signature() const
328 if (_signature == 0){
331 _signature = new UString("(");
332 for (i = 0; i < _numParameters; i++) {
333 JavaParameter *aParameter = static_cast<JavaParameter *>(parameterAt(i));
334 JNIType _JNIType = aParameter->getJNIType();
335 _signature->append(signatureFromPrimitiveType (_JNIType));
336 if (_JNIType == object_type) {
337 appendClassName (_signature, aParameter->type());
338 _signature->append(";");
341 _signature->append(")");
343 const char *returnType = _returnType.UTF8String();
344 if (returnType[0] == '[') {
345 appendClassName (_signature, returnType);
348 _signature->append(signatureFromPrimitiveType (_JNIReturnType));
349 if (_JNIReturnType == object_type) {
350 appendClassName (_signature, returnType);
351 _signature->append(";");
356 return _signature->ascii();
359 JNIType JavaMethod::JNIReturnType() const
361 return _JNIReturnType;
364 jmethodID JavaMethod::methodID (jobject obj) const
366 if (_methodID == 0) {
367 _methodID = getMethodID (obj, name(), signature());
373 JavaArray::JavaArray (jobject a, const char *t, const RootObject *r)
375 _array = new JObjectWrapper (a);
376 // Java array are fixed length, so we can cache length.
377 JNIEnv *env = getJNIEnv();
378 _length = env->GetArrayLength((jarray)_array->_instance);
383 JavaArray::~JavaArray ()
386 free ((void *)_type);
390 JavaArray::JavaArray (const JavaArray &other) : Array()
392 _array = other._array;
394 _type = strdup(other._type);
397 void JavaArray::setValueAt(KJS::ExecState *exec, unsigned int index, const KJS::Value &aValue) const
399 JNIEnv *env = getJNIEnv();
400 char *javaClassName = 0;
402 JNIType arrayType = JNITypeFromPrimitiveType(_type[1]);
403 if (_type[1] == 'L'){
404 // The type of the array will be something like:
405 // "[Ljava.lang.string;". This is guaranteed, so no need
406 // for extra sanity checks.
407 javaClassName = strdup(&_type[2]);
408 javaClassName[strchr(javaClassName, ';')-javaClassName] = 0;
410 jvalue aJValue = convertValueToJValue (exec, aValue, arrayType, javaClassName);
414 env->SetObjectArrayElement((jobjectArray)javaArray(), index, aJValue.l);
419 env->SetBooleanArrayRegion((jbooleanArray)javaArray(), index, 1, &aJValue.z);
424 env->SetByteArrayRegion((jbyteArray)javaArray(), index, 1, &aJValue.b);
429 env->SetCharArrayRegion((jcharArray)javaArray(), index, 1, &aJValue.c);
434 env->SetShortArrayRegion((jshortArray)javaArray(), index, 1, &aJValue.s);
439 env->SetIntArrayRegion((jintArray)javaArray(), index, 1, &aJValue.i);
444 env->SetLongArrayRegion((jlongArray)javaArray(), index, 1, &aJValue.j);
448 env->SetFloatArrayRegion((jfloatArray)javaArray(), index, 1, &aJValue.f);
453 env->SetDoubleArrayRegion((jdoubleArray)javaArray(), index, 1, &aJValue.d);
461 free ((void *)javaClassName);
465 KJS::Value JavaArray::valueAt(KJS::ExecState *exec, unsigned int index) const
467 JNIEnv *env = getJNIEnv();
468 JNIType arrayType = JNITypeFromPrimitiveType(_type[1]);
471 jobjectArray objectArray = (jobjectArray)javaArray();
473 anObject = env->GetObjectArrayElement(objectArray, index);
476 if (_type[1] == '[') {
477 return JavaArray::convertJObjectToArray (exec, anObject, _type+1, executionContext());
479 // or array of other object type?
480 return Instance::createRuntimeObject(Instance::JavaLanguage, anObject, executionContext());
484 jbooleanArray booleanArray = (jbooleanArray)javaArray();
486 env->GetBooleanArrayRegion(booleanArray, index, 1, &aBoolean);
487 return KJS::Boolean (aBoolean);
491 jbyteArray byteArray = (jbyteArray)javaArray();
493 env->GetByteArrayRegion(byteArray, index, 1, &aByte);
494 return Number (aByte);
498 jcharArray charArray = (jcharArray)javaArray();
500 env->GetCharArrayRegion(charArray, index, 1, &aChar);
501 return Number (aChar);
506 jshortArray shortArray = (jshortArray)javaArray();
508 env->GetShortArrayRegion(shortArray, index, 1, &aShort);
509 return Number (aShort);
513 jintArray intArray = (jintArray)javaArray();
515 env->GetIntArrayRegion(intArray, index, 1, &anInt);
516 return Number (anInt);
520 jlongArray longArray = (jlongArray)javaArray();
522 env->GetLongArrayRegion(longArray, index, 1, &aLong);
523 return Number ((long int)aLong);
527 jfloatArray floatArray = (jfloatArray)javaArray();
529 env->GetFloatArrayRegion(floatArray, index, 1, &aFloat);
530 return Number (aFloat);
534 jdoubleArray doubleArray = (jdoubleArray)javaArray();
536 env->GetDoubleArrayRegion(doubleArray, index, 1, &aDouble);
537 return Number (aDouble);
545 unsigned int JavaArray::getLength() const