/* * Copyright (c) 2001, 2003, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ #include "jni.h" #include "jvm.h" #include #define NATIVE_METHOD(className, functionName, signature) \ { #functionName, signature, (void*)(className ## _ ## functionName) } static jclass noSuchMethodErrCl; static void ObjectStreamClass_initNative(JNIEnv *env) { jclass cl = (*env)->FindClass(env, "java/lang/NoSuchMethodError"); if (cl == NULL) { /* exception thrown */ return; } noSuchMethodErrCl = (*env)->NewGlobalRef(env, cl); } /* * Class: java_io_ObjectStreamClass * Method: hasStaticInitializer * Signature: (Ljava/lang/Class;Z)Z * * Returns true if the given class defines a ()V method; returns false * otherwise. */ JNIEXPORT jboolean JNICALL // Android-changed: Added inheritStaticInitializer parameter. // The inheritStaticInitializer parameter is set to JNI_TRUE when behavior compatible with // Android version 23 is required. ObjectStreamClass_hasStaticInitializer(JNIEnv *env, jclass this, jclass clazz, jboolean inheritStaticInitializer) { jclass superCl = NULL; jmethodID superClinitId = NULL; // Android-changed: Added comment to explain behavior. // Search for a static initializer method in this class and up through its // ancestor super classes, returning the id of the first method found. jmethodID clinitId = (*env)->GetStaticMethodID(env, clazz, "", "()V"); if (clinitId == NULL) { /* error thrown */ jthrowable th = (*env)->ExceptionOccurred(env); (*env)->ExceptionClear(env); /* normal return */ if (!(*env)->IsInstanceOf(env, th, noSuchMethodErrCl)) { (*env)->Throw(env, th); } return JNI_FALSE; } // BEGIN Android-changed: Exit immediately if version 23 behavior is required. // At this point the class or one of its ancestor super classes has a // static initializer. If inheritStaticInitializer is true then this method // can simply return true. Otherwise, it needs to check that the static // initializer was not inherited (b/29064453). if (inheritStaticInitializer == JNI_TRUE) { return JNI_TRUE; } // END Android-changed: Exit immediately if version 23 behavior is required. /* * Check superclass for static initializer as well--if the same method ID * is returned, then the static initializer is from a superclass. * Empirically, this step appears to be unnecessary in 1.4; however, the * JNI spec makes no guarantee that GetStaticMethodID will not return the * ID for a superclass initializer. */ if ((superCl = (*env)->GetSuperclass(env, clazz)) == NULL) { return JNI_TRUE; } superClinitId = (*env)->GetStaticMethodID(env, superCl, "", "()V"); if (superClinitId == NULL) { /* error thrown */ jthrowable th = (*env)->ExceptionOccurred(env); (*env)->ExceptionClear(env); /* normal return */ if (!(*env)->IsInstanceOf(env, th, noSuchMethodErrCl)) { (*env)->Throw(env, th); } return JNI_TRUE; } return (clinitId != superClinitId); } static JNINativeMethod gMethods[] = { NATIVE_METHOD(ObjectStreamClass, hasStaticInitializer, "(Ljava/lang/Class;Z)Z"), }; void register_java_io_ObjectStreamClass(JNIEnv* env) { jniRegisterNativeMethods(env, "java/io/ObjectStreamClass", gMethods, NELEM(gMethods)); ObjectStreamClass_initNative(env); }