| /* | 
 |  * Copyright (C) 2016 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. | 
 |  */ | 
 |  | 
 | #include "java_lang_reflect_Parameter.h" | 
 |  | 
 | #include "android-base/stringprintf.h" | 
 | #include "nativehelper/jni_macros.h" | 
 |  | 
 | #include "art_method-inl.h" | 
 | #include "base/utils.h" | 
 | #include "common_throws.h" | 
 | #include "dex/dex_file-inl.h" | 
 | #include "dex/dex_file_annotations.h" | 
 | #include "jni/jni_internal.h" | 
 | #include "native_util.h" | 
 | #include "scoped_fast_native_object_access-inl.h" | 
 |  | 
 | namespace art { | 
 |  | 
 | using android::base::StringPrintf; | 
 |  | 
 | static jobject Parameter_getAnnotationNative(JNIEnv* env, | 
 |                                              jclass, | 
 |                                              jobject javaMethod, | 
 |                                              jint parameterIndex, | 
 |                                              jclass annotationType) { | 
 |   ScopedFastNativeObjectAccess soa(env); | 
 |   if (UNLIKELY(javaMethod == nullptr)) { | 
 |     ThrowNullPointerException("javaMethod == null"); | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   ArtMethod* method = ArtMethod::FromReflectedMethod(soa, javaMethod); | 
 |   if (method->IsProxyMethod()) { | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   uint32_t parameter_count = method->GetParameterTypeList()->Size(); | 
 |   if (UNLIKELY(parameterIndex < 0 || static_cast<uint32_t>(parameterIndex) >= parameter_count)) { | 
 |     ThrowIllegalArgumentException( | 
 |         StringPrintf("Illegal parameterIndex %d for %s, parameter_count is %d", | 
 |                      parameterIndex, | 
 |                      method->PrettyMethod().c_str(), | 
 |                      parameter_count).c_str()); | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   uint32_t annotated_parameter_count = annotations::GetNumberOfAnnotatedMethodParameters(method); | 
 |   if (annotated_parameter_count == 0u) { | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   // For constructors with implicit arguments, we may need to adjust | 
 |   // annotation positions based on whether the implicit parameters are | 
 |   // expected to known and not just a compiler implementation detail. | 
 |   if (method->IsConstructor()) { | 
 |     StackHandleScope<1> hs(soa.Self()); | 
 |     // If declaring class is a local or an enum, do not pad parameter | 
 |     // annotations, as the implicit constructor parameters are an | 
 |     // implementation detail rather than required by JLS. | 
 |     Handle<mirror::Class> declaring_class = hs.NewHandle(method->GetDeclaringClass()); | 
 |     if (annotations::GetEnclosingMethod(declaring_class) == nullptr && !declaring_class->IsEnum()) { | 
 |       // Adjust the parameter index if the number of annotations does | 
 |       // not match the number of parameters. | 
 |       if (annotated_parameter_count <= parameter_count) { | 
 |         // Workaround for dexer not inserting annotation state for implicit parameters (b/68033708). | 
 |         uint32_t skip_count = parameter_count - annotated_parameter_count; | 
 |         DCHECK_GE(2u, skip_count); | 
 |         if (parameterIndex < static_cast<jint>(skip_count)) { | 
 |           return nullptr; | 
 |         } | 
 |         parameterIndex -= skip_count; | 
 |       } else { | 
 |         // Workaround for Jack erroneously inserting implicit parameter for local classes | 
 |         // (b/68033708). | 
 |         DCHECK_EQ(1u, annotated_parameter_count - parameter_count); | 
 |         parameterIndex += static_cast<jint>(annotated_parameter_count - parameter_count); | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   StackHandleScope<1> hs(soa.Self()); | 
 |   Handle<mirror::Class> klass(hs.NewHandle(soa.Decode<mirror::Class>(annotationType))); | 
 |   return soa.AddLocalReference<jobject>( | 
 |       annotations::GetAnnotationForMethodParameter(method, parameterIndex, klass)); | 
 | } | 
 |  | 
 | static JNINativeMethod gMethods[] = { | 
 |   FAST_NATIVE_METHOD( | 
 |       Parameter, | 
 |       getAnnotationNative, | 
 |       "(Ljava/lang/reflect/Executable;ILjava/lang/Class;)Ljava/lang/annotation/Annotation;"), | 
 | }; | 
 |  | 
 | void register_java_lang_reflect_Parameter(JNIEnv* env) { | 
 |   REGISTER_NATIVE_METHODS("java/lang/reflect/Parameter"); | 
 | } | 
 |  | 
 | }  // namespace art |