1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2017 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 <inttypes.h>
19*795d594fSAndroid Build Coastguard Worker
20*795d594fSAndroid Build Coastguard Worker #include <cstdio>
21*795d594fSAndroid Build Coastguard Worker #include <memory>
22*795d594fSAndroid Build Coastguard Worker
23*795d594fSAndroid Build Coastguard Worker #include "android-base/stringprintf.h"
24*795d594fSAndroid Build Coastguard Worker #include "jni.h"
25*795d594fSAndroid Build Coastguard Worker #include "jvmti.h"
26*795d594fSAndroid Build Coastguard Worker
27*795d594fSAndroid Build Coastguard Worker // Test infrastructure
28*795d594fSAndroid Build Coastguard Worker #include "jni_binder.h"
29*795d594fSAndroid Build Coastguard Worker #include "jvmti_helper.h"
30*795d594fSAndroid Build Coastguard Worker #include "scoped_local_ref.h"
31*795d594fSAndroid Build Coastguard Worker #include "test_env.h"
32*795d594fSAndroid Build Coastguard Worker
33*795d594fSAndroid Build Coastguard Worker namespace art {
34*795d594fSAndroid Build Coastguard Worker namespace Test986NativeBind {
35*795d594fSAndroid Build Coastguard Worker
doUpPrintCall(JNIEnv * env,const char * function)36*795d594fSAndroid Build Coastguard Worker static void doUpPrintCall(JNIEnv* env, const char* function) {
37*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> klass(env, env->FindClass("art/Test986"));
38*795d594fSAndroid Build Coastguard Worker jmethodID targetMethod = env->GetStaticMethodID(klass.get(), function, "()V");
39*795d594fSAndroid Build Coastguard Worker env->CallStaticVoidMethod(klass.get(), targetMethod);
40*795d594fSAndroid Build Coastguard Worker }
41*795d594fSAndroid Build Coastguard Worker
Java_art_Test986_00024Transform_sayHi__(JNIEnv * env,jclass klass)42*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL Java_art_Test986_00024Transform_sayHi__(
43*795d594fSAndroid Build Coastguard Worker JNIEnv* env, [[maybe_unused]] jclass klass) {
44*795d594fSAndroid Build Coastguard Worker doUpPrintCall(env, "doSayHi");
45*795d594fSAndroid Build Coastguard Worker }
46*795d594fSAndroid Build Coastguard Worker
Java_art_Test986_00024Transform_sayHi2(JNIEnv * env,jclass klass)47*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL Java_art_Test986_00024Transform_sayHi2(
48*795d594fSAndroid Build Coastguard Worker JNIEnv* env, [[maybe_unused]] jclass klass) {
49*795d594fSAndroid Build Coastguard Worker doUpPrintCall(env, "doSayHi2");
50*795d594fSAndroid Build Coastguard Worker }
51*795d594fSAndroid Build Coastguard Worker
NoReallySayGoodbye(JNIEnv * env,jclass klass)52*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL NoReallySayGoodbye(JNIEnv* env, [[maybe_unused]] jclass klass) {
53*795d594fSAndroid Build Coastguard Worker doUpPrintCall(env, "doSayBye");
54*795d594fSAndroid Build Coastguard Worker }
55*795d594fSAndroid Build Coastguard Worker
doJvmtiMethodBind(jvmtiEnv * jvmtienv,JNIEnv * env,jthread thread,jmethodID m,void * address,void ** out_address)56*795d594fSAndroid Build Coastguard Worker static void doJvmtiMethodBind([[maybe_unused]] jvmtiEnv* jvmtienv,
57*795d594fSAndroid Build Coastguard Worker JNIEnv* env,
58*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] jthread thread,
59*795d594fSAndroid Build Coastguard Worker jmethodID m,
60*795d594fSAndroid Build Coastguard Worker void* address,
61*795d594fSAndroid Build Coastguard Worker /*out*/void** out_address) {
62*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> method_class(env, env->FindClass("java/lang/reflect/Method"));
63*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jobject> method_obj(env, env->ToReflectedMethod(method_class.get(), m, false));
64*795d594fSAndroid Build Coastguard Worker Dl_info addr_info;
65*795d594fSAndroid Build Coastguard Worker if (dladdr(address, &addr_info) == 0 || addr_info.dli_sname == nullptr) {
66*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> exception_class(env, env->FindClass("java/lang/Exception"));
67*795d594fSAndroid Build Coastguard Worker env->ThrowNew(exception_class.get(), "dladdr failure!");
68*795d594fSAndroid Build Coastguard Worker return;
69*795d594fSAndroid Build Coastguard Worker }
70*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jstring> sym_name(env, env->NewStringUTF(addr_info.dli_sname));
71*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> klass(env, env->FindClass("art/Test986"));
72*795d594fSAndroid Build Coastguard Worker jmethodID upcallMethod = env->GetStaticMethodID(
73*795d594fSAndroid Build Coastguard Worker klass.get(),
74*795d594fSAndroid Build Coastguard Worker "doNativeMethodBind",
75*795d594fSAndroid Build Coastguard Worker "(Ljava/lang/reflect/Method;Ljava/lang/String;)Ljava/lang/String;");
76*795d594fSAndroid Build Coastguard Worker if (env->ExceptionCheck()) {
77*795d594fSAndroid Build Coastguard Worker return;
78*795d594fSAndroid Build Coastguard Worker }
79*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jstring> new_symbol(env,
80*795d594fSAndroid Build Coastguard Worker reinterpret_cast<jstring>(
81*795d594fSAndroid Build Coastguard Worker env->CallStaticObjectMethod(klass.get(),
82*795d594fSAndroid Build Coastguard Worker upcallMethod,
83*795d594fSAndroid Build Coastguard Worker method_obj.get(),
84*795d594fSAndroid Build Coastguard Worker sym_name.get())));
85*795d594fSAndroid Build Coastguard Worker const char* new_symbol_chars = env->GetStringUTFChars(new_symbol.get(), nullptr);
86*795d594fSAndroid Build Coastguard Worker if (strcmp(new_symbol_chars, addr_info.dli_sname) != 0) {
87*795d594fSAndroid Build Coastguard Worker *out_address = dlsym(RTLD_DEFAULT, new_symbol_chars);
88*795d594fSAndroid Build Coastguard Worker if (*out_address == nullptr) {
89*795d594fSAndroid Build Coastguard Worker ScopedLocalRef<jclass> exception_class(env, env->FindClass("java/lang/Exception"));
90*795d594fSAndroid Build Coastguard Worker env->ThrowNew(exception_class.get(), "dlsym failure!");
91*795d594fSAndroid Build Coastguard Worker return;
92*795d594fSAndroid Build Coastguard Worker }
93*795d594fSAndroid Build Coastguard Worker }
94*795d594fSAndroid Build Coastguard Worker env->ReleaseStringUTFChars(new_symbol.get(), new_symbol_chars);
95*795d594fSAndroid Build Coastguard Worker }
96*795d594fSAndroid Build Coastguard Worker
Java_art_Test986_setupNativeBindNotify(JNIEnv * env,jclass klass)97*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL Java_art_Test986_setupNativeBindNotify(
98*795d594fSAndroid Build Coastguard Worker [[maybe_unused]] JNIEnv* env, [[maybe_unused]] jclass klass) {
99*795d594fSAndroid Build Coastguard Worker jvmtiEventCallbacks cb;
100*795d594fSAndroid Build Coastguard Worker memset(&cb, 0, sizeof(cb));
101*795d594fSAndroid Build Coastguard Worker cb.NativeMethodBind = doJvmtiMethodBind;
102*795d594fSAndroid Build Coastguard Worker jvmti_env->SetEventCallbacks(&cb, sizeof(cb));
103*795d594fSAndroid Build Coastguard Worker }
104*795d594fSAndroid Build Coastguard Worker
Java_art_Test986_setNativeBindNotify(JNIEnv * env,jclass klass,jboolean enable)105*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL Java_art_Test986_setNativeBindNotify(
106*795d594fSAndroid Build Coastguard Worker JNIEnv* env, [[maybe_unused]] jclass klass, jboolean enable) {
107*795d594fSAndroid Build Coastguard Worker jvmtiError res = jvmti_env->SetEventNotificationMode(enable ? JVMTI_ENABLE : JVMTI_DISABLE,
108*795d594fSAndroid Build Coastguard Worker JVMTI_EVENT_NATIVE_METHOD_BIND,
109*795d594fSAndroid Build Coastguard Worker nullptr);
110*795d594fSAndroid Build Coastguard Worker if (res != JVMTI_ERROR_NONE) {
111*795d594fSAndroid Build Coastguard Worker JvmtiErrorToException(env, jvmti_env, res);
112*795d594fSAndroid Build Coastguard Worker }
113*795d594fSAndroid Build Coastguard Worker }
114*795d594fSAndroid Build Coastguard Worker
Java_art_Test986_rebindTransformClass(JNIEnv * env,jclass klass,jclass k)115*795d594fSAndroid Build Coastguard Worker extern "C" JNIEXPORT void JNICALL Java_art_Test986_rebindTransformClass(
116*795d594fSAndroid Build Coastguard Worker JNIEnv* env, [[maybe_unused]] jclass klass, jclass k) {
117*795d594fSAndroid Build Coastguard Worker JNINativeMethod m[2];
118*795d594fSAndroid Build Coastguard Worker m[0].name = "sayHi";
119*795d594fSAndroid Build Coastguard Worker m[0].signature = "()V";
120*795d594fSAndroid Build Coastguard Worker m[0].fnPtr = reinterpret_cast<void*>(Java_art_Test986_00024Transform_sayHi__);
121*795d594fSAndroid Build Coastguard Worker m[1].name = "sayHi2";
122*795d594fSAndroid Build Coastguard Worker m[1].signature = "()V";
123*795d594fSAndroid Build Coastguard Worker m[1].fnPtr = reinterpret_cast<void*>(Java_art_Test986_00024Transform_sayHi2);
124*795d594fSAndroid Build Coastguard Worker env->RegisterNatives(k, m, 2);
125*795d594fSAndroid Build Coastguard Worker }
126*795d594fSAndroid Build Coastguard Worker
127*795d594fSAndroid Build Coastguard Worker } // namespace Test986NativeBind
128*795d594fSAndroid Build Coastguard Worker } // namespace art
129