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 #include "chre/platform/platform_nanoapp.h"
18 
19 #include <dlfcn.h>
20 #include <cinttypes>
21 
22 #include "chre/platform/assert.h"
23 #include "chre/platform/log.h"
24 #include "chre/platform/shared/nanoapp_dso_util.h"
25 #include "chre/util/system/napp_permissions.h"
26 #include "chre_api/chre/version.h"
27 
28 namespace chre {
29 
~PlatformNanoapp()30 PlatformNanoapp::~PlatformNanoapp() {
31   closeNanoapp();
32 }
33 
start()34 bool PlatformNanoapp::start() {
35   return openNanoapp() && mAppInfo->entryPoints.start();
36 }
37 
handleEvent(uint32_t senderInstanceId,uint16_t eventType,const void * eventData)38 void PlatformNanoapp::handleEvent(uint32_t senderInstanceId, uint16_t eventType,
39                                   const void *eventData) {
40   mAppInfo->entryPoints.handleEvent(senderInstanceId, eventType, eventData);
41 }
42 
end()43 void PlatformNanoapp::end() {
44   mAppInfo->entryPoints.end();
45   closeNanoapp();
46 }
47 
getAppId() const48 uint64_t PlatformNanoapp::getAppId() const {
49   return (mAppInfo == nullptr) ? 0 : mAppInfo->appId;
50 }
51 
getAppVersion() const52 uint32_t PlatformNanoapp::getAppVersion() const {
53   return mAppInfo->appVersion;
54 }
55 
getTargetApiVersion() const56 uint32_t PlatformNanoapp::getTargetApiVersion() const {
57   return CHRE_API_VERSION;
58 }
59 
getAppName() const60 const char *PlatformNanoapp::getAppName() const {
61   return (mAppInfo != nullptr) ? mAppInfo->name : "Unknown";
62 }
63 
supportsAppPermissions() const64 bool PlatformNanoapp::supportsAppPermissions() const {
65   return (mAppInfo != nullptr) ? (mAppInfo->structMinorVersion >=
66                                   CHRE_NSL_NANOAPP_INFO_STRUCT_MINOR_VERSION)
67                                : false;
68 }
69 
getAppPermissions() const70 uint32_t PlatformNanoapp::getAppPermissions() const {
71   return (supportsAppPermissions())
72              ? mAppInfo->appPermissions
73              : static_cast<uint32_t>(chre::NanoappPermissions::CHRE_PERMS_NONE);
74 }
75 
isSystemNanoapp() const76 bool PlatformNanoapp::isSystemNanoapp() const {
77   return (mAppInfo != nullptr && mAppInfo->isSystemNanoapp);
78 }
79 
logStateToBuffer(DebugDumpWrapper &) const80 void PlatformNanoapp::logStateToBuffer(
81     DebugDumpWrapper & /* debugDump */) const {}
82 
loadFromFile(const std::string & filename)83 void PlatformNanoappBase::loadFromFile(const std::string &filename) {
84   CHRE_ASSERT(!isLoaded());
85   mFilename = filename;
86 }
87 
loadStatic(const struct chreNslNanoappInfo * appInfo)88 void PlatformNanoappBase::loadStatic(const struct chreNslNanoappInfo *appInfo) {
89   CHRE_ASSERT(!isLoaded());
90   mIsStatic = true;
91   mAppInfo = appInfo;
92 }
93 
isLoaded() const94 bool PlatformNanoappBase::isLoaded() const {
95   return (mIsStatic || mDsoHandle != nullptr);
96 }
97 
openNanoapp()98 bool PlatformNanoappBase::openNanoapp() {
99   bool success = false;
100 
101   if (mIsStatic) {
102     success = true;
103   } else if (!mFilename.empty()) {
104     success = openNanoappFromFile();
105   } else {
106     CHRE_ASSERT(false);
107   }
108 
109   return success;
110 }
111 
openNanoappFromFile()112 bool PlatformNanoappBase::openNanoappFromFile() {
113   CHRE_ASSERT(!mFilename.empty());
114   CHRE_ASSERT_LOG(mDsoHandle == nullptr, "Re-opening nanoapp");
115   bool success = false;
116 
117   mDsoHandle = dlopen(mFilename.c_str(), RTLD_NOW | RTLD_GLOBAL);
118   if (mDsoHandle == nullptr) {
119     LOGE("Failed to load nanoapp from file %s: %s", mFilename.c_str(),
120          dlerror());
121   } else {
122     mAppInfo = static_cast<const struct chreNslNanoappInfo *>(
123         dlsym(mDsoHandle, CHRE_NSL_DSO_NANOAPP_INFO_SYMBOL_NAME));
124     if (mAppInfo == nullptr) {
125       LOGE("Failed to find app info symbol in %s: %s", mFilename.c_str(),
126            dlerror());
127     } else {
128       // TODO(b/120778991): reenable this check after adding support for passing
129       // in the .napp_header to the simulator
130       // success = validateAppInfo(0 /* skip ID validation */, 0, mAppInfo);
131       success = true;
132       if (!success) {
133         mAppInfo = nullptr;
134       } else {
135         LOGI("Nanoapp loaded: %s (0x%016" PRIx64 ") version 0x%" PRIx32
136              " uimg %d system %d from file %s",
137              mAppInfo->name, mAppInfo->appId, mAppInfo->appVersion,
138              mAppInfo->isTcmNanoapp, mAppInfo->isSystemNanoapp,
139              mFilename.c_str());
140         if (mAppInfo->structMinorVersion >=
141             CHRE_NSL_NANOAPP_INFO_STRUCT_MINOR_VERSION) {
142           LOGI("Nanoapp permissions: 0x%" PRIx32, mAppInfo->appPermissions);
143         }
144       }
145     }
146   }
147 
148   return success;
149 }
150 
closeNanoapp()151 void PlatformNanoappBase::closeNanoapp() {
152   if (mDsoHandle != nullptr) {
153     mAppInfo = nullptr;
154     if (dlclose(mDsoHandle) != 0) {
155       LOGE("dlclose failed: %s", dlerror());
156     }
157     mDsoHandle = nullptr;
158   }
159 }
160 
161 }  // namespace chre
162