xref: /aosp_15_r20/system/chre/platform/shared/host_link.cc (revision 84e339476a462649f82315436d70fd732297a399)
1*84e33947SAndroid Build Coastguard Worker /*
2*84e33947SAndroid Build Coastguard Worker  * Copyright (C) 2022 The Android Open Source Project
3*84e33947SAndroid Build Coastguard Worker  *
4*84e33947SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*84e33947SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*84e33947SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*84e33947SAndroid Build Coastguard Worker  *
8*84e33947SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*84e33947SAndroid Build Coastguard Worker  *
10*84e33947SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*84e33947SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*84e33947SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*84e33947SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*84e33947SAndroid Build Coastguard Worker  * limitations under the License.
15*84e33947SAndroid Build Coastguard Worker  */
16*84e33947SAndroid Build Coastguard Worker 
17*84e33947SAndroid Build Coastguard Worker #include "chre/core/event_loop_manager.h"
18*84e33947SAndroid Build Coastguard Worker #include "chre/platform/shared/host_protocol_chre.h"
19*84e33947SAndroid Build Coastguard Worker #include "chre/platform/shared/nanoapp_load_manager.h"
20*84e33947SAndroid Build Coastguard Worker 
21*84e33947SAndroid Build Coastguard Worker namespace chre {
22*84e33947SAndroid Build Coastguard Worker 
23*84e33947SAndroid Build Coastguard Worker NanoappLoadManager gLoadManager;
24*84e33947SAndroid Build Coastguard Worker 
getLoadManager()25*84e33947SAndroid Build Coastguard Worker inline NanoappLoadManager &getLoadManager() {
26*84e33947SAndroid Build Coastguard Worker   return gLoadManager;
27*84e33947SAndroid Build Coastguard Worker }
28*84e33947SAndroid Build Coastguard Worker 
handleDebugConfiguration(const fbs::DebugConfiguration * debugConfiguration)29*84e33947SAndroid Build Coastguard Worker void HostMessageHandlers::handleDebugConfiguration(
30*84e33947SAndroid Build Coastguard Worker     const fbs::DebugConfiguration *debugConfiguration) {
31*84e33947SAndroid Build Coastguard Worker   EventLoopManagerSingleton::get()
32*84e33947SAndroid Build Coastguard Worker       ->getSystemHealthMonitor()
33*84e33947SAndroid Build Coastguard Worker       .setFatalErrorOnCheckFailure(
34*84e33947SAndroid Build Coastguard Worker           debugConfiguration->health_monitor_failure_crash());
35*84e33947SAndroid Build Coastguard Worker }
36*84e33947SAndroid Build Coastguard Worker 
finishLoadingNanoappCallback(SystemCallbackType,UniquePtr<LoadNanoappCallbackData> && cbData)37*84e33947SAndroid Build Coastguard Worker void HostMessageHandlers::finishLoadingNanoappCallback(
38*84e33947SAndroid Build Coastguard Worker     SystemCallbackType /*type*/, UniquePtr<LoadNanoappCallbackData> &&cbData) {
39*84e33947SAndroid Build Coastguard Worker   constexpr size_t kInitialBufferSize = 48;
40*84e33947SAndroid Build Coastguard Worker   ChreFlatBufferBuilder builder(kInitialBufferSize);
41*84e33947SAndroid Build Coastguard Worker 
42*84e33947SAndroid Build Coastguard Worker   CHRE_ASSERT(cbData != nullptr);
43*84e33947SAndroid Build Coastguard Worker 
44*84e33947SAndroid Build Coastguard Worker   EventLoop &eventLoop = EventLoopManagerSingleton::get()->getEventLoop();
45*84e33947SAndroid Build Coastguard Worker   bool success = false;
46*84e33947SAndroid Build Coastguard Worker 
47*84e33947SAndroid Build Coastguard Worker   if (cbData->nanoapp->isLoaded()) {
48*84e33947SAndroid Build Coastguard Worker     success = eventLoop.startNanoapp(cbData->nanoapp);
49*84e33947SAndroid Build Coastguard Worker   } else {
50*84e33947SAndroid Build Coastguard Worker     LOGE("Nanoapp is not loaded");
51*84e33947SAndroid Build Coastguard Worker   }
52*84e33947SAndroid Build Coastguard Worker 
53*84e33947SAndroid Build Coastguard Worker   if (cbData->sendFragmentResponse) {
54*84e33947SAndroid Build Coastguard Worker     sendFragmentResponse(cbData->hostClientId, cbData->transactionId,
55*84e33947SAndroid Build Coastguard Worker                          cbData->fragmentId, success);
56*84e33947SAndroid Build Coastguard Worker   }
57*84e33947SAndroid Build Coastguard Worker }
58*84e33947SAndroid Build Coastguard Worker 
loadNanoappData(uint16_t hostClientId,uint32_t transactionId,uint64_t appId,uint32_t appVersion,uint32_t appFlags,uint32_t targetApiVersion,const void * buffer,size_t bufferLen,uint32_t fragmentId,size_t appBinaryLen,bool respondBeforeStart)59*84e33947SAndroid Build Coastguard Worker void HostMessageHandlers::loadNanoappData(
60*84e33947SAndroid Build Coastguard Worker     uint16_t hostClientId, uint32_t transactionId, uint64_t appId,
61*84e33947SAndroid Build Coastguard Worker     uint32_t appVersion, uint32_t appFlags, uint32_t targetApiVersion,
62*84e33947SAndroid Build Coastguard Worker     const void *buffer, size_t bufferLen, uint32_t fragmentId,
63*84e33947SAndroid Build Coastguard Worker     size_t appBinaryLen, bool respondBeforeStart) {
64*84e33947SAndroid Build Coastguard Worker   bool success = true;
65*84e33947SAndroid Build Coastguard Worker 
66*84e33947SAndroid Build Coastguard Worker   if (fragmentId == 0 || fragmentId == 1) {
67*84e33947SAndroid Build Coastguard Worker     size_t totalAppBinaryLen = (fragmentId == 0) ? bufferLen : appBinaryLen;
68*84e33947SAndroid Build Coastguard Worker     LOGD("Load nanoapp request for app ID 0x%016" PRIx64 " ver 0x%" PRIx32
69*84e33947SAndroid Build Coastguard Worker          " flags 0x%" PRIx32 " target API 0x%08" PRIx32
70*84e33947SAndroid Build Coastguard Worker          " size %zu (txnId %" PRIu32 " client %" PRIu16 ")",
71*84e33947SAndroid Build Coastguard Worker          appId, appVersion, appFlags, targetApiVersion, totalAppBinaryLen,
72*84e33947SAndroid Build Coastguard Worker          transactionId, hostClientId);
73*84e33947SAndroid Build Coastguard Worker 
74*84e33947SAndroid Build Coastguard Worker     if (getLoadManager().hasPendingLoadTransaction()) {
75*84e33947SAndroid Build Coastguard Worker       FragmentedLoadInfo info = getLoadManager().getTransactionInfo();
76*84e33947SAndroid Build Coastguard Worker       LOGW("A pending load transaction already exists (clientId=%" PRIu16
77*84e33947SAndroid Build Coastguard Worker            ", txnId=%" PRIu32 ", nextFragmentId=%" PRIu32 "). Overriding it",
78*84e33947SAndroid Build Coastguard Worker            info.hostClientId, info.transactionId, info.nextFragmentId);
79*84e33947SAndroid Build Coastguard Worker       // Send a failure response to host where nextFragmentId is either current
80*84e33947SAndroid Build Coastguard Worker       // or future to the host.
81*84e33947SAndroid Build Coastguard Worker       sendFragmentResponse(info.hostClientId, info.transactionId,
82*84e33947SAndroid Build Coastguard Worker                            info.nextFragmentId, /* success= */ false);
83*84e33947SAndroid Build Coastguard Worker       getLoadManager().markFailure();
84*84e33947SAndroid Build Coastguard Worker     }
85*84e33947SAndroid Build Coastguard Worker 
86*84e33947SAndroid Build Coastguard Worker     success = getLoadManager().prepareForLoad(
87*84e33947SAndroid Build Coastguard Worker         hostClientId, transactionId, appId, appVersion, appFlags,
88*84e33947SAndroid Build Coastguard Worker         totalAppBinaryLen, targetApiVersion);
89*84e33947SAndroid Build Coastguard Worker   }
90*84e33947SAndroid Build Coastguard Worker 
91*84e33947SAndroid Build Coastguard Worker   if (success) {
92*84e33947SAndroid Build Coastguard Worker     success = getLoadManager().copyNanoappFragment(
93*84e33947SAndroid Build Coastguard Worker         hostClientId, transactionId, (fragmentId == 0) ? 1 : fragmentId, buffer,
94*84e33947SAndroid Build Coastguard Worker         bufferLen);
95*84e33947SAndroid Build Coastguard Worker   } else {
96*84e33947SAndroid Build Coastguard Worker     LOGE("Failed to prepare for load");
97*84e33947SAndroid Build Coastguard Worker   }
98*84e33947SAndroid Build Coastguard Worker 
99*84e33947SAndroid Build Coastguard Worker   if (getLoadManager().isLoadComplete()) {
100*84e33947SAndroid Build Coastguard Worker     LOGD("Load manager load complete...");
101*84e33947SAndroid Build Coastguard Worker     auto cbData = MakeUnique<LoadNanoappCallbackData>();
102*84e33947SAndroid Build Coastguard Worker     if (cbData.isNull()) {
103*84e33947SAndroid Build Coastguard Worker       LOG_OOM();
104*84e33947SAndroid Build Coastguard Worker     } else {
105*84e33947SAndroid Build Coastguard Worker       cbData->transactionId = transactionId;
106*84e33947SAndroid Build Coastguard Worker       cbData->hostClientId = hostClientId;
107*84e33947SAndroid Build Coastguard Worker       cbData->appId = appId;
108*84e33947SAndroid Build Coastguard Worker       cbData->fragmentId = fragmentId;
109*84e33947SAndroid Build Coastguard Worker       cbData->nanoapp = getLoadManager().releaseNanoapp();
110*84e33947SAndroid Build Coastguard Worker       cbData->sendFragmentResponse = !respondBeforeStart;
111*84e33947SAndroid Build Coastguard Worker 
112*84e33947SAndroid Build Coastguard Worker       LOGD("Instance ID %" PRIu16 " assigned to app ID 0x%" PRIx64,
113*84e33947SAndroid Build Coastguard Worker            cbData->nanoapp->getInstanceId(), appId);
114*84e33947SAndroid Build Coastguard Worker 
115*84e33947SAndroid Build Coastguard Worker       // Note that if this fails, we'll generate the error response in
116*84e33947SAndroid Build Coastguard Worker       // the normal deferred callback
117*84e33947SAndroid Build Coastguard Worker       EventLoopManagerSingleton::get()->deferCallback(
118*84e33947SAndroid Build Coastguard Worker           SystemCallbackType::FinishLoadingNanoapp, std::move(cbData),
119*84e33947SAndroid Build Coastguard Worker           finishLoadingNanoappCallback);
120*84e33947SAndroid Build Coastguard Worker       if (respondBeforeStart) {
121*84e33947SAndroid Build Coastguard Worker         sendFragmentResponse(hostClientId, transactionId, fragmentId, success);
122*84e33947SAndroid Build Coastguard Worker       }  // else the response will be sent in finishLoadingNanoappCallback
123*84e33947SAndroid Build Coastguard Worker     }
124*84e33947SAndroid Build Coastguard Worker   } else {
125*84e33947SAndroid Build Coastguard Worker     // send a response for this fragment
126*84e33947SAndroid Build Coastguard Worker     sendFragmentResponse(hostClientId, transactionId, fragmentId, success);
127*84e33947SAndroid Build Coastguard Worker   }
128*84e33947SAndroid Build Coastguard Worker }
129*84e33947SAndroid Build Coastguard Worker 
130*84e33947SAndroid Build Coastguard Worker }  // namespace chre
131