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