/*
* 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