/* * Copyright 2017 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 <jni.h> #include <stdio.h> #include <iostream> #include <fstream> #include <stdio.h> #include <sstream> #include "jvmti.h" #include "exec_utils.h" #include "utils.h" namespace art { // Should we do a 'full_rewrite' with this test? static constexpr bool kDoFullRewrite = true; struct StressData { std::string dexter_cmd; std::string out_temp_dex; std::string in_temp_dex; bool vm_class_loader_initialized; }; static void WriteToFile(const std::string& fname, jint data_len, const unsigned char* data) { std::ofstream file(fname, std::ios::binary | std::ios::out | std::ios::trunc); file.write(reinterpret_cast<const char*>(data), data_len); file.flush(); } static bool ReadIntoBuffer(const std::string& fname, /*out*/std::vector<unsigned char>* data) { std::ifstream file(fname, std::ios::binary | std::ios::in); file.seekg(0, std::ios::end); size_t len = file.tellg(); data->resize(len); file.seekg(0); file.read(reinterpret_cast<char*>(data->data()), len); return len != 0; } // TODO rewrite later. static bool DoExtractClassFromData(StressData* data, const std::string& class_name, jint in_len, const unsigned char* in_data, /*out*/std::vector<unsigned char>* dex) { // Write the dex file into a temporary file. WriteToFile(data->in_temp_dex, in_len, in_data); // Clear out file so even if something suppresses the exit value we will still detect dexter // failure. WriteToFile(data->out_temp_dex, 0, nullptr); // Have dexter do the extraction. std::vector<std::string> args; args.push_back(data->dexter_cmd); if (kDoFullRewrite) { args.push_back("-x"); args.push_back("full_rewrite"); } args.push_back("-e"); args.push_back(class_name); args.push_back("-o"); args.push_back(data->out_temp_dex); args.push_back(data->in_temp_dex); std::string error; if (ExecAndReturnCode(args, &error) != 0) { LOG(ERROR) << "unable to execute dexter: " << error; return false; } return ReadIntoBuffer(data->out_temp_dex, dex); } static void doJvmtiMethodBind(jvmtiEnv* jvmtienv, JNIEnv* env, jthread thread, jmethodID m, void* address, /*out*/void** out_address) { *out_address = address; jvmtiThreadInfo info; if (thread == nullptr) { info.name = const_cast<char*>("<NULLPTR>"); } else if (jvmtienv->GetThreadInfo(thread, &info) != JVMTI_ERROR_NONE) { LOG(WARNING) << "Unable to get thread info!"; info.name = const_cast<char*>("<UNKNOWN THREAD>"); } char *fname, *fsig, *fgen; char *cname, *cgen; jclass klass = nullptr; if (jvmtienv->GetMethodDeclaringClass(m, &klass) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to get method declaring class!"; return; } if (jvmtienv->GetMethodName(m, &fname, &fsig, &fgen) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to get method name!"; env->DeleteLocalRef(klass); return; } if (jvmtienv->GetClassSignature(klass, &cname, &cgen) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to get class name!"; env->DeleteLocalRef(klass); return; } LOG(INFO) << "Loading native method \"" << cname << "->" << fname << fsig << "\". Thread is " << info.name; jvmtienv->Deallocate(reinterpret_cast<unsigned char*>(cname)); jvmtienv->Deallocate(reinterpret_cast<unsigned char*>(cgen)); jvmtienv->Deallocate(reinterpret_cast<unsigned char*>(fname)); jvmtienv->Deallocate(reinterpret_cast<unsigned char*>(fsig)); jvmtienv->Deallocate(reinterpret_cast<unsigned char*>(fgen)); env->DeleteLocalRef(klass); return; } // The hook we are using. void JNICALL ClassFileLoadHookSecretNoOp(jvmtiEnv* jvmti, JNIEnv* jni_env ATTRIBUTE_UNUSED, jclass class_being_redefined ATTRIBUTE_UNUSED, jobject loader ATTRIBUTE_UNUSED, const char* name, jobject protection_domain ATTRIBUTE_UNUSED, jint class_data_len, const unsigned char* class_data, jint* new_class_data_len, unsigned char** new_class_data) { std::vector<unsigned char> out; std::string name_str(name); // Make the jvmti semi-descriptor into the java style descriptor (though with $ for inner // classes). std::replace(name_str.begin(), name_str.end(), '/', '.'); StressData* data = nullptr; CHECK_EQ(jvmti->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)), JVMTI_ERROR_NONE); if (!data->vm_class_loader_initialized) { LOG(WARNING) << "Ignoring load of class " << name << " because VMClassLoader is not yet " << "initialized. Transforming this class could cause spurious test failures."; return; } else if (DoExtractClassFromData(data, name_str, class_data_len, class_data, /*out*/ &out)) { LOG(INFO) << "Extracted class: " << name; unsigned char* new_data; CHECK_EQ(JVMTI_ERROR_NONE, jvmti->Allocate(out.size(), &new_data)); memcpy(new_data, out.data(), out.size()); *new_class_data_len = static_cast<jint>(out.size()); *new_class_data = new_data; } else { std::cerr << "Unable to extract class " << name_str << std::endl; *new_class_data_len = 0; *new_class_data = nullptr; } } // Options are ${DEXTER_BINARY},${TEMP_FILE_1},${TEMP_FILE_2} static void ReadOptions(StressData* data, char* options) { std::string ops(options); data->dexter_cmd = ops.substr(0, ops.find(',')); ops = ops.substr(ops.find(',') + 1); data->in_temp_dex = ops.substr(0, ops.find(',')); ops = ops.substr(ops.find(',') + 1); data->out_temp_dex = ops; } // We need to make sure that VMClassLoader is initialized before we start redefining anything since // it can give (non-fatal) error messages if it's initialized after we've redefined BCP classes. // These error messages are expected and no problem but they will mess up our testing // infrastructure. static void JNICALL EnsureVMClassloaderInitializedCB(jvmtiEnv *jvmti_env, JNIEnv* jni_env, jthread thread ATTRIBUTE_UNUSED) { // Load the VMClassLoader class. We will get a ClassNotFound exception because we don't have // visibility but the class will be loaded behind the scenes. LOG(INFO) << "manual load & initialization of class java/lang/VMClassLoader!"; jclass klass = jni_env->FindClass("java/lang/VMClassLoader"); if (klass == nullptr) { // Probably on RI. Clear the exception so we can continue but don't mark vmclassloader as // initialized. LOG(WARNING) << "Unable to find VMClassLoader class!"; jni_env->ExceptionClear(); } else { // GetMethodID is spec'd to cause the class to be initialized. jni_env->GetMethodID(klass, "hashCode", "()I"); jni_env->DeleteLocalRef(klass); StressData* data = nullptr; CHECK_EQ(jvmti_env->GetEnvironmentLocalStorage(reinterpret_cast<void**>(&data)), JVMTI_ERROR_NONE); data->vm_class_loader_initialized = true; } } extern "C" JNIEXPORT jint JNICALL Agent_OnLoad(JavaVM* vm, char* options, void* reserved ATTRIBUTE_UNUSED) { jvmtiEnv* jvmti = nullptr; if (vm->GetEnv(reinterpret_cast<void**>(&jvmti), JVMTI_VERSION_1_0)) { LOG(ERROR) << "Unable to get jvmti env."; return 1; } StressData* data = nullptr; if (JVMTI_ERROR_NONE != jvmti->Allocate(sizeof(StressData), reinterpret_cast<unsigned char**>(&data))) { LOG(ERROR) << "Unable to allocate data for stress test."; return 1; } memset(data, 0, sizeof(StressData)); // Read the options into the static variables that hold them. ReadOptions(data, options); // Save the data if (JVMTI_ERROR_NONE != jvmti->SetEnvironmentLocalStorage(data)) { LOG(ERROR) << "Unable to save stress test data."; return 1; } // Just get all capabilities. jvmtiCapabilities caps; jvmti->GetPotentialCapabilities(&caps); jvmti->AddCapabilities(&caps); // Set callbacks. jvmtiEventCallbacks cb; memset(&cb, 0, sizeof(cb)); cb.ClassFileLoadHook = ClassFileLoadHookSecretNoOp; cb.NativeMethodBind = doJvmtiMethodBind; cb.VMInit = EnsureVMClassloaderInitializedCB; if (jvmti->SetEventCallbacks(&cb, sizeof(cb)) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to set class file load hook cb!"; return 1; } if (jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_NATIVE_METHOD_BIND, nullptr) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to enable JVMTI_EVENT_NATIVE_METHOD_BIND event!"; return 1; } if (jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_VM_INIT, nullptr) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to enable JVMTI_EVENT_VM_INIT event!"; return 1; } if (jvmti->SetEventNotificationMode(JVMTI_ENABLE, JVMTI_EVENT_CLASS_FILE_LOAD_HOOK, nullptr) != JVMTI_ERROR_NONE) { LOG(ERROR) << "Unable to enable CLASS_FILE_LOAD_HOOK event!"; return 1; } return 0; } } // namespace art