113 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			113 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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
 |