| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 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 |  | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 17 | #include <dlfcn.h> | 
| Mark Salyzyn | a5e161b | 2016-09-29 08:08:05 -0700 | [diff] [blame] | 18 | #include <sys/types.h> | 
|  | 19 | #include <unistd.h> | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 20 |  | 
| Mark Salyzyn | 7823e12 | 2016-09-29 08:08:05 -0700 | [diff] [blame] | 21 | #include <log/log.h> | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 22 |  | 
|  | 23 | #include "jni.h" | 
|  | 24 | #include "DdmConnection.h" | 
|  | 25 |  | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 26 | namespace android { | 
|  | 27 |  | 
| Keun young Park | 63f165f | 2012-08-31 10:53:36 -0700 | [diff] [blame] | 28 | void DdmConnection_start(const char* name) { | 
|  | 29 | ALOGI("DdmConnection_start"); | 
|  | 30 | DdmConnection::start(name); | 
|  | 31 | } | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 32 |  | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 33 | void DdmConnection::start(const char* name) { | 
|  | 34 | JavaVM* vm; | 
|  | 35 | JNIEnv* env; | 
|  | 36 |  | 
|  | 37 | // start a VM | 
|  | 38 | JavaVMInitArgs args; | 
|  | 39 | JavaVMOption opt; | 
|  | 40 |  | 
|  | 41 | opt.optionString = | 
|  | 42 | "-agentlib:jdwp=transport=dt_android_adb,suspend=n,server=y"; | 
|  | 43 |  | 
|  | 44 | args.version = JNI_VERSION_1_4; | 
|  | 45 | args.options = &opt; | 
|  | 46 | args.nOptions = 1; | 
|  | 47 | args.ignoreUnrecognized = JNI_FALSE; | 
|  | 48 |  | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 49 |  | 
| Brian Carlstrom | aefe55f | 2014-05-28 21:27:47 -0700 | [diff] [blame] | 50 | // TODO: Should this just link against libnativehelper and use its | 
|  | 51 | // JNI_CreateJavaVM wrapper that essential does this dlopen/dlsym | 
|  | 52 | // work based on the current system default runtime? | 
|  | 53 | void* libart_dso = dlopen("libart.so", RTLD_NOW); | 
|  | 54 | ALOGE_IF(!libart_dso, "DdmConnection: %s", dlerror()); | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 55 |  | 
|  | 56 | void* libandroid_runtime_dso = dlopen("libandroid_runtime.so", RTLD_NOW); | 
|  | 57 | ALOGE_IF(!libandroid_runtime_dso, "DdmConnection: %s", dlerror()); | 
|  | 58 |  | 
| Brian Carlstrom | aefe55f | 2014-05-28 21:27:47 -0700 | [diff] [blame] | 59 | if (!libart_dso || !libandroid_runtime_dso) { | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 60 | goto error; | 
|  | 61 | } | 
|  | 62 |  | 
|  | 63 | jint (*JNI_CreateJavaVM)(JavaVM** p_vm, JNIEnv** p_env, void* vm_args); | 
| Dan Stoza | f10c46e | 2014-11-11 10:32:31 -0800 | [diff] [blame] | 64 | JNI_CreateJavaVM = reinterpret_cast<decltype(JNI_CreateJavaVM)>( | 
|  | 65 | dlsym(libart_dso, "JNI_CreateJavaVM")); | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 66 | ALOGE_IF(!JNI_CreateJavaVM, "DdmConnection: %s", dlerror()); | 
|  | 67 |  | 
|  | 68 | jint (*registerNatives)(JNIEnv* env, jclass clazz); | 
| Dan Stoza | f10c46e | 2014-11-11 10:32:31 -0800 | [diff] [blame] | 69 | registerNatives = reinterpret_cast<decltype(registerNatives)>( | 
|  | 70 | dlsym(libandroid_runtime_dso, | 
|  | 71 | "Java_com_android_internal_util_WithFramework_registerNatives")); | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 72 | ALOGE_IF(!registerNatives, "DdmConnection: %s", dlerror()); | 
|  | 73 |  | 
|  | 74 | if (!JNI_CreateJavaVM || !registerNatives) { | 
|  | 75 | goto error; | 
|  | 76 | } | 
|  | 77 |  | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 78 | if (JNI_CreateJavaVM(&vm, &env, &args) == 0) { | 
|  | 79 | jclass startClass; | 
|  | 80 | jmethodID startMeth; | 
|  | 81 |  | 
|  | 82 | // register native code | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 83 | if (registerNatives(env, 0) == 0) { | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 84 | // set our name by calling DdmHandleAppName.setAppName() | 
|  | 85 | startClass = env->FindClass("android/ddm/DdmHandleAppName"); | 
|  | 86 | if (startClass) { | 
|  | 87 | startMeth = env->GetStaticMethodID(startClass, | 
| Mathias Agopian | 1b3aeb4 | 2012-10-07 16:41:12 -0700 | [diff] [blame] | 88 | "setAppName", "(Ljava/lang/String;I)V"); | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 89 | if (startMeth) { | 
|  | 90 | jstring str = env->NewStringUTF(name); | 
| Mathias Agopian | 1b3aeb4 | 2012-10-07 16:41:12 -0700 | [diff] [blame] | 91 | env->CallStaticVoidMethod(startClass, startMeth, str, getuid()); | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 92 | env->DeleteLocalRef(str); | 
|  | 93 | } | 
|  | 94 | } | 
|  | 95 |  | 
|  | 96 | // initialize DDMS communication by calling | 
|  | 97 | // DdmRegister.registerHandlers() | 
|  | 98 | startClass = env->FindClass("android/ddm/DdmRegister"); | 
|  | 99 | if (startClass) { | 
|  | 100 | startMeth = env->GetStaticMethodID(startClass, | 
|  | 101 | "registerHandlers", "()V"); | 
|  | 102 | if (startMeth) { | 
|  | 103 | env->CallStaticVoidMethod(startClass, startMeth); | 
|  | 104 | } | 
|  | 105 | } | 
|  | 106 | } | 
|  | 107 | } | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 108 | return; | 
|  | 109 |  | 
|  | 110 | error: | 
|  | 111 | if (libandroid_runtime_dso) { | 
|  | 112 | dlclose(libandroid_runtime_dso); | 
|  | 113 | } | 
| Brian Carlstrom | aefe55f | 2014-05-28 21:27:47 -0700 | [diff] [blame] | 114 | if (libart_dso) { | 
|  | 115 | dlclose(libart_dso); | 
| Mathias Agopian | c1d359d | 2012-08-04 20:09:03 -0700 | [diff] [blame] | 116 | } | 
| Mathias Agopian | 8afb7e3 | 2011-08-15 20:44:40 -0700 | [diff] [blame] | 117 | } | 
|  | 118 |  | 
|  | 119 | }; // namespace android |