1 /*
2  * Copyright 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 #undef LOG_NDEBUG
18 #undef LOG_TAG
19 #define LOG_NDEBUG 0
20 #define LOG_TAG "ContextHubService"
21 
22 #include <inttypes.h>
23 #include <jni.h>
24 #include <stdint.h>
25 #include <stdio.h>
26 #include <stdlib.h>
27 #include <string.h>
28 #include <sys/endian.h>
29 
30 #include <chrono>
31 #include <mutex>
32 #include <queue>
33 #include <unordered_map>
34 #include <utility>
35 
36 #include <android-base/macros.h>
37 #include <android/hardware/contexthub/1.0/IContexthub.h>
38 #include <cutils/log.h>
39 
40 #include "core_jni_helpers.h"
41 #include "JNIHelp.h"
42 
43 using android::hardware::contexthub::V1_0::AsyncEventType;
44 using android::hardware::contexthub::V1_0::ContextHub;
45 using android::hardware::contexthub::V1_0::ContextHubMsg;
46 using android::hardware::contexthub::V1_0::HubAppInfo;
47 using android::hardware::contexthub::V1_0::IContexthub;
48 using android::hardware::contexthub::V1_0::IContexthubCallback;
49 using android::hardware::contexthub::V1_0::NanoAppBinary;
50 using android::hardware::contexthub::V1_0::Result;
51 using android::hardware::contexthub::V1_0::TransactionResult;
52 
53 using android::hardware::Return;
54 
55 using std::chrono::steady_clock;
56 
57 // If a transaction takes longer than this, we'll allow it to be
58 // canceled by a new transaction.  Note we do _not_ automatically
59 // cancel a transaction after this much time.  We can have a
60 // legal transaction which takes longer than this amount of time,
61 // as long as no other new transactions are attempted after this
62 // time has expired.
63 constexpr auto kMinTransactionCancelTime = std::chrono::seconds(29);
64 
65 namespace android {
66 
67 constexpr uint32_t kNanoAppBinaryHeaderVersion = 1;
68 
69 // Important: this header is explicitly defined as little endian byte order, and
70 // therefore may not match host endianness
71 struct NanoAppBinaryHeader {
72     uint32_t headerVersion;        // 0x1 for this version
73     uint32_t magic;                // "NANO" (see NANOAPP_MAGIC in context_hub.h)
74     uint64_t appId;                // App Id, contains vendor id
75     uint32_t appVersion;           // Version of the app
76     uint32_t flags;                // Signed, encrypted
77     uint64_t hwHubType;            // Which hub type is this compiled for
78     uint8_t targetChreApiMajorVersion; // Which CHRE API version this is compiled for
79     uint8_t targetChreApiMinorVersion;
80     uint8_t reserved[6];
81 } __attribute__((packed));
82 
83 enum HubMessageType {
84     CONTEXT_HUB_APPS_ENABLE  = 1, // Enables loaded nano-app(s)
85     CONTEXT_HUB_APPS_DISABLE = 2, // Disables loaded nano-app(s)
86     CONTEXT_HUB_LOAD_APP     = 3, // Load a supplied app
87     CONTEXT_HUB_UNLOAD_APP   = 4, // Unload a specified app
88     CONTEXT_HUB_QUERY_APPS   = 5, // Query for app(s) info on hub
89     CONTEXT_HUB_QUERY_MEMORY = 6, // Query for memory info
90     CONTEXT_HUB_OS_REBOOT    = 7, // Request to reboot context HUB OS
91 };
92 
93 constexpr jint OS_APP_ID = -1;
94 constexpr jint INVALID_APP_ID = -2;
95 
96 constexpr jint MIN_APP_ID = 1;
97 constexpr jint MAX_APP_ID = 128;
98 
99 constexpr size_t MSG_HEADER_SIZE = 4;
100 constexpr size_t HEADER_FIELD_MSG_TYPE = 0;
101 constexpr size_t HEADER_FIELD_MSG_VERSION = 1;
102 constexpr size_t HEADER_FIELD_HUB_HANDLE = 2;
103 constexpr size_t HEADER_FIELD_APP_INSTANCE = 3;
104 
105 constexpr size_t HEADER_FIELD_LOAD_APP_ID_LO = MSG_HEADER_SIZE;
106 constexpr size_t HEADER_FIELD_LOAD_APP_ID_HI = MSG_HEADER_SIZE + 1;
107 constexpr size_t MSG_HEADER_SIZE_LOAD_APP = MSG_HEADER_SIZE + 2;
108 
109 jint getAppInstanceForAppId(uint64_t app_id);
110 int onMessageReceipt(const uint32_t *header,
111                      size_t headerLen,
112                      const char *msg,
113                      size_t msgLen);
114 void onHubReset(uint32_t hubId);
115 void queryHubForApps(uint32_t hubId);
116 void passOnOsResponse(uint32_t hubHandle,
117                       uint32_t msgType,
118                       TransactionResult result,
119                       const int8_t *additionalData,
120                       size_t additionalDataLen);
121 
122 bool closeLoadTxn(bool success, jint *appInstanceHandle);
123 void closeUnloadTxn(bool success);
124 int handleQueryAppsResponse(const std::vector<HubAppInfo> apps,
125                                uint32_t hubHandle);
126 
127 struct JniInfo {
128     JavaVM *vm;
129     jclass contextHubInfoClass;
130     jclass contextHubServiceClass;
131     jclass memoryRegionsClass;
132 
133     jobject jContextHubService;
134 
135     jmethodID msgReceiptCallBack;
136 
137     jmethodID contextHubInfoCtor;
138     jmethodID contextHubInfoSetId;
139     jmethodID contextHubInfoSetName;
140     jmethodID contextHubInfoSetVendor;
141     jmethodID contextHubInfoSetToolchain;
142     jmethodID contextHubInfoSetPlatformVersion;
143     jmethodID contextHubInfoSetStaticSwVersion;
144     jmethodID contextHubInfoSetToolchainVersion;
145     jmethodID contextHubInfoSetPeakMips;
146     jmethodID contextHubInfoSetStoppedPowerDrawMw;
147     jmethodID contextHubInfoSetSleepPowerDrawMw;
148     jmethodID contextHubInfoSetPeakPowerDrawMw;
149     jmethodID contextHubInfoSetSupportedSensors;
150     jmethodID contextHubInfoSetMemoryRegions;
151     jmethodID contextHubInfoSetMaxPacketLenBytes;
152 
153     jmethodID contextHubServiceMsgReceiptCallback;
154     jmethodID contextHubServiceAddAppInstance;
155     jmethodID contextHubServiceDeleteAppInstance;
156 };
157 
158 
159 
160 class TxnManager {
161 public:
TxnManager()162     TxnManager() {
163         mData = nullptr;
164         mIsPending = false;
165     }
166 
~TxnManager()167     ~TxnManager() {
168         closeTxn();
169     }
170 
addTxn(HubMessageType txnIdentifier,void * txnData)171     int addTxn(HubMessageType txnIdentifier, void *txnData) {
172         std::lock_guard<std::mutex>lock(mLock);
173         if (mIsPending) {
174             ALOGW("Transaction already found pending when trying to add a new one.");
175             return -1;
176         }
177         mIsPending = true;
178         mFirstTimeTxnCanBeCanceled = steady_clock::now() + kMinTransactionCancelTime;
179         mData = txnData;
180         mIdentifier = txnIdentifier;
181 
182         return 0;
183     }
184 
closeTxn()185     int closeTxn() {
186         std::lock_guard<std::mutex>lock(mLock);
187         closeTxnUnlocked();
188         return 0;
189     }
190 
isTxnPending()191     bool isTxnPending() {
192         std::lock_guard<std::mutex>lock(mLock);
193         return mIsPending;
194     }
195 
closeAnyStaleTxns()196     void closeAnyStaleTxns() {
197         std::lock_guard<std::mutex>lock(mLock);
198         if (mIsPending && steady_clock::now() >= mFirstTimeTxnCanBeCanceled) {
199             ALOGW("Stale transaction canceled");
200             closeTxnUnlocked();
201         }
202     }
203 
fetchTxnData(HubMessageType * id,void ** data)204     int fetchTxnData(HubMessageType *id, void **data) {
205         if (id == nullptr || data == nullptr) {
206             ALOGW("Null Params isNull{id, data} {%d, %d}",
207                   id == nullptr ? 1 : 0,
208                   data == nullptr ? 1 : 0);
209             return -1;
210         }
211 
212         std::lock_guard<std::mutex>lock(mLock);
213         if (!mIsPending) {
214             ALOGW("No Transactions pending");
215             return -1;
216         }
217 
218         *id = mIdentifier;
219         *data = mData;
220         return 0;
221     }
222 
223  private:
224     bool mIsPending;            // Is a transaction pending
225     std::mutex mLock;           // mutex for manager
226     HubMessageType mIdentifier; // What are we doing
227     void *mData;                // Details
228     steady_clock::time_point mFirstTimeTxnCanBeCanceled;
229 
230     // Only call this if you hold the lock.
closeTxnUnlocked()231     void closeTxnUnlocked() {
232         mIsPending = false;
233         free(mData);
234         mData = nullptr;
235     }
236 };
237 
238 
239 struct ContextHubServiceCallback : IContexthubCallback {
240     uint32_t mContextHubId;
241 
ContextHubServiceCallbackandroid::ContextHubServiceCallback242     ContextHubServiceCallback(uint32_t hubId) {
243         mContextHubId = hubId;
244     }
245 
handleClientMsgandroid::ContextHubServiceCallback246     virtual Return<void> handleClientMsg(const ContextHubMsg &msg) {
247         jint appHandle = getAppInstanceForAppId(msg.appName);
248         if (appHandle < 0) {
249             ALOGE("Filtering out message due to invalid App Instance.");
250         } else {
251             uint32_t msgHeader[MSG_HEADER_SIZE] = {};
252             msgHeader[HEADER_FIELD_MSG_TYPE] = msg.msgType;
253             msgHeader[HEADER_FIELD_HUB_HANDLE] = mContextHubId;
254             msgHeader[HEADER_FIELD_APP_INSTANCE] = appHandle;
255             onMessageReceipt(msgHeader,
256                              MSG_HEADER_SIZE,
257                              reinterpret_cast<const char *>(msg.msg.data()),
258                              msg.msg.size());
259         }
260 
261         return android::hardware::Void();
262     }
263 
handleHubEventandroid::ContextHubServiceCallback264     virtual Return<void> handleHubEvent(AsyncEventType evt) {
265         if (evt == AsyncEventType::RESTARTED) {
266             ALOGW("Context Hub handle %d restarted", mContextHubId);
267             onHubReset(mContextHubId);
268         } else {
269             ALOGW("Cannot handle event %u from hub %d", evt, mContextHubId);
270         }
271 
272         return android::hardware::Void();
273     }
274 
handleTxnResultandroid::ContextHubServiceCallback275     virtual Return<void> handleTxnResult(uint32_t txnId,
276                                          TransactionResult result) {
277         ALOGI("Handle transaction result , hubId %" PRIu32 ", txnId %" PRIu32 ", result %" PRIu32,
278               mContextHubId,
279               txnId,
280               result);
281 
282         switch(txnId) {
283             case CONTEXT_HUB_APPS_ENABLE:
284             case CONTEXT_HUB_APPS_DISABLE:
285                 passOnOsResponse(mContextHubId, txnId, result, nullptr, 0);
286                 break;
287 
288             case CONTEXT_HUB_UNLOAD_APP:
289                 closeUnloadTxn(result == TransactionResult::SUCCESS);
290                 passOnOsResponse(mContextHubId, txnId, result, nullptr, 0);
291                 break;
292 
293             case CONTEXT_HUB_LOAD_APP:
294                 {
295                     jint appInstanceHandle = INVALID_APP_ID;
296                     bool appRunningOnHub = (result == TransactionResult::SUCCESS);
297                     if (!(closeLoadTxn(appRunningOnHub, &appInstanceHandle))) {
298                         if (appRunningOnHub) {
299                             // Now we're in an odd situation.  Our nanoapp
300                             // is up and running on the Context Hub.  However,
301                             // something went wrong in our Service code so that
302                             // we're not able to properly track this nanoapp
303                             // in our Service code.  If we tell the Java layer
304                             // things are good, it's a lie because the handle
305                             // we give them will fail when used with the Service.
306                             // If we tell the Java layer this failed, it's kind
307                             // of a lie as well, since this nanoapp is running.
308                             //
309                             // We leave a more robust fix for later, and for
310                             // now just tell the user things have failed.
311                             //
312                             // TODO(b/30835981): Make this situation better.
313                             result = TransactionResult::FAILURE;
314                         }
315                     }
316 
317                     passOnOsResponse(mContextHubId,
318                                      txnId,
319                                      result,
320                                      reinterpret_cast<int8_t *>(&appInstanceHandle),
321                                      sizeof(appInstanceHandle));
322                     break;
323                 }
324 
325             default:
326                 ALOGI("unrecognized transction id %" PRIu32, txnId);
327                 break;
328         }
329         return android::hardware::Void();
330     }
331 
handleAppsInfoandroid::ContextHubServiceCallback332     virtual Return<void> handleAppsInfo(
333             const android::hardware::hidl_vec<HubAppInfo>& apps) {
334         TransactionResult result = TransactionResult::SUCCESS;
335         handleQueryAppsResponse(apps,mContextHubId);
336         passOnOsResponse(mContextHubId, CONTEXT_HUB_QUERY_APPS, result, nullptr, 0);
337         return android::hardware::Void();
338     }
339 
handleAppAbortandroid::ContextHubServiceCallback340     virtual Return<void> handleAppAbort(uint64_t appId, uint32_t abortCode) {
341         ALOGI("Handle app aport called from %" PRIx64 " with abort code %" PRIu32,
342             appId,
343             abortCode);
344 
345         // TODO: Plumb this to the clients interested in this app
346         return android::hardware::Void();
347     }
348 
setContextHubIdandroid::ContextHubServiceCallback349     void setContextHubId(uint32_t id) {
350         mContextHubId = id;
351     }
352 
getContextHubIdandroid::ContextHubServiceCallback353     uint32_t getContextHubId() {
354         return(mContextHubId);
355     }
356 };
357 
358 struct AppInstanceInfo {
359     HubAppInfo appInfo;          // returned from the HAL
360     uint64_t truncName;          // Possibly truncated name for logging
361     uint32_t hubHandle;          // Id of the hub this app is on
362     jint instanceId;             // system wide unique instance id - assigned
363 };
364 
365 struct ContextHubInfo {
366     int numHubs;
367     Vector<ContextHub> hubs;
368     sp<IContexthub> contextHub;
369 };
370 
371 struct ContextHubServiceDb {
372     int initialized;
373     ContextHubInfo hubInfo;
374     JniInfo jniInfo;
375     std::queue<jint> freeIds;
376     std::unordered_map<jint, AppInstanceInfo> appInstances;
377     TxnManager txnManager;
378     std::vector<ContextHubServiceCallback *> regCallBacks;
379 };
380 
381 ContextHubServiceDb db;
382 
getHubIdForHubHandle(int hubHandle,uint32_t * hubId)383 bool getHubIdForHubHandle(int hubHandle, uint32_t *hubId) {
384     if (hubHandle < 0 || hubHandle >= db.hubInfo.numHubs || hubId == nullptr) {
385         return false;
386     } else {
387         *hubId = db.hubInfo.hubs[hubHandle].hubId;
388         return true;
389     }
390 }
391 
getHubHandleForAppInstance(jint id)392 int getHubHandleForAppInstance(jint id) {
393     if (!db.appInstances.count(id)) {
394         ALOGD("%s: Cannot find app for app instance %" PRId32,
395               __FUNCTION__,
396               id);
397         return -1;
398     }
399 
400     return db.appInstances[id].hubHandle;
401 }
402 
getAppInstanceForAppId(uint64_t app_id)403 jint getAppInstanceForAppId(uint64_t app_id) {
404     auto end = db.appInstances.end();
405     for (auto current = db.appInstances.begin(); current != end; ++current) {
406         if (current->second.appInfo.appId == app_id) {
407             return current->first;
408         }
409     }
410     ALOGD("Cannot find app for app id %" PRIu64 ".", app_id);
411     return -1;
412 }
413 
getAppIdForAppInstance(jint id)414 uint64_t getAppIdForAppInstance(jint id) {
415     if (!db.appInstances.count(id)) {
416         return INVALID_APP_ID;
417     }
418     return db.appInstances[id].appInfo.appId;
419 }
420 
queryHubForApps(uint32_t hubId)421 void queryHubForApps(uint32_t hubId) {
422     Result r = db.hubInfo.contextHub->queryApps(hubId);
423     ALOGD("Sent query for apps to hub %" PRIu32 " with result %" PRIu32, hubId, r);
424 }
425 
sendQueryForApps()426 void sendQueryForApps() {
427     for (int i = 0; i < db.hubInfo.numHubs; i++ ) {
428         queryHubForApps(db.hubInfo.hubs[i].hubId);
429     }
430 }
431 
returnId(jint id)432 int returnId(jint id) {
433     // Note : This method is not thread safe.
434     // id returned is guaranteed to be in use
435     if (id >= 0) {
436         db.freeIds.push(id);
437         return 0;
438     }
439 
440     return -1;
441 }
442 
generateId()443 jint generateId() {
444     // Note : This method is not thread safe.
445     jint retVal = -1;
446 
447     if (!db.freeIds.empty()) {
448         retVal = db.freeIds.front();
449         db.freeIds.pop();
450     }
451 
452     return retVal;
453 }
454 
addAppInstance(const HubAppInfo * appInfo,uint32_t hubHandle,jint appInstanceHandle,JNIEnv * env)455 jint addAppInstance(const HubAppInfo *appInfo, uint32_t hubHandle,
456         jint appInstanceHandle, JNIEnv *env) {
457     // Not checking if the apps are indeed distinct
458     AppInstanceInfo entry;
459     assert(appInfo);
460 
461 
462     entry.appInfo = *appInfo;
463 
464     entry.instanceId = appInstanceHandle;
465     entry.truncName = appInfo->appId;
466     entry.hubHandle = hubHandle;
467     db.appInstances[appInstanceHandle] = entry;
468     // Finally - let the service know of this app instance, to populate
469     // the Java cache.
470     env->CallIntMethod(db.jniInfo.jContextHubService,
471                        db.jniInfo.contextHubServiceAddAppInstance,
472                        hubHandle, entry.instanceId,
473                        entry.truncName,
474                        entry.appInfo.version);
475 
476     const char *action = (db.appInstances.count(appInstanceHandle) == 0) ? "Added" : "Updated";
477     ALOGI("%s App 0x%" PRIx64 " on hub Handle %" PRId32
478           " as appInstance %" PRId32, action, entry.truncName,
479           entry.hubHandle, appInstanceHandle);
480 
481     return appInstanceHandle;
482 }
483 
deleteAppInstance(jint id,JNIEnv * env)484 int deleteAppInstance(jint id, JNIEnv *env) {
485     bool fullyDeleted = true;
486 
487     if (db.appInstances.count(id)) {
488         db.appInstances.erase(id);
489     } else {
490         ALOGW("Cannot delete App id (%" PRId32 ") from the JNI C++ cache", id);
491         fullyDeleted = false;
492     }
493     returnId(id);
494 
495     if ((env == nullptr) ||
496         (env->CallIntMethod(db.jniInfo.jContextHubService,
497                        db.jniInfo.contextHubServiceDeleteAppInstance,
498                        id) != 0)) {
499         ALOGW("Cannot delete App id (%" PRId32 ") from Java cache", id);
500         fullyDeleted = false;
501     }
502 
503     if (fullyDeleted) {
504         ALOGI("Deleted App id : %" PRId32, id);
505         return 0;
506     }
507     return -1;
508 }
509 
startLoadAppTxn(uint64_t appId,int hubHandle)510 int startLoadAppTxn(uint64_t appId, int hubHandle) {
511     AppInstanceInfo *txnInfo = new AppInstanceInfo();
512     jint instanceId = generateId();
513 
514     if (!txnInfo || instanceId < 0) {
515         returnId(instanceId);
516         free(txnInfo);
517         return -1;
518     }
519 
520     txnInfo->truncName = appId;
521     txnInfo->hubHandle = hubHandle;
522     txnInfo->instanceId = instanceId;
523 
524     txnInfo->appInfo.appId = appId;
525     txnInfo->appInfo.version = -1; // Awaited
526 
527     if (db.txnManager.addTxn(CONTEXT_HUB_LOAD_APP, txnInfo) != 0) {
528         returnId(instanceId);
529         free(txnInfo);
530         return -1;
531     }
532 
533     return 0;
534 }
535 
startUnloadAppTxn(jint appInstanceHandle)536 int startUnloadAppTxn(jint appInstanceHandle) {
537     jint *txnData = new(jint);
538     if (!txnData) {
539         ALOGW("Cannot allocate memory to start unload transaction");
540         return -1;
541     }
542 
543     *txnData = appInstanceHandle;
544 
545     if (db.txnManager.addTxn(CONTEXT_HUB_UNLOAD_APP, txnData) != 0) {
546         free(txnData);
547         ALOGW("Cannot start transaction to unload app");
548         return -1;
549     }
550 
551     return 0;
552 }
553 
getHubsCb(const::android::hardware::hidl_vec<ContextHub> & hubs)554 void getHubsCb(const ::android::hardware::hidl_vec<ContextHub>& hubs)  {
555     for (size_t i = 0; i < hubs.size(); i++) {
556         db.hubInfo.hubs.push_back(hubs[i]);
557     }
558 }
559 
initContextHubService()560 void initContextHubService() {
561     db.hubInfo.numHubs = 0;
562 
563     db.hubInfo.contextHub = IContexthub::getService();
564 
565     if (db.hubInfo.contextHub == nullptr) {
566         ALOGE("Could not load context hub hal");
567     } else {
568         ALOGI("Loaded context hub hal, isRemote %s", db.hubInfo.contextHub->isRemote() ? "TRUE" : "FALSE");
569     }
570 
571     // Prep for storing app info
572     for (jint i = MIN_APP_ID; i <= MAX_APP_ID; i++) {
573         db.freeIds.push(i);
574     }
575 
576     if (db.hubInfo.contextHub != nullptr) {
577         std::function<void(const ::android::hardware::hidl_vec<ContextHub>& hubs)> f = getHubsCb;
578         if(!db.hubInfo.contextHub->getHubs(f).isOk()) {
579             ALOGW("GetHubs Failed! transport error.");
580             return;
581         };
582 
583         int retNumHubs = db.hubInfo.hubs.size();
584         ALOGD("ContextHubModule returned %d hubs ", retNumHubs);
585         db.hubInfo.numHubs = retNumHubs;
586 
587         for (int i = 0; i < db.hubInfo.numHubs; i++) {
588             ALOGI("Subscribing to hubHandle %d", i);
589 
590             ContextHubServiceCallback *callBackPtr =
591                 new ContextHubServiceCallback(db.hubInfo.hubs[i].hubId);
592             db.hubInfo.contextHub->registerCallback(db.hubInfo.hubs[i].hubId,
593                                                     callBackPtr);
594             db.regCallBacks.push_back(callBackPtr);
595         }
596 
597         sendQueryForApps();
598 
599     } else {
600         ALOGW("No Context Hub Module present");
601     }
602 }
603 
onHubReset(uint32_t hubId)604 void onHubReset(uint32_t hubId) {
605     TransactionResult result = TransactionResult::SUCCESS;
606     db.txnManager.closeTxn();
607     // TODO : Expose this through an api
608     passOnOsResponse(hubId, CONTEXT_HUB_OS_REBOOT, result, nullptr, 0);
609     queryHubForApps(hubId);
610 }
611 
onMessageReceipt(const uint32_t * header,size_t headerLen,const char * msg,size_t msgLen)612 int onMessageReceipt(const uint32_t *header,
613                      size_t headerLen,
614                      const char *msg,
615                      size_t msgLen) {
616     JNIEnv *env;
617 
618     if ((db.jniInfo.vm)->AttachCurrentThread(&env, nullptr) != JNI_OK) {
619       return -1;
620     }
621 
622     jbyteArray jmsg = env->NewByteArray(msgLen);
623     if (jmsg == nullptr) {
624         ALOGW("Can't allocate %zu byte array", msgLen);
625         return -1;
626     }
627     jintArray jheader = env->NewIntArray(headerLen);
628     if (jheader == nullptr) {
629         env->DeleteLocalRef(jmsg);
630         ALOGW("Can't allocate %zu int array", headerLen);
631         return -1;
632     }
633 
634     env->SetByteArrayRegion(jmsg, 0, msgLen, reinterpret_cast<const jbyte *>(msg));
635     env->SetIntArrayRegion(jheader, 0, headerLen, reinterpret_cast<const jint *>(header));
636 
637     int ret = (env->CallIntMethod(db.jniInfo.jContextHubService,
638                                   db.jniInfo.contextHubServiceMsgReceiptCallback,
639                                   jheader,
640                                   jmsg) != 0);
641     env->DeleteLocalRef(jmsg);
642     env->DeleteLocalRef(jheader);
643 
644     return ret;
645 }
646 
handleQueryAppsResponse(const std::vector<HubAppInfo> apps,uint32_t hubHandle)647 int handleQueryAppsResponse(const std::vector<HubAppInfo> apps,
648                                uint32_t hubHandle) {
649     JNIEnv *env;
650     if ((db.jniInfo.vm)->AttachCurrentThread(&env, nullptr) != JNI_OK) {
651             return -1;
652     }
653 
654     int numApps = apps.size();
655 
656     // We use this information to sync our JNI and Java caches of nanoapp info.
657     // We want to accomplish two things here:
658     // 1) Remove entries from our caches which are stale, and pertained to
659     //    apps no longer running on Context Hub.
660     // 2) Populate our caches with the latest information of all these apps.
661 
662     // We make a couple of assumptions here:
663     // A) The JNI and Java caches are in sync with each other (this isn't
664     //    necessarily true; any failure of a single call into Java land to
665     //    update its cache will leave that cache in a bad state.  For NYC,
666     //    we're willing to tolerate this for now).
667     // B) The total number of apps is relatively small, so horribly inefficent
668     //    algorithms aren't too painful.
669     // C) We're going to call this relatively infrequently, so its inefficency
670     //    isn't a big impact.
671 
672 
673     // (1).  Looking for stale cache entries.  Yes, this is O(N^2).  See
674     // assumption (B).  Per assumption (A), it is sufficient to iterate
675     // over just the JNI cache.
676     auto end = db.appInstances.end();
677     for (auto current = db.appInstances.begin(); current != end; ) {
678         AppInstanceInfo cacheEntry = current->second;
679         // We perform our iteration here because if we call
680         // delete_app_instance() below, it will erase() this entry.
681         current++;
682         bool entryIsStale = true;
683         for (int i = 0; i < numApps; i++) {
684             if (apps[i].appId == cacheEntry.appInfo.appId) {
685                 // We found a match; this entry is current.
686                 entryIsStale = false;
687                 break;
688             }
689         }
690 
691         if (entryIsStale) {
692             deleteAppInstance(cacheEntry.instanceId, env);
693         }
694     }
695 
696     // (2).  Update our caches with the latest.
697     for (int i = 0; i < numApps; i++) {
698         // We will only have one instance of the app
699         // TODO : Change this logic once we support multiple instances of the same app
700         jint appInstance = getAppInstanceForAppId(apps[i].appId);
701         if (appInstance == -1) {
702             // This is a previously unknown app, let's allocate an "id" for it.
703             appInstance = generateId();
704         }
705         addAppInstance(&apps[i], hubHandle, appInstance, env);
706     }
707     return 0;
708 }
709 
710 // TODO(b/30807327): Do not use raw bytes for additional data.  Use the
711 //     JNI interfaces for the appropriate types.
passOnOsResponse(uint32_t hubHandle,uint32_t msgType,TransactionResult result,const int8_t * additionalData,size_t additionalDataLen)712 void passOnOsResponse(uint32_t hubHandle,
713                       uint32_t msgType,
714                       TransactionResult result,
715                       const int8_t *additionalData,
716                       size_t additionalDataLen) {
717     JNIEnv *env;
718 
719     if ((db.jniInfo.vm)->AttachCurrentThread(&env, nullptr) != JNI_OK) {
720         ALOGW("Cannot latch to JNI env, dropping OS response %" PRIu32,
721               msgType);
722         return;
723     }
724 
725     uint32_t header[MSG_HEADER_SIZE];
726     memset(header, 0, sizeof(header));
727 
728     if (!additionalData) {
729         additionalDataLen = 0; // clamp
730     }
731     int msgLen = 1 + additionalDataLen;
732 
733     int8_t *msg = new int8_t[msgLen];
734 
735     if (!msg) {
736         ALOGW("Unexpected : Ran out of memory, cannot send response");
737         return;
738     }
739 
740     header[HEADER_FIELD_MSG_TYPE] = msgType;
741     header[HEADER_FIELD_MSG_VERSION] = 0;
742     header[HEADER_FIELD_HUB_HANDLE] = hubHandle;
743     header[HEADER_FIELD_APP_INSTANCE] = OS_APP_ID;
744 
745     // Due to API constraints, at the moment we can't change the fact that
746     // we're changing our 4-byte response to a 1-byte value.  But we can prevent
747     // the possible change in sign (and thus meaning) that would happen from
748     // a naive cast.  Further, we can log when we're losing part of the value.
749     // TODO(b/30918279): Don't truncate this result.
750     int8_t truncatedResult;
751     truncatedResult = static_cast<int8_t>(result);
752     msg[0] = truncatedResult;
753 
754     if (additionalData) {
755         memcpy(&msg[1], additionalData, additionalDataLen);
756     }
757 
758     jbyteArray jmsg = env->NewByteArray(msgLen);
759     jintArray jheader = env->NewIntArray(arraysize(header));
760 
761     env->SetByteArrayRegion(jmsg, 0, msgLen, reinterpret_cast<jbyte *>(msg));
762     env->SetIntArrayRegion(jheader, 0, arraysize(header), reinterpret_cast<jint *>(header));
763 
764     ALOGI("Passing msg type %" PRIu32 " from app %" PRIu32 " from hub %" PRIu32,
765           header[HEADER_FIELD_MSG_TYPE],
766           header[HEADER_FIELD_APP_INSTANCE],
767           header[HEADER_FIELD_HUB_HANDLE]);
768 
769     env->CallIntMethod(db.jniInfo.jContextHubService,
770                        db.jniInfo.contextHubServiceMsgReceiptCallback,
771                        jheader,
772                        jmsg);
773 
774     env->DeleteLocalRef(jmsg);
775     env->DeleteLocalRef(jheader);
776 
777     delete[] msg;
778 }
779 
closeUnloadTxn(bool success)780 void closeUnloadTxn(bool success) {
781     void *txnData = nullptr;
782     HubMessageType txnId;
783 
784     if (success && db.txnManager.fetchTxnData(&txnId, &txnData) == 0 &&
785         txnId == CONTEXT_HUB_UNLOAD_APP) {
786         JNIEnv *env;
787         if ((db.jniInfo.vm)->AttachCurrentThread(&env, nullptr) != JNI_OK) {
788             ALOGW("Could not attach to JVM !");
789             env = nullptr;
790         }
791         jint handle = *reinterpret_cast<jint *>(txnData);
792         deleteAppInstance(handle, env);
793     } else {
794         ALOGW("Could not unload the app successfully ! success %d, txnData %p",
795               success,
796               txnData);
797     }
798 
799     db.txnManager.closeTxn();
800 }
801 
closeLoadTxn(bool success,jint * appInstanceHandle)802 bool closeLoadTxn(bool success, jint *appInstanceHandle) {
803     void *txnData;
804     HubMessageType txnId;
805 
806     if (success && db.txnManager.fetchTxnData(&txnId, &txnData) == 0 &&
807         txnId == CONTEXT_HUB_LOAD_APP) {
808         AppInstanceInfo *info = static_cast<AppInstanceInfo *>(txnData);
809         *appInstanceHandle = info->instanceId;
810 
811         JNIEnv *env;
812         if ((db.jniInfo.vm)->AttachCurrentThread(&env, nullptr) == JNI_OK) {
813             addAppInstance(&info->appInfo, info->hubHandle, info->instanceId, env);
814         } else {
815             ALOGW("Could not attach to JVM !");
816             success = false;
817         }
818         // While we just called addAppInstance above, our info->appInfo was
819         // incomplete (for example, the 'version' is hardcoded to -1).  So we
820         // trigger an additional query to the CHRE, so we'll be able to get
821         // all the app "info", and have our JNI and Java caches with the
822         // full information.
823         sendQueryForApps();
824     } else {
825         ALOGW("Could not load the app successfully ! Unexpected failure");
826         *appInstanceHandle = INVALID_APP_ID;
827         success = false;
828     }
829 
830     db.txnManager.closeTxn();
831     return success;
832 }
833 
initJni(JNIEnv * env,jobject instance)834 int initJni(JNIEnv *env, jobject instance) {
835     if (env->GetJavaVM(&db.jniInfo.vm) != JNI_OK) {
836         return -1;
837     }
838 
839     db.jniInfo.jContextHubService = env->NewGlobalRef(instance);
840 
841     db.jniInfo.contextHubInfoClass =
842             env->FindClass("android/hardware/location/ContextHubInfo");
843     db.jniInfo.contextHubServiceClass =
844             env->FindClass("com/android/server/location/ContextHubService");
845 
846     db.jniInfo.memoryRegionsClass =
847             env->FindClass("android/hardware/location/MemoryRegion");
848 
849     db.jniInfo.contextHubInfoCtor =
850             env->GetMethodID(db.jniInfo.contextHubInfoClass, "<init>", "()V");
851     db.jniInfo.contextHubInfoSetId =
852             env->GetMethodID(db.jniInfo.contextHubInfoClass, "setId", "(I)V");
853     db.jniInfo.contextHubInfoSetName =
854             env->GetMethodID(db.jniInfo.contextHubInfoClass, "setName", "(Ljava/lang/String;)V");
855     db.jniInfo.contextHubInfoSetVendor =
856             env->GetMethodID(db.jniInfo.contextHubInfoClass,
857                              "setVendor",
858                              "(Ljava/lang/String;)V");
859     db.jniInfo.contextHubInfoSetToolchain =
860             env->GetMethodID(db.jniInfo.contextHubInfoClass,
861                              "setToolchain",
862                              "(Ljava/lang/String;)V");
863     db.jniInfo.contextHubInfoSetPlatformVersion =
864             env->GetMethodID(db.jniInfo.contextHubInfoClass,
865                              "setPlatformVersion",
866                              "(I)V");
867     db.jniInfo.contextHubInfoSetStaticSwVersion =
868             env->GetMethodID(db.jniInfo.contextHubInfoClass,
869                              "setStaticSwVersion",
870                              "(I)V");
871     db.jniInfo.contextHubInfoSetToolchainVersion =
872             env->GetMethodID(db.jniInfo.contextHubInfoClass,
873                              "setToolchainVersion",
874                              "(I)V");
875     db.jniInfo.contextHubInfoSetPeakMips =
876             env->GetMethodID(db.jniInfo.contextHubInfoClass,
877                              "setPeakMips",
878                              "(F)V");
879     db.jniInfo.contextHubInfoSetStoppedPowerDrawMw =
880             env->GetMethodID(db.jniInfo.contextHubInfoClass,
881                              "setStoppedPowerDrawMw",
882                              "(F)V");
883     db.jniInfo.contextHubInfoSetSleepPowerDrawMw =
884             env->GetMethodID(db.jniInfo.contextHubInfoClass,
885                              "setSleepPowerDrawMw",
886                              "(F)V");
887     db.jniInfo.contextHubInfoSetPeakPowerDrawMw =
888             env->GetMethodID(db.jniInfo.contextHubInfoClass,
889                              "setPeakPowerDrawMw",
890                              "(F)V");
891     db.jniInfo.contextHubInfoSetSupportedSensors =
892             env->GetMethodID(db.jniInfo.contextHubInfoClass,
893                              "setSupportedSensors",
894                              "([I)V");
895     db.jniInfo.contextHubInfoSetMemoryRegions =
896             env->GetMethodID(db.jniInfo.contextHubInfoClass,
897                              "setMemoryRegions",
898                              "([Landroid/hardware/location/MemoryRegion;)V");
899     db.jniInfo.contextHubInfoSetMaxPacketLenBytes =
900              env->GetMethodID(db.jniInfo.contextHubInfoClass,
901                               "setMaxPacketLenBytes",
902                               "(I)V");
903     db.jniInfo.contextHubServiceMsgReceiptCallback =
904             env->GetMethodID(db.jniInfo.contextHubServiceClass,
905                              "onMessageReceipt",
906                              "([I[B)I");
907     db.jniInfo.contextHubInfoSetName =
908             env->GetMethodID(db.jniInfo.contextHubInfoClass,
909                              "setName",
910                              "(Ljava/lang/String;)V");
911     db.jniInfo.contextHubServiceAddAppInstance =
912                  env->GetMethodID(db.jniInfo.contextHubServiceClass,
913                                   "addAppInstance",
914                                   "(IIJI)I");
915     db.jniInfo.contextHubServiceDeleteAppInstance =
916                  env->GetMethodID(db.jniInfo.contextHubServiceClass,
917                                   "deleteAppInstance",
918                                   "(I)I");
919 
920     return 0;
921 }
922 
constructJContextHubInfo(JNIEnv * env,const ContextHub & hub)923 jobject constructJContextHubInfo(JNIEnv *env, const ContextHub &hub) {
924     jstring jstrBuf;
925     jintArray jintBuf;
926     jobjectArray jmemBuf;
927 
928     jobject jHub = env->NewObject(db.jniInfo.contextHubInfoClass,
929                                   db.jniInfo.contextHubInfoCtor);
930     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetId, hub.hubId);
931 
932     jstrBuf = env->NewStringUTF(hub.name.c_str());
933     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetName, jstrBuf);
934     env->DeleteLocalRef(jstrBuf);
935 
936     jstrBuf = env->NewStringUTF(hub.vendor.c_str());
937     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetVendor, jstrBuf);
938     env->DeleteLocalRef(jstrBuf);
939 
940     jstrBuf = env->NewStringUTF(hub.toolchain.c_str());
941     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetToolchain, jstrBuf);
942     env->DeleteLocalRef(jstrBuf);
943 
944     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetPlatformVersion, hub.platformVersion);
945     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetToolchainVersion, hub.toolchainVersion);
946     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetPeakMips, hub.peakMips);
947     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetStoppedPowerDrawMw,
948                         hub.stoppedPowerDrawMw);
949     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetSleepPowerDrawMw,
950                         hub.sleepPowerDrawMw);
951     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetPeakPowerDrawMw,
952                         hub.peakPowerDrawMw);
953     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetMaxPacketLenBytes,
954                         hub.maxSupportedMsgLen);
955 
956 
957     jintBuf = env->NewIntArray(hub.connectedSensors.size());
958     int *connectedSensors = new int[hub.connectedSensors.size()];
959 
960     if (!connectedSensors) {
961       ALOGW("Cannot allocate memory! Unexpected");
962       assert(false);
963     } else {
964       for (unsigned int i = 0; i < hub.connectedSensors.size(); i++) {
965         // TODO :: Populate connected sensors.
966         //connectedSensors[i] = hub.connectedSensors[i].sensorType;
967         connectedSensors[i] = 0;
968       }
969     }
970 
971     env->SetIntArrayRegion(jintBuf, 0, hub.connectedSensors.size(),
972                            connectedSensors);
973 
974     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetSupportedSensors, jintBuf);
975     env->DeleteLocalRef(jintBuf);
976 
977     // We are not getting the memory regions from the CH Hal - change this when it is available
978     jmemBuf = env->NewObjectArray(0, db.jniInfo.memoryRegionsClass, nullptr);
979     // Note the zero size above. We do not need to set any elements
980     env->CallVoidMethod(jHub, db.jniInfo.contextHubInfoSetMemoryRegions, jmemBuf);
981     env->DeleteLocalRef(jmemBuf);
982 
983 
984     delete[] connectedSensors;
985     return jHub;
986 }
987 
nativeInitialize(JNIEnv * env,jobject instance)988 jobjectArray nativeInitialize(JNIEnv *env, jobject instance) {
989     jobject hub;
990     jobjectArray retArray;
991 
992     if (initJni(env, instance) < 0) {
993         return nullptr;
994     }
995 
996     initContextHubService();
997 
998     if (db.hubInfo.numHubs > 1) {
999         ALOGW("Clamping the number of hubs to 1");
1000         db.hubInfo.numHubs = 1;
1001     }
1002 
1003     retArray = env->NewObjectArray(db.hubInfo.numHubs, db.jniInfo.contextHubInfoClass, nullptr);
1004 
1005     for(int i = 0; i < db.hubInfo.numHubs; i++) {
1006         hub = constructJContextHubInfo(env, db.hubInfo.hubs[i]);
1007         env->SetObjectArrayElement(retArray, i, hub);
1008     }
1009 
1010     return retArray;
1011 }
1012 
sendLoadNanoAppRequest(uint32_t hubId,jbyte * data,size_t dataBufferLength)1013 Result sendLoadNanoAppRequest(uint32_t hubId,
1014                               jbyte *data,
1015                               size_t dataBufferLength) {
1016     auto header = reinterpret_cast<const NanoAppBinaryHeader *>(data);
1017     Result result;
1018 
1019     if (dataBufferLength < sizeof(NanoAppBinaryHeader)) {
1020         ALOGE("Got short NanoApp, length %zu", dataBufferLength);
1021         result = Result::BAD_PARAMS;
1022     } else if (header->headerVersion != htole32(kNanoAppBinaryHeaderVersion)) {
1023         ALOGE("Got unexpected NanoApp header version %" PRIu32,
1024               letoh32(header->headerVersion));
1025         result = Result::BAD_PARAMS;
1026     } else {
1027         NanoAppBinary nanoapp;
1028 
1029         // Data from the common nanoapp header goes into explicit fields
1030         nanoapp.appId      = letoh64(header->appId);
1031         nanoapp.appVersion = letoh32(header->appVersion);
1032         nanoapp.flags      = letoh32(header->flags);
1033         nanoapp.targetChreApiMajorVersion = header->targetChreApiMajorVersion;
1034         nanoapp.targetChreApiMinorVersion = header->targetChreApiMinorVersion;
1035 
1036         // Everything past the header goes in customBinary
1037         auto dataBytes = reinterpret_cast<const uint8_t *>(data);
1038         std::vector<uint8_t> customBinary(
1039             dataBytes + sizeof(NanoAppBinaryHeader),
1040             dataBytes + dataBufferLength);
1041         nanoapp.customBinary = std::move(customBinary);
1042 
1043         ALOGW("Calling Load NanoApp on hub %d", hubId);
1044         result = db.hubInfo.contextHub->loadNanoApp(hubId,
1045                                                     nanoapp,
1046                                                     CONTEXT_HUB_LOAD_APP);
1047     }
1048 
1049     return result;
1050 }
1051 
nativeSendMessage(JNIEnv * env,jobject instance,jintArray header_,jbyteArray data_)1052 jint nativeSendMessage(JNIEnv *env,
1053                        jobject instance,
1054                        jintArray header_,
1055                        jbyteArray data_) {
1056     // With the new binderized HAL definition, this function can be made much simpler.
1057     // All the magic can be removed. This is not however needed for the default implementation
1058     // TODO :: Change the JNI interface to conform to the new HAL interface and clean up this
1059     // function
1060     jint retVal = -1; // Default to failure
1061 
1062     jint *header = env->GetIntArrayElements(header_, 0);
1063     size_t numHeaderElements = env->GetArrayLength(header_);
1064     jbyte *data = env->GetByteArrayElements(data_, 0);
1065     size_t dataBufferLength = env->GetArrayLength(data_);
1066 
1067     if (numHeaderElements < MSG_HEADER_SIZE) {
1068         ALOGW("Malformed header len");
1069         return -1;
1070     }
1071 
1072     jint appInstanceHandle = header[HEADER_FIELD_APP_INSTANCE];
1073     uint32_t msgType = header[HEADER_FIELD_MSG_TYPE];
1074     int hubHandle = -1;
1075     uint64_t appId;
1076 
1077     if (msgType == CONTEXT_HUB_UNLOAD_APP) {
1078         hubHandle = getHubHandleForAppInstance(appInstanceHandle);
1079     } else if (msgType == CONTEXT_HUB_LOAD_APP) {
1080         if (numHeaderElements < MSG_HEADER_SIZE_LOAD_APP) {
1081             return -1;
1082         }
1083         uint64_t appIdLo = header[HEADER_FIELD_LOAD_APP_ID_LO];
1084         uint64_t appIdHi = header[HEADER_FIELD_LOAD_APP_ID_HI];
1085         appId = appIdHi << 32 | appIdLo;
1086 
1087         hubHandle = header[HEADER_FIELD_HUB_HANDLE];
1088     } else {
1089         hubHandle = header[HEADER_FIELD_HUB_HANDLE];
1090     }
1091 
1092     uint32_t hubId = -1;
1093     if (!getHubIdForHubHandle(hubHandle, &hubId)) {
1094         ALOGD("Invalid hub Handle %d", hubHandle);
1095         return -1;
1096     }
1097 
1098     if (msgType == CONTEXT_HUB_LOAD_APP ||
1099         msgType == CONTEXT_HUB_UNLOAD_APP) {
1100 
1101         db.txnManager.closeAnyStaleTxns();
1102 
1103         if (db.txnManager.isTxnPending()) {
1104             // TODO : There is a race conditio
1105             ALOGW("Cannot load or unload app while a transaction is pending !");
1106             return -1;
1107         } else if (msgType == CONTEXT_HUB_LOAD_APP) {
1108             if (startLoadAppTxn(appId, hubHandle) != 0) {
1109                 ALOGW("Cannot Start Load Transaction");
1110                 return -1;
1111             }
1112         } else if (msgType == CONTEXT_HUB_UNLOAD_APP) {
1113             if (startUnloadAppTxn(appInstanceHandle) != 0) {
1114                 ALOGW("Cannot Start UnLoad Transaction");
1115                 return -1;
1116             }
1117         }
1118     }
1119 
1120     Result result;
1121 
1122     if (msgType == CONTEXT_HUB_UNLOAD_APP) {
1123         ALOGW("Calling UnLoad NanoApp for app %" PRIx64 " on hub %" PRIu32,
1124               db.appInstances[appInstanceHandle].appInfo.appId,
1125               hubId);
1126         result = db.hubInfo.contextHub->unloadNanoApp(
1127                 hubId, db.appInstances[appInstanceHandle].appInfo.appId, CONTEXT_HUB_UNLOAD_APP);
1128     } else {
1129         if (appInstanceHandle == OS_APP_ID) {
1130             if (msgType == CONTEXT_HUB_LOAD_APP) {
1131                 result = sendLoadNanoAppRequest(hubId, data, dataBufferLength);
1132             } else if (msgType == CONTEXT_HUB_QUERY_APPS) {
1133                 result = db.hubInfo.contextHub->queryApps(hubId);
1134             } else {
1135                 ALOGD("Dropping OS addresses message of type - %" PRIu32, msgType);
1136                 result = Result::BAD_PARAMS;
1137             }
1138         } else {
1139             appId = getAppIdForAppInstance(appInstanceHandle);
1140             if (appId == static_cast<uint64_t>(INVALID_APP_ID)) {
1141                 ALOGD("Cannot find application instance %d", appInstanceHandle);
1142                 result = Result::BAD_PARAMS;
1143             } else if (hubHandle != getHubHandleForAppInstance(appInstanceHandle)) {
1144                 ALOGE("Given hubHandle (%d) doesn't match expected for app instance (%d)",
1145                       hubHandle,
1146                       getHubHandleForAppInstance(appInstanceHandle));
1147                 result = Result::BAD_PARAMS;
1148             } else {
1149                 ContextHubMsg msg;
1150                 msg.appName = appId;
1151                 msg.msgType = msgType;
1152                 msg.msg.setToExternal((unsigned char *)data, dataBufferLength);
1153 
1154                 ALOGW("Sending msg of type %" PRIu32 " len %zu to app %" PRIx64 " on hub %" PRIu32,
1155                        msgType,
1156                        dataBufferLength,
1157                        appId,
1158                        hubId);
1159                 result = db.hubInfo.contextHub->sendMessageToHub(hubId, msg);
1160             }
1161         }
1162     }
1163 
1164     if (result != Result::OK) {
1165         ALOGD("Send Message failure - %d", retVal);
1166         if (msgType == CONTEXT_HUB_LOAD_APP) {
1167             jint ignored;
1168             closeLoadTxn(false, &ignored);
1169         } else if (msgType == CONTEXT_HUB_UNLOAD_APP) {
1170             closeUnloadTxn(false);
1171         }
1172     } else {
1173         retVal = 0;
1174     }
1175 
1176     env->ReleaseIntArrayElements(header_, header, 0);
1177     env->ReleaseByteArrayElements(data_, data, 0);
1178 
1179     return retVal;
1180 }
1181 
1182 //--------------------------------------------------------------------------------------------------
1183 //
1184 const JNINativeMethod gContextHubServiceMethods[] = {
1185     {"nativeInitialize",
1186             "()[Landroid/hardware/location/ContextHubInfo;",
1187             reinterpret_cast<void*>(nativeInitialize)},
1188     {"nativeSendMessage",
1189             "([I[B)I",
1190             reinterpret_cast<void*>(nativeSendMessage)}
1191 };
1192 
register_android_server_location_ContextHubService(JNIEnv * env)1193 int register_android_server_location_ContextHubService(JNIEnv *env)
1194 {
1195     RegisterMethodsOrDie(env, "com/android/server/location/ContextHubService",
1196             gContextHubServiceMethods, NELEM(gContextHubServiceMethods));
1197 
1198     return 0;
1199 }
1200 
1201 }//namespace android
1202