84 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			84 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
|  * 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 <cstdio>
 | |
| #include <memory>
 | |
| #include <mutex>
 | |
| #include <string>
 | |
| #include <vector>
 | |
| 
 | |
| #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<ArtField> rf(fhs.NewHandle(jni::DecodeArtField(fid)));
 | |
|   ReflectiveHandle<ArtField> 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<mirror::ObjectArray<mirror::Class>> mt_arr(
 | |
|       hs.NewHandle(mirror::ObjectArray<mirror::Class>::Alloc(
 | |
|           soa.Self(),
 | |
|           Runtime::Current()->GetClassLinker()->FindArrayClass(soa.Self(),
 | |
|                                                                GetClassRoot<mirror::Class>()),
 | |
|           0)));
 | |
|   Handle<mirror::MethodType> mt(hs.NewHandle(mirror::MethodType::Create(
 | |
|       soa.Self(), hs.NewHandle(GetClassRoot<mirror::Object>()), mt_arr)));
 | |
|   Handle<mirror::MethodHandleImpl> mhi(hs.NewHandle(
 | |
|       mirror::MethodHandleImpl::Create(soa.Self(),
 | |
|                                        reinterpret_cast<uintptr_t>(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<jobject>(mhi.Get());
 | |
| }
 | |
| 
 | |
| }  // namespace Test1985StructuralRedefineStackScope
 | |
| }  // namespace art
 |