/* * Copyright 2010, The Android Open Source Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "JavaClassJobjectV8.h" #if ENABLE(JAVA_BRIDGE) #include "JavaFieldJobjectV8.h" #include "JavaMethodJobject.h" using namespace JSC::Bindings; #if PLATFORM(ANDROID) const char kJavaScriptInterfaceAnnotation[] = "android/webkit/JavascriptInterface"; const char kIsAnnotationPresent[] = "isAnnotationPresent"; const char kGetMethods[] = "getMethods"; static jclass safeAnnotationClazz = NULL; JavaClassJobject::JavaClassJobject(jobject anInstance, bool requireAnnotation) : m_requireAnnotation(requireAnnotation) #else JavaClassJobject::JavaClassJobject(jobject anInstance) #endif { jobject aClass = callJNIMethod(anInstance, "getClass", "()Ljava/lang/Class;"); if (!aClass) { LOG_ERROR("unable to call getClass on instance %p", anInstance); return; } JNIEnv* env = getJNIEnv(); // Get the fields jarray fields = static_cast(callJNIMethod(aClass, "getFields", "()[Ljava/lang/reflect/Field;")); int numFields = env->GetArrayLength(fields); for (int i = 0; i < numFields; i++) { jobject aJField = env->GetObjectArrayElement(static_cast(fields), i); JavaField* aField = new JavaFieldJobject(env, aJField); // deleted in the JavaClass destructor m_fields.set(aField->name(), aField); env->DeleteLocalRef(aJField); } // Get the methods jarray methods = static_cast(callJNIMethod(aClass, "getMethods", "()[Ljava/lang/reflect/Method;")); int numMethods = env->GetArrayLength(methods); #if PLATFORM(ANDROID) jmethodID isAnnotationPresentMethodID = getAnnotationMethodID(env); if (!isAnnotationPresentMethodID) { LOG_ERROR("unable to find method %s on instance %p", kIsAnnotationPresent, anInstance); return; } #endif for (int i = 0; i < numMethods; i++) { jobject aJMethod = env->GetObjectArrayElement(static_cast(methods), i); #if PLATFORM(ANDROID) if (jsAccessAllowed(env, isAnnotationPresentMethodID, aJMethod)) { #endif JavaMethod* aMethod = new JavaMethodJobject(env, aJMethod); // deleted in the JavaClass destructor MethodList* methodList = m_methods.get(aMethod->name()); if (!methodList) { methodList = new MethodList(); m_methods.set(aMethod->name(), methodList); } methodList->append(aMethod); #if PLATFORM(ANDROID) } #endif env->DeleteLocalRef(aJMethod); } env->DeleteLocalRef(fields); env->DeleteLocalRef(methods); env->DeleteLocalRef(aClass); } JavaClassJobject::~JavaClassJobject() { deleteAllValues(m_fields); m_fields.clear(); MethodListMap::const_iterator end = m_methods.end(); for (MethodListMap::const_iterator it = m_methods.begin(); it != end; ++it) { const MethodList* methodList = it->second; deleteAllValues(*methodList); delete methodList; } m_methods.clear(); } #if PLATFORM(ANDROID) bool JavaClassJobject::jsAccessAllowed(JNIEnv* env, jmethodID mid, jobject aJMethod) { if (!m_requireAnnotation) return true; bool accessAllowed = env->CallBooleanMethod(aJMethod, mid, safeAnnotationClazz); if (env->ExceptionCheck()) { env->ExceptionDescribe(); env->ExceptionClear(); return false; } return accessAllowed; } jmethodID JavaClassJobject::getAnnotationMethodID(JNIEnv* env) { jclass methodClass = env->FindClass("java/lang/reflect/Method"); jmethodID mid = 0; if (methodClass) mid = env->GetMethodID(methodClass, kIsAnnotationPresent, "(Ljava/lang/Class;)Z"); if (!methodClass || !mid) { env->ExceptionDescribe(); env->ExceptionClear(); } env->DeleteLocalRef(methodClass); return mid; } bool JavaClassJobject::RegisterJavaClassJobject(JNIEnv* env) { safeAnnotationClazz = reinterpret_cast(env->NewGlobalRef(env->FindClass(kJavaScriptInterfaceAnnotation))); if (!safeAnnotationClazz) { LOG_ERROR("failed to register %s", kJavaScriptInterfaceAnnotation); return false; } return true; } #endif MethodList JavaClassJobject::methodsNamed(const char* name) const { MethodList* methodList = m_methods.get(name); if (methodList) return *methodList; return MethodList(); } JavaField* JavaClassJobject::fieldNamed(const char* name) const { return m_fields.get(name); } #endif // ENABLE(JAVA_BRIDGE)