Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2016 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 | //#define LOG_NDEBUG 0 |
| 18 | #define LOG_TAG "android_os_HwBinder" |
| 19 | #include <android-base/logging.h> |
| 20 | |
| 21 | #include "android_os_HwBinder.h" |
| 22 | |
| 23 | #include "android_os_HwParcel.h" |
| 24 | #include "android_os_HwRemoteBinder.h" |
| 25 | |
Steven Moreland | afe95a9 | 2017-05-23 12:45:16 -0700 | [diff] [blame] | 26 | #include <cstring> |
| 27 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 28 | #include <JNIHelp.h> |
Steven Moreland | a813686 | 2016-10-24 13:44:41 -0700 | [diff] [blame] | 29 | #include <android/hidl/manager/1.0/IServiceManager.h> |
Yifan Hong | bae1b55 | 2016-12-01 10:25:05 -0800 | [diff] [blame] | 30 | #include <android/hidl/base/1.0/IBase.h> |
Yifan Hong | 90b6a37 | 2017-01-09 17:58:33 -0800 | [diff] [blame] | 31 | #include <android/hidl/base/1.0/BpHwBase.h> |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 32 | #include <android_runtime/AndroidRuntime.h> |
Steven Moreland | a813686 | 2016-10-24 13:44:41 -0700 | [diff] [blame] | 33 | #include <hidl/ServiceManagement.h> |
Martijn Coenen | aa2c32f | 2016-09-01 01:37:05 +0200 | [diff] [blame] | 34 | #include <hidl/Status.h> |
Martijn Coenen | 1298711 | 2016-12-08 10:48:19 +0100 | [diff] [blame] | 35 | #include <hidl/HidlTransportSupport.h> |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 36 | #include <hwbinder/ProcessState.h> |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 37 | #include <nativehelper/ScopedLocalRef.h> |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 38 | #include <vintf/parse_string.h> |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 39 | |
| 40 | #include "core_jni_helpers.h" |
| 41 | |
| 42 | using android::AndroidRuntime; |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 43 | using android::hardware::hidl_vec; |
| 44 | using android::hardware::hidl_string; |
Steven Moreland | c0631d0 | 2017-01-04 10:37:59 -0800 | [diff] [blame] | 45 | template<typename T> |
| 46 | using Return = android::hardware::Return<T>; |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 47 | |
| 48 | #define PACKAGE_PATH "android/os" |
| 49 | #define CLASS_NAME "HwBinder" |
| 50 | #define CLASS_PATH PACKAGE_PATH "/" CLASS_NAME |
| 51 | |
| 52 | namespace android { |
| 53 | |
Steven Moreland | e62b1f3 | 2016-12-20 15:55:48 -0800 | [diff] [blame] | 54 | static jclass gErrorClass; |
| 55 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 56 | static struct fields_t { |
| 57 | jfieldID contextID; |
| 58 | jmethodID onTransactID; |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 59 | } gFields; |
| 60 | |
| 61 | // static |
| 62 | void JHwBinder::InitClass(JNIEnv *env) { |
| 63 | ScopedLocalRef<jclass> clazz( |
| 64 | env, FindClassOrDie(env, CLASS_PATH)); |
| 65 | |
| 66 | gFields.contextID = |
| 67 | GetFieldIDOrDie(env, clazz.get(), "mNativeContext", "J"); |
| 68 | |
| 69 | gFields.onTransactID = |
| 70 | GetMethodIDOrDie( |
| 71 | env, |
| 72 | clazz.get(), |
| 73 | "onTransact", |
| 74 | "(IL" PACKAGE_PATH "/HwParcel;L" PACKAGE_PATH "/HwParcel;I)V"); |
| 75 | } |
| 76 | |
| 77 | // static |
| 78 | sp<JHwBinder> JHwBinder::SetNativeContext( |
| 79 | JNIEnv *env, jobject thiz, const sp<JHwBinder> &context) { |
| 80 | sp<JHwBinder> old = |
| 81 | (JHwBinder *)env->GetLongField(thiz, gFields.contextID); |
| 82 | |
| 83 | if (context != NULL) { |
| 84 | context->incStrong(NULL /* id */); |
| 85 | } |
| 86 | |
| 87 | if (old != NULL) { |
| 88 | old->decStrong(NULL /* id */); |
| 89 | } |
| 90 | |
| 91 | env->SetLongField(thiz, gFields.contextID, (long)context.get()); |
| 92 | |
| 93 | return old; |
| 94 | } |
| 95 | |
| 96 | // static |
| 97 | sp<JHwBinder> JHwBinder::GetNativeContext( |
| 98 | JNIEnv *env, jobject thiz) { |
| 99 | return (JHwBinder *)env->GetLongField(thiz, gFields.contextID); |
| 100 | } |
| 101 | |
| 102 | JHwBinder::JHwBinder(JNIEnv *env, jobject thiz) { |
| 103 | jclass clazz = env->GetObjectClass(thiz); |
| 104 | CHECK(clazz != NULL); |
| 105 | |
| 106 | mClass = (jclass)env->NewGlobalRef(clazz); |
| 107 | mObject = env->NewWeakGlobalRef(thiz); |
| 108 | } |
| 109 | |
| 110 | JHwBinder::~JHwBinder() { |
| 111 | JNIEnv *env = AndroidRuntime::getJNIEnv(); |
| 112 | |
| 113 | env->DeleteWeakGlobalRef(mObject); |
| 114 | mObject = NULL; |
| 115 | |
| 116 | env->DeleteGlobalRef(mClass); |
| 117 | mClass = NULL; |
| 118 | } |
| 119 | |
| 120 | status_t JHwBinder::onTransact( |
| 121 | uint32_t code, |
| 122 | const hardware::Parcel &data, |
| 123 | hardware::Parcel *reply, |
| 124 | uint32_t flags, |
| 125 | TransactCallback callback) { |
| 126 | JNIEnv *env = AndroidRuntime::getJNIEnv(); |
Martijn Coenen | 60bf84a | 2017-02-08 10:22:28 +0100 | [diff] [blame] | 127 | bool isOneway = (flags & TF_ONE_WAY) != 0; |
| 128 | ScopedLocalRef<jobject> replyObj(env, nullptr); |
| 129 | sp<JHwParcel> replyContext = nullptr; |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 130 | |
| 131 | ScopedLocalRef<jobject> requestObj(env, JHwParcel::NewObject(env)); |
| 132 | JHwParcel::GetNativeContext(env, requestObj.get())->setParcel( |
| 133 | const_cast<hardware::Parcel *>(&data), false /* assumeOwnership */); |
| 134 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 135 | |
Martijn Coenen | 60bf84a | 2017-02-08 10:22:28 +0100 | [diff] [blame] | 136 | if (!isOneway) { |
| 137 | replyObj.reset(JHwParcel::NewObject(env)); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 138 | |
Martijn Coenen | 60bf84a | 2017-02-08 10:22:28 +0100 | [diff] [blame] | 139 | replyContext = JHwParcel::GetNativeContext(env, replyObj.get()); |
| 140 | |
| 141 | replyContext->setParcel(reply, false /* assumeOwnership */); |
| 142 | replyContext->setTransactCallback(callback); |
| 143 | } |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 144 | |
| 145 | env->CallVoidMethod( |
| 146 | mObject, |
| 147 | gFields.onTransactID, |
| 148 | code, |
| 149 | requestObj.get(), |
| 150 | replyObj.get(), |
| 151 | flags); |
| 152 | |
Steven Moreland | e62b1f3 | 2016-12-20 15:55:48 -0800 | [diff] [blame] | 153 | if (env->ExceptionCheck()) { |
| 154 | jthrowable excep = env->ExceptionOccurred(); |
| 155 | env->ExceptionDescribe(); |
| 156 | |
| 157 | if (env->IsInstanceOf(excep, gErrorClass)) { |
| 158 | /* It's an error */ |
| 159 | LOG(ERROR) << "Forcefully exiting"; |
| 160 | exit(1); |
| 161 | } else { |
| 162 | env->ExceptionClear(); |
| 163 | LOG(ERROR) << "Uncaught exception!"; |
| 164 | } |
| 165 | |
| 166 | env->DeleteLocalRef(excep); |
| 167 | } |
| 168 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 169 | status_t err = OK; |
| 170 | |
Martijn Coenen | 60bf84a | 2017-02-08 10:22:28 +0100 | [diff] [blame] | 171 | if (!isOneway) { |
| 172 | if (!replyContext->wasSent()) { |
| 173 | // The implementation never finished the transaction. |
| 174 | err = UNKNOWN_ERROR; // XXX special error code instead? |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 175 | |
Martijn Coenen | 60bf84a | 2017-02-08 10:22:28 +0100 | [diff] [blame] | 176 | reply->setDataPosition(0 /* pos */); |
| 177 | } |
| 178 | |
| 179 | // Release all temporary storage now that scatter-gather data |
| 180 | // has been consolidated, either by calling the TransactCallback, |
| 181 | // if wasSent() == true or clearing the reply parcel (setDataOffset above). |
| 182 | replyContext->getStorage()->release(env); |
| 183 | |
| 184 | // We cannot permanently pass ownership of "data" and "reply" over to their |
| 185 | // Java object wrappers (we don't own them ourselves). |
| 186 | replyContext->setParcel( |
| 187 | NULL /* parcel */, false /* assumeOwnership */); |
| 188 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 189 | } |
| 190 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 191 | JHwParcel::GetNativeContext(env, requestObj.get())->setParcel( |
| 192 | NULL /* parcel */, false /* assumeOwnership */); |
| 193 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 194 | return err; |
| 195 | } |
| 196 | |
| 197 | } // namespace android |
| 198 | |
| 199 | //////////////////////////////////////////////////////////////////////////////// |
| 200 | |
| 201 | using namespace android; |
| 202 | |
| 203 | static void releaseNativeContext(void *nativeContext) { |
| 204 | sp<JHwBinder> binder = (JHwBinder *)nativeContext; |
| 205 | |
| 206 | if (binder != NULL) { |
| 207 | binder->decStrong(NULL /* id */); |
| 208 | } |
| 209 | } |
| 210 | |
| 211 | static jlong JHwBinder_native_init(JNIEnv *env) { |
| 212 | JHwBinder::InitClass(env); |
| 213 | |
| 214 | return reinterpret_cast<jlong>(&releaseNativeContext); |
| 215 | } |
| 216 | |
| 217 | static void JHwBinder_native_setup(JNIEnv *env, jobject thiz) { |
| 218 | sp<JHwBinder> context = new JHwBinder(env, thiz); |
| 219 | |
| 220 | JHwBinder::SetNativeContext(env, thiz, context); |
| 221 | } |
| 222 | |
| 223 | static void JHwBinder_native_transact( |
| 224 | JNIEnv * /* env */, |
| 225 | jobject /* thiz */, |
| 226 | jint /* code */, |
| 227 | jobject /* requestObj */, |
| 228 | jobject /* replyObj */, |
| 229 | jint /* flags */) { |
| 230 | CHECK(!"Should not be here"); |
| 231 | } |
| 232 | |
| 233 | static void JHwBinder_native_registerService( |
Andreas Huber | 35eb799 | 2016-10-25 13:29:30 -0700 | [diff] [blame] | 234 | JNIEnv *env, |
| 235 | jobject thiz, |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 236 | jstring serviceNameObj) { |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 237 | if (serviceNameObj == NULL) { |
| 238 | jniThrowException(env, "java/lang/NullPointerException", NULL); |
| 239 | return; |
| 240 | } |
| 241 | |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 242 | const char *serviceName = env->GetStringUTFChars(serviceNameObj, NULL); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 243 | if (serviceName == NULL) { |
| 244 | return; // XXX exception already pending? |
| 245 | } |
| 246 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 247 | sp<hardware::IBinder> binder = JHwBinder::GetNativeContext(env, thiz); |
Steven Moreland | 520d12c | 2016-12-15 15:50:17 -0800 | [diff] [blame] | 248 | |
Martijn Coenen | 1298711 | 2016-12-08 10:48:19 +0100 | [diff] [blame] | 249 | /* TODO(b/33440494) this is not right */ |
Yifan Hong | 90b6a37 | 2017-01-09 17:58:33 -0800 | [diff] [blame] | 250 | sp<hidl::base::V1_0::IBase> base = new hidl::base::V1_0::BpHwBase(binder); |
Steven Moreland | 520d12c | 2016-12-15 15:50:17 -0800 | [diff] [blame] | 251 | |
| 252 | auto manager = hardware::defaultServiceManager(); |
| 253 | |
| 254 | if (manager == nullptr) { |
| 255 | LOG(ERROR) << "Could not get hwservicemanager."; |
Steven Moreland | c0631d0 | 2017-01-04 10:37:59 -0800 | [diff] [blame] | 256 | signalExceptionForError(env, UNKNOWN_ERROR, true /* canThrowRemoteException */); |
Steven Moreland | 520d12c | 2016-12-15 15:50:17 -0800 | [diff] [blame] | 257 | return; |
| 258 | } |
| 259 | |
Martijn Coenen | 85d12da | 2017-03-06 13:07:53 +0100 | [diff] [blame] | 260 | Return<bool> ret = manager->add(serviceName, base); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 261 | |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 262 | env->ReleaseStringUTFChars(serviceNameObj, serviceName); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 263 | serviceName = NULL; |
| 264 | |
Steven Moreland | c0631d0 | 2017-01-04 10:37:59 -0800 | [diff] [blame] | 265 | bool ok = ret.isOk() && ret; |
| 266 | |
Steven Moreland | a813686 | 2016-10-24 13:44:41 -0700 | [diff] [blame] | 267 | if (ok) { |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 268 | LOG(INFO) << "Starting thread pool."; |
| 269 | ::android::hardware::ProcessState::self()->startThreadPool(); |
| 270 | } |
| 271 | |
Steven Moreland | c0631d0 | 2017-01-04 10:37:59 -0800 | [diff] [blame] | 272 | signalExceptionForError(env, (ok ? OK : UNKNOWN_ERROR), true /* canThrowRemoteException */); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 273 | } |
| 274 | |
| 275 | static jobject JHwBinder_native_getService( |
Andreas Huber | 35eb799 | 2016-10-25 13:29:30 -0700 | [diff] [blame] | 276 | JNIEnv *env, |
| 277 | jclass /* clazzObj */, |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 278 | jstring ifaceNameObj, |
| 279 | jstring serviceNameObj) { |
| 280 | |
Steven Moreland | f7019c2 | 2017-04-06 09:28:32 -0700 | [diff] [blame] | 281 | using ::android::hidl::base::V1_0::IBase; |
| 282 | using ::android::hidl::manager::V1_0::IServiceManager; |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 283 | |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 284 | if (ifaceNameObj == NULL) { |
| 285 | jniThrowException(env, "java/lang/NullPointerException", NULL); |
| 286 | return NULL; |
| 287 | } |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 288 | if (serviceNameObj == NULL) { |
| 289 | jniThrowException(env, "java/lang/NullPointerException", NULL); |
| 290 | return NULL; |
| 291 | } |
| 292 | |
Steven Moreland | 2f379ca | 2017-02-01 09:58:00 -0800 | [diff] [blame] | 293 | auto manager = hardware::defaultServiceManager(); |
| 294 | |
| 295 | if (manager == nullptr) { |
| 296 | LOG(ERROR) << "Could not get hwservicemanager."; |
| 297 | signalExceptionForError(env, UNKNOWN_ERROR, true /* canThrowRemoteException */); |
| 298 | return NULL; |
| 299 | } |
| 300 | |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 301 | const char *ifaceNameCStr = env->GetStringUTFChars(ifaceNameObj, NULL); |
| 302 | if (ifaceNameCStr == NULL) { |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 303 | return NULL; // XXX exception already pending? |
Andreas Huber | 35eb799 | 2016-10-25 13:29:30 -0700 | [diff] [blame] | 304 | } |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 305 | std::string ifaceName(ifaceNameCStr); |
| 306 | env->ReleaseStringUTFChars(ifaceNameObj, ifaceNameCStr); |
| 307 | ::android::hardware::hidl_string ifaceNameHStr; |
| 308 | ifaceNameHStr.setToExternal(ifaceName.c_str(), ifaceName.size()); |
| 309 | |
| 310 | const char *serviceNameCStr = env->GetStringUTFChars(serviceNameObj, NULL); |
| 311 | if (serviceNameCStr == NULL) { |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 312 | return NULL; // XXX exception already pending? |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 313 | } |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 314 | std::string serviceName(serviceNameCStr); |
| 315 | env->ReleaseStringUTFChars(serviceNameObj, serviceNameCStr); |
| 316 | ::android::hardware::hidl_string serviceNameHStr; |
| 317 | serviceNameHStr.setToExternal(serviceName.c_str(), serviceName.size()); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 318 | |
Steven Moreland | 2f379ca | 2017-02-01 09:58:00 -0800 | [diff] [blame] | 319 | LOG(INFO) << "Looking for service " |
| 320 | << ifaceName |
| 321 | << "/" |
| 322 | << serviceName; |
Steven Moreland | 520d12c | 2016-12-15 15:50:17 -0800 | [diff] [blame] | 323 | |
Steven Moreland | f7019c2 | 2017-04-06 09:28:32 -0700 | [diff] [blame] | 324 | Return<IServiceManager::Transport> transportRet = |
| 325 | manager->getTransport(ifaceNameHStr, serviceNameHStr); |
| 326 | |
| 327 | if (!transportRet.isOk()) { |
| 328 | signalExceptionForError(env, UNKNOWN_ERROR, true /* canThrowRemoteException */); |
| 329 | return NULL; |
| 330 | } |
| 331 | |
| 332 | IServiceManager::Transport transport = transportRet; |
| 333 | |
Steven Moreland | afe95a9 | 2017-05-23 12:45:16 -0700 | [diff] [blame] | 334 | #ifdef __ANDROID_TREBLE__ |
| 335 | #ifdef __ANDROID_DEBUGGABLE__ |
| 336 | const char* testingOverride = std::getenv("TREBLE_TESTING_OVERRIDE"); |
| 337 | const bool vintfLegacy = (transport == IServiceManager::Transport::EMPTY) |
| 338 | && testingOverride && !strcmp(testingOverride, "true"); |
| 339 | #else // __ANDROID_TREBLE__ but not __ANDROID_DEBUGGABLE__ |
| 340 | const bool vintfLegacy = false; |
| 341 | #endif // __ANDROID_DEBUGGABLE__ |
| 342 | #else // not __ANDROID_TREBLE__ |
| 343 | const bool vintfLegacy = (transport == IServiceManager::Transport::EMPTY); |
| 344 | #endif // __ANDROID_TREBLE__"; |
| 345 | |
| 346 | if (transport != IServiceManager::Transport::HWBINDER && !vintfLegacy) { |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 347 | LOG(ERROR) << "service " << ifaceName << " declares transport method " |
Steven Moreland | f7019c2 | 2017-04-06 09:28:32 -0700 | [diff] [blame] | 348 | << toString(transport) << " but framework expects hwbinder."; |
Steven Moreland | 4c5ffe4 | 2017-06-28 10:21:00 -0700 | [diff] [blame^] | 349 | signalExceptionForError(env, NAME_NOT_FOUND, true /* canThrowRemoteException */); |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 350 | return NULL; |
| 351 | } |
Steven Moreland | c0631d0 | 2017-01-04 10:37:59 -0800 | [diff] [blame] | 352 | |
Yifan Hong | 0382be2 | 2017-02-06 12:43:08 -0800 | [diff] [blame] | 353 | Return<sp<hidl::base::V1_0::IBase>> ret = manager->get(ifaceNameHStr, serviceNameHStr); |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 354 | |
Steven Moreland | 2f379ca | 2017-02-01 09:58:00 -0800 | [diff] [blame] | 355 | if (!ret.isOk()) { |
| 356 | signalExceptionForError(env, UNKNOWN_ERROR, true /* canThrowRemoteException */); |
| 357 | return NULL; |
| 358 | } |
| 359 | |
| 360 | sp<hardware::IBinder> service = hardware::toBinder< |
| 361 | hidl::base::V1_0::IBase, hidl::base::V1_0::BpHwBase>(ret); |
| 362 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 363 | if (service == NULL) { |
| 364 | signalExceptionForError(env, NAME_NOT_FOUND); |
| 365 | return NULL; |
| 366 | } |
| 367 | |
Andreas Huber | 781c083 | 2016-10-05 11:10:26 -0700 | [diff] [blame] | 368 | LOG(INFO) << "Starting thread pool."; |
| 369 | ::android::hardware::ProcessState::self()->startThreadPool(); |
| 370 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 371 | return JHwRemoteBinder::NewObject(env, service); |
| 372 | } |
| 373 | |
| 374 | static JNINativeMethod gMethods[] = { |
| 375 | { "native_init", "()J", (void *)JHwBinder_native_init }, |
| 376 | { "native_setup", "()V", (void *)JHwBinder_native_setup }, |
| 377 | |
| 378 | { "transact", |
| 379 | "(IL" PACKAGE_PATH "/HwParcel;L" PACKAGE_PATH "/HwParcel;I)V", |
| 380 | (void *)JHwBinder_native_transact }, |
| 381 | |
Martijn Coenen | 85d12da | 2017-03-06 13:07:53 +0100 | [diff] [blame] | 382 | { "registerService", "(Ljava/lang/String;)V", |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 383 | (void *)JHwBinder_native_registerService }, |
| 384 | |
Steven Moreland | f3c5349 | 2016-11-03 15:17:04 -0700 | [diff] [blame] | 385 | { "getService", "(Ljava/lang/String;Ljava/lang/String;)L" PACKAGE_PATH "/IHwBinder;", |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 386 | (void *)JHwBinder_native_getService }, |
| 387 | }; |
| 388 | |
| 389 | namespace android { |
| 390 | |
| 391 | int register_android_os_HwBinder(JNIEnv *env) { |
Steven Moreland | e62b1f3 | 2016-12-20 15:55:48 -0800 | [diff] [blame] | 392 | jclass errorClass = FindClassOrDie(env, "java/lang/Error"); |
| 393 | gErrorClass = MakeGlobalRefOrDie(env, errorClass); |
| 394 | |
Andreas Huber | dab5fc6 | 2016-08-15 09:25:02 -0700 | [diff] [blame] | 395 | return RegisterMethodsOrDie(env, CLASS_PATH, gMethods, NELEM(gMethods)); |
| 396 | } |
| 397 | |
| 398 | } // namespace android |