/* * Copyright (C) 2019 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 #include #include #include #include #include "class_linker.h" #include "class_root-inl.h" #include "jni.h" #include "jni/jni_internal.h" #include "mirror/class.h" #include "mirror/method_handle_impl.h" #include "mirror/object-inl.h" #include "mirror/object_array-alloc-inl.h" #include "reflection.h" #include "reflective_handle.h" #include "reflective_handle_scope-inl.h" #include "runtime.h" #include "scoped_thread_state_change-inl.h" #include "thread-inl.h" namespace art { namespace Test1985StructuralRedefineStackScope { extern "C" JNICALL jobject JNIEXPORT Java_Main_NativeFieldScopeCheck(JNIEnv* env, jclass, jobject field, jobject runnable) { jfieldID fid = env->FromReflectedField(field); jclass runnable_klass = env->FindClass("java/lang/Runnable"); jmethodID run = env->GetMethodID(runnable_klass, "run", "()V"); ScopedObjectAccess soa(Thread::Current()); StackHandleScope<4> hs(soa.Self()); StackArtFieldHandleScope<1> fhs(soa.Self()); StackArtFieldHandleScope<1> bhs(soa.Self()); ReflectiveHandle rf(fhs.NewHandle(jni::DecodeArtField(fid))); ReflectiveHandle bf(bhs.NewHandle(jni::DecodeArtField(fid))); ArtField* pre_ptr = rf.Get(); { ScopedThreadSuspension sts(soa.Self(), ThreadState::kNative); // Upcall to perform redefinition. env->CallVoidMethod(runnable, run); } Handle> mt_arr( hs.NewHandle(mirror::ObjectArray::Alloc( soa.Self(), Runtime::Current()->GetClassLinker()->FindArrayClass(soa.Self(), GetClassRoot()), 0))); Handle mt(hs.NewHandle(mirror::MethodType::Create( soa.Self(), hs.NewHandle(GetClassRoot()), mt_arr))); Handle mhi(hs.NewHandle( mirror::MethodHandleImpl::Create(soa.Self(), reinterpret_cast(rf.Get()), (rf->IsStatic() ? mirror::MethodHandle::Kind::kStaticGet : mirror::MethodHandle::Kind::kInstanceGet), mt))); CHECK_EQ(rf.Get(), bf.Get()) << "rf: " << rf->PrettyField() << " bf: " << bf->PrettyField(); // TODO Modify this to work for when run doesn't cause a change. CHECK_NE(pre_ptr, rf.Get()) << "pre_ptr: " << pre_ptr->PrettyField() << " rf: " << rf->PrettyField(); CHECK_EQ(fid, jni::EncodeArtField(rf)); return soa.AddLocalReference(mhi.Get()); } } // namespace Test1985StructuralRedefineStackScope } // namespace art