1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2016 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker 17*795d594fSAndroid Build Coastguard Worker #include <dlfcn.h> 18*795d594fSAndroid Build Coastguard Worker #include <iostream> 19*795d594fSAndroid Build Coastguard Worker 20*795d594fSAndroid Build Coastguard Worker #include "base/casts.h" 21*795d594fSAndroid Build Coastguard Worker #include "base/macros.h" 22*795d594fSAndroid Build Coastguard Worker #include "jni/java_vm_ext.h" 23*795d594fSAndroid Build Coastguard Worker #include "jni/jni_env_ext.h" 24*795d594fSAndroid Build Coastguard Worker #include "thread-current-inl.h" 25*795d594fSAndroid Build Coastguard Worker 26*795d594fSAndroid Build Coastguard Worker namespace art { 27*795d594fSAndroid Build Coastguard Worker namespace { 28*795d594fSAndroid Build Coastguard Worker 29*795d594fSAndroid Build Coastguard Worker static volatile std::atomic<bool> vm_was_shutdown(false); 30*795d594fSAndroid Build Coastguard Worker static const int kThreadCount = 4; 31*795d594fSAndroid Build Coastguard Worker 32*795d594fSAndroid Build Coastguard Worker static std::atomic<int> barrier_count(kThreadCount + 1); 33*795d594fSAndroid Build Coastguard Worker JniThreadBarrierWait()34*795d594fSAndroid Build Coastguard Workerstatic void JniThreadBarrierWait() { 35*795d594fSAndroid Build Coastguard Worker barrier_count--; 36*795d594fSAndroid Build Coastguard Worker while (barrier_count.load() != 0) { 37*795d594fSAndroid Build Coastguard Worker usleep(1000); 38*795d594fSAndroid Build Coastguard Worker } 39*795d594fSAndroid Build Coastguard Worker } 40*795d594fSAndroid Build Coastguard Worker Java_Main_waitAndCallIntoJniEnv(JNIEnv * env,jclass)41*795d594fSAndroid Build Coastguard Workerextern "C" JNIEXPORT void JNICALL Java_Main_waitAndCallIntoJniEnv(JNIEnv* env, jclass) { 42*795d594fSAndroid Build Coastguard Worker // Wait for all threads to enter JNI together. 43*795d594fSAndroid Build Coastguard Worker JniThreadBarrierWait(); 44*795d594fSAndroid Build Coastguard Worker // Wait until the runtime is shutdown. 45*795d594fSAndroid Build Coastguard Worker while (!vm_was_shutdown.load()) { 46*795d594fSAndroid Build Coastguard Worker usleep(1000); 47*795d594fSAndroid Build Coastguard Worker } 48*795d594fSAndroid Build Coastguard Worker std::cout << "About to call exception check\n"; 49*795d594fSAndroid Build Coastguard Worker env->ExceptionCheck(); 50*795d594fSAndroid Build Coastguard Worker LOG(ERROR) << "Should not be reached!"; 51*795d594fSAndroid Build Coastguard Worker } 52*795d594fSAndroid Build Coastguard Worker 53*795d594fSAndroid Build Coastguard Worker // NO_RETURN does not work with extern "C" for target builds. Java_Main_destroyJavaVMAndExit(JNIEnv * env,jclass)54*795d594fSAndroid Build Coastguard Workerextern "C" JNIEXPORT void JNICALL Java_Main_destroyJavaVMAndExit(JNIEnv* env, jclass) { 55*795d594fSAndroid Build Coastguard Worker // Wait for all threads to enter JNI together. 56*795d594fSAndroid Build Coastguard Worker JniThreadBarrierWait(); 57*795d594fSAndroid Build Coastguard Worker // Fake up the managed stack so we can detach. 58*795d594fSAndroid Build Coastguard Worker Thread* const self = Thread::Current(); 59*795d594fSAndroid Build Coastguard Worker self->SetTopOfStack(nullptr); 60*795d594fSAndroid Build Coastguard Worker self->SetTopOfShadowStack(nullptr); 61*795d594fSAndroid Build Coastguard Worker JavaVM* vm = down_cast<JNIEnvExt*>(env)->GetVm(); 62*795d594fSAndroid Build Coastguard Worker vm->DetachCurrentThread(); 63*795d594fSAndroid Build Coastguard Worker // Open ourself again to make sure the native library does not get unloaded from 64*795d594fSAndroid Build Coastguard Worker // underneath us due to DestroyJavaVM. b/28406866 65*795d594fSAndroid Build Coastguard Worker void* handle = dlopen(kIsDebugBuild ? "libarttestd.so" : "libarttest.so", RTLD_NOW); 66*795d594fSAndroid Build Coastguard Worker CHECK(handle != nullptr); 67*795d594fSAndroid Build Coastguard Worker vm->DestroyJavaVM(); 68*795d594fSAndroid Build Coastguard Worker vm_was_shutdown.store(true); 69*795d594fSAndroid Build Coastguard Worker // Give threads some time to get stuck in ExceptionCheck. 70*795d594fSAndroid Build Coastguard Worker usleep(1000000); 71*795d594fSAndroid Build Coastguard Worker if (env != nullptr) { 72*795d594fSAndroid Build Coastguard Worker // Use env != nullptr to trick noreturn. 73*795d594fSAndroid Build Coastguard Worker exit(0); 74*795d594fSAndroid Build Coastguard Worker } 75*795d594fSAndroid Build Coastguard Worker } 76*795d594fSAndroid Build Coastguard Worker 77*795d594fSAndroid Build Coastguard Worker } // namespace 78*795d594fSAndroid Build Coastguard Worker } // namespace art 79