1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "jni/jni_internal.h"
18 
19 #include <pthread.h>
20 
21 #include "common_runtime_test.h"
22 #include "gc/heap.h"
23 #include "java_vm_ext.h"
24 #include "runtime.h"
25 
26 namespace art HIDDEN {
27 
28 class JavaVmExtTest : public CommonRuntimeTest {
29  protected:
JavaVmExtTest()30   JavaVmExtTest() {
31     this->use_boot_image_ = true;  // Make the Runtime creation cheaper.
32   }
33 
SetUp()34   void SetUp() override {
35     CommonRuntimeTest::SetUp();
36 
37     vm_ = Runtime::Current()->GetJavaVM();
38   }
39 
40 
TearDown()41   void TearDown() override {
42     CommonRuntimeTest::TearDown();
43   }
44 
45   JavaVMExt* vm_;
46 };
47 
TEST_F(JavaVmExtTest,JNI_GetDefaultJavaVMInitArgs)48 TEST_F(JavaVmExtTest, JNI_GetDefaultJavaVMInitArgs) {
49   jint err = JNI_GetDefaultJavaVMInitArgs(nullptr);
50   EXPECT_EQ(JNI_ERR, err);
51 }
52 
TEST_F(JavaVmExtTest,JNI_GetCreatedJavaVMs)53 TEST_F(JavaVmExtTest, JNI_GetCreatedJavaVMs) {
54   JavaVM* vms_buf[1];
55   jsize num_vms;
56   jint ok = JNI_GetCreatedJavaVMs(vms_buf, arraysize(vms_buf), &num_vms);
57   EXPECT_EQ(JNI_OK, ok);
58   EXPECT_EQ(1, num_vms);
59   EXPECT_EQ(vms_buf[0], vm_);
60 }
61 
62 static bool gSmallStack = false;
63 static bool gAsDaemon = false;
64 
attach_current_thread_callback(void * arg)65 static void* attach_current_thread_callback([[maybe_unused]] void* arg) {
66   JavaVM* vms_buf[1];
67   jsize num_vms;
68   JNIEnv* env;
69   jint ok = JNI_GetCreatedJavaVMs(vms_buf, arraysize(vms_buf), &num_vms);
70   EXPECT_EQ(JNI_OK, ok);
71   if (ok == JNI_OK) {
72     if (!gAsDaemon) {
73       ok = vms_buf[0]->AttachCurrentThread(&env, nullptr);
74     } else {
75       ok = vms_buf[0]->AttachCurrentThreadAsDaemon(&env, nullptr);
76     }
77     // TODO: Find a way to test with exact SMALL_STACK value, for which we would bail. The pthreads
78     //       spec says that the stack size argument is a lower bound, and bionic currently gives us
79     //       a chunk more on arm64.
80     if (!gSmallStack) {
81       EXPECT_EQ(JNI_OK, ok);
82     }
83     if (ok == JNI_OK) {
84       ok = vms_buf[0]->DetachCurrentThread();
85       EXPECT_EQ(JNI_OK, ok);
86     }
87   }
88   return nullptr;
89 }
90 
TEST_F(JavaVmExtTest,AttachCurrentThread)91 TEST_F(JavaVmExtTest, AttachCurrentThread) {
92   pthread_t pthread;
93   const char* reason = __PRETTY_FUNCTION__;
94   gSmallStack = false;
95   gAsDaemon = false;
96   CHECK_PTHREAD_CALL(pthread_create, (&pthread, nullptr, attach_current_thread_callback,
97       nullptr), reason);
98   void* ret_val;
99   CHECK_PTHREAD_CALL(pthread_join, (pthread, &ret_val), reason);
100   EXPECT_EQ(ret_val, nullptr);
101 }
102 
TEST_F(JavaVmExtTest,AttachCurrentThreadAsDaemon)103 TEST_F(JavaVmExtTest, AttachCurrentThreadAsDaemon) {
104   pthread_t pthread;
105   const char* reason = __PRETTY_FUNCTION__;
106   gSmallStack = false;
107   gAsDaemon = true;
108   CHECK_PTHREAD_CALL(pthread_create, (&pthread, nullptr, attach_current_thread_callback,
109       nullptr), reason);
110   void* ret_val;
111   CHECK_PTHREAD_CALL(pthread_join, (pthread, &ret_val), reason);
112   EXPECT_EQ(ret_val, nullptr);
113 }
114 
TEST_F(JavaVmExtTest,AttachCurrentThread_SmallStack)115 TEST_F(JavaVmExtTest, AttachCurrentThread_SmallStack) {
116   TEST_DISABLED_FOR_MEMORY_TOOL();  // b/123500163
117   pthread_t pthread;
118   pthread_attr_t attr;
119   const char* reason = __PRETTY_FUNCTION__;
120   gSmallStack = true;
121   gAsDaemon = false;
122   CHECK_PTHREAD_CALL(pthread_attr_init, (&attr), reason);
123   CHECK_PTHREAD_CALL(pthread_attr_setstacksize, (&attr, PTHREAD_STACK_MIN), reason);
124   CHECK_PTHREAD_CALL(pthread_create, (&pthread, &attr, attach_current_thread_callback,
125       nullptr), reason);
126   CHECK_PTHREAD_CALL(pthread_attr_destroy, (&attr), reason);
127   void* ret_val;
128   CHECK_PTHREAD_CALL(pthread_join, (pthread, &ret_val), reason);
129   EXPECT_EQ(ret_val, nullptr);
130 }
131 
TEST_F(JavaVmExtTest,DetachCurrentThread)132 TEST_F(JavaVmExtTest, DetachCurrentThread) {
133   JNIEnv* env;
134   jint ok = vm_->AttachCurrentThread(&env, nullptr);
135   ASSERT_EQ(JNI_OK, ok);
136   ok = vm_->DetachCurrentThread();
137   EXPECT_EQ(JNI_OK, ok);
138 
139   jint err = vm_->DetachCurrentThread();
140   EXPECT_EQ(JNI_ERR, err);
141 }
142 
143 class JavaVmExtStackTraceTest : public JavaVmExtTest {
144  protected:
SetUpRuntimeOptions(RuntimeOptions * options)145   void SetUpRuntimeOptions(RuntimeOptions* options) override {
146     options->emplace_back("-XX:GlobalRefAllocStackTraceLimit=50000", nullptr);
147   }
148 };
149 
TEST_F(JavaVmExtStackTraceTest,TestEnableDisable)150 TEST_F(JavaVmExtStackTraceTest, TestEnableDisable) {
151   ASSERT_FALSE(Runtime::Current()->GetHeap()->IsAllocTrackingEnabled());
152 
153   JNIEnv* env;
154   jint ok = vm_->AttachCurrentThread(&env, nullptr);
155   ASSERT_EQ(JNI_OK, ok);
156 
157   std::vector<jobject> global_refs_;
158   jobject local_ref = env->NewStringUTF("Hello");
159   for (size_t i = 0; i < 2000; ++i) {
160     global_refs_.push_back(env->NewGlobalRef(local_ref));
161   }
162 
163   EXPECT_TRUE(Runtime::Current()->GetHeap()->IsAllocTrackingEnabled());
164 
165   for (jobject global_ref : global_refs_) {
166     env->DeleteGlobalRef(global_ref);
167   }
168 
169   EXPECT_FALSE(Runtime::Current()->GetHeap()->IsAllocTrackingEnabled());
170 
171   global_refs_.clear();
172   for (size_t i = 0; i < 2000; ++i) {
173     global_refs_.push_back(env->NewGlobalRef(local_ref));
174   }
175 
176   EXPECT_TRUE(Runtime::Current()->GetHeap()->IsAllocTrackingEnabled());
177 
178   for (jobject global_ref : global_refs_) {
179     env->DeleteGlobalRef(global_ref);
180   }
181 
182   EXPECT_FALSE(Runtime::Current()->GetHeap()->IsAllocTrackingEnabled());
183 
184   ok = vm_->DetachCurrentThread();
185   EXPECT_EQ(JNI_OK, ok);
186 }
187 
188 }  // namespace art
189