/* * 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 <dlfcn.h> #include <iostream> #include "base/casts.h" #include "base/macros.h" #include "jni/java_vm_ext.h" #include "jni/jni_env_ext.h" #include "thread-current-inl.h" namespace art { namespace { static volatile std::atomic<bool> vm_was_shutdown(false); static const int kThreadCount = 4; static std::atomic<int> barrier_count(kThreadCount + 1); static void JniThreadBarrierWait() { barrier_count--; while (barrier_count.load() != 0) { usleep(1000); } } extern "C" JNIEXPORT void JNICALL Java_Main_waitAndCallIntoJniEnv(JNIEnv* env, jclass) { // Wait for all threads to enter JNI together. JniThreadBarrierWait(); // Wait until the runtime is shutdown. while (!vm_was_shutdown.load()) { usleep(1000); } std::cout << "About to call exception check\n"; env->ExceptionCheck(); LOG(ERROR) << "Should not be reached!"; } // NO_RETURN does not work with extern "C" for target builds. extern "C" JNIEXPORT void JNICALL Java_Main_destroyJavaVMAndExit(JNIEnv* env, jclass) { // Wait for all threads to enter JNI together. JniThreadBarrierWait(); // Fake up the managed stack so we can detach. Thread* const self = Thread::Current(); self->SetTopOfStack(nullptr); self->SetTopOfShadowStack(nullptr); JavaVM* vm = down_cast<JNIEnvExt*>(env)->GetVm(); vm->DetachCurrentThread(); // Open ourself again to make sure the native library does not get unloaded from // underneath us due to DestroyJavaVM. b/28406866 void* handle = dlopen(kIsDebugBuild ? "libarttestd.so" : "libarttest.so", RTLD_NOW); CHECK(handle != nullptr); vm->DestroyJavaVM(); vm_was_shutdown.store(true); // Give threads some time to get stuck in ExceptionCheck. usleep(1000000); if (env != nullptr) { // Use env != nullptr to trick noreturn. exit(0); } } } // namespace } // namespace art