1 /*
2  * Copyright (C) 2021 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_TAG "InvalidDevice"
18 
19 #include "InvalidDevice.h"
20 
21 #include <aidl/android/hardware/neuralnetworks/BnBuffer.h>
22 #include <aidl/android/hardware/neuralnetworks/BnDevice.h>
23 #include <aidl/android/hardware/neuralnetworks/BnPreparedModel.h>
24 #include <android/binder_auto_utils.h>
25 
26 #include "Conversions.h"
27 #include "Utils.h"
28 
29 #include <memory>
30 #include <string>
31 #include <utility>
32 #include <vector>
33 
34 namespace aidl::android::hardware::neuralnetworks {
35 namespace {
36 
toAStatus(ErrorStatus errorStatus,const std::string & errorMessage)37 ndk::ScopedAStatus toAStatus(ErrorStatus errorStatus, const std::string& errorMessage) {
38     if (errorStatus == ErrorStatus::NONE) {
39         return ndk::ScopedAStatus::ok();
40     }
41     return ndk::ScopedAStatus::fromServiceSpecificErrorWithMessage(
42             static_cast<int32_t>(errorStatus), errorMessage.c_str());
43 }
44 
45 }  // namespace
46 
create()47 std::shared_ptr<InvalidDevice> InvalidDevice::create() {
48     constexpr auto perf = PerformanceInfo{
49             .execTime = std::numeric_limits<float>::max(),
50             .powerUsage = std::numeric_limits<float>::max(),
51     };
52     auto capabilities = Capabilities{
53             .relaxedFloat32toFloat16PerformanceScalar = perf,
54             .relaxedFloat32toFloat16PerformanceTensor = perf,
55             .operandPerformance = {},
56             .ifPerformance = perf,
57             .whilePerformance = perf,
58     };
59     constexpr auto numberOfCacheFiles = NumberOfCacheFiles{
60             .numModelCache = 0,
61             .numDataCache = 0,
62     };
63     std::vector<Extension> extensions{};
64     constexpr auto deviceType = DeviceType::OTHER;
65     std::string versionString = "invalid";
66 
67     return ndk::SharedRefBase::make<InvalidDevice>(std::move(capabilities), numberOfCacheFiles,
68                                                    std::move(extensions), deviceType,
69                                                    std::move(versionString));
70 }
71 
InvalidDevice(Capabilities capabilities,const NumberOfCacheFiles & numberOfCacheFiles,std::vector<Extension> extensions,DeviceType deviceType,std::string versionString)72 InvalidDevice::InvalidDevice(Capabilities capabilities,
73                              const NumberOfCacheFiles& numberOfCacheFiles,
74                              std::vector<Extension> extensions, DeviceType deviceType,
75                              std::string versionString)
76     : kCapabilities(std::move(capabilities)),
77       kNumberOfCacheFiles(numberOfCacheFiles),
78       kExtensions(std::move(extensions)),
79       kDeviceType(deviceType),
80       kVersionString(std::move(versionString)) {}
81 
allocate(const BufferDesc &,const std::vector<IPreparedModelParcel> &,const std::vector<BufferRole> &,const std::vector<BufferRole> &,DeviceBuffer *)82 ndk::ScopedAStatus InvalidDevice::allocate(
83         const BufferDesc& /*desc*/, const std::vector<IPreparedModelParcel>& /*preparedModels*/,
84         const std::vector<BufferRole>& /*inputRoles*/,
85         const std::vector<BufferRole>& /*outputRoles*/, DeviceBuffer* /*deviceBuffer*/) {
86     return toAStatus(ErrorStatus::GENERAL_FAILURE, "InvalidDevice");
87 }
88 
getCapabilities(Capabilities * capabilities)89 ndk::ScopedAStatus InvalidDevice::getCapabilities(Capabilities* capabilities) {
90     *capabilities = kCapabilities;
91     return ndk::ScopedAStatus::ok();
92 }
93 
getNumberOfCacheFilesNeeded(NumberOfCacheFiles * numberOfCacheFiles)94 ndk::ScopedAStatus InvalidDevice::getNumberOfCacheFilesNeeded(
95         NumberOfCacheFiles* numberOfCacheFiles) {
96     *numberOfCacheFiles = kNumberOfCacheFiles;
97     return ndk::ScopedAStatus::ok();
98 }
99 
getSupportedExtensions(std::vector<Extension> * extensions)100 ndk::ScopedAStatus InvalidDevice::getSupportedExtensions(std::vector<Extension>* extensions) {
101     *extensions = kExtensions;
102     return ndk::ScopedAStatus::ok();
103 }
104 
getSupportedOperations(const Model & model,std::vector<bool> * supportedOperations)105 ndk::ScopedAStatus InvalidDevice::getSupportedOperations(const Model& model,
106                                                          std::vector<bool>* supportedOperations) {
107     if (const auto result = utils::validate(model); !result.ok()) {
108         return toAStatus(ErrorStatus::INVALID_ARGUMENT, result.error());
109     }
110     *supportedOperations = std::vector<bool>(model.main.operations.size(), false);
111     return ndk::ScopedAStatus::ok();
112 }
113 
getType(DeviceType * deviceType)114 ndk::ScopedAStatus InvalidDevice::getType(DeviceType* deviceType) {
115     *deviceType = kDeviceType;
116     return ndk::ScopedAStatus::ok();
117 }
118 
getVersionString(std::string * versionString)119 ndk::ScopedAStatus InvalidDevice::getVersionString(std::string* versionString) {
120     *versionString = kVersionString;
121     return ndk::ScopedAStatus::ok();
122 }
123 
prepareModel(const Model & model,ExecutionPreference preference,Priority priority,int64_t deadline,const std::vector<ndk::ScopedFileDescriptor> & modelCache,const std::vector<ndk::ScopedFileDescriptor> & dataCache,const std::vector<uint8_t> & token,const std::shared_ptr<IPreparedModelCallback> & callback)124 ndk::ScopedAStatus InvalidDevice::prepareModel(
125         const Model& model, ExecutionPreference preference, Priority priority, int64_t deadline,
126         const std::vector<ndk::ScopedFileDescriptor>& modelCache,
127         const std::vector<ndk::ScopedFileDescriptor>& dataCache, const std::vector<uint8_t>& token,
128         const std::shared_ptr<IPreparedModelCallback>& callback) {
129     if (callback.get() == nullptr) {
130         return toAStatus(ErrorStatus::INVALID_ARGUMENT,
131                          "invalid callback passed to InvalidDevice::prepareModel");
132     }
133     if (const auto result = utils::validate(model); !result.ok()) {
134         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
135         return toAStatus(ErrorStatus::INVALID_ARGUMENT, result.error());
136     }
137     if (const auto result = utils::validate(preference); !result.ok()) {
138         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
139         return toAStatus(ErrorStatus::INVALID_ARGUMENT, result.error());
140     }
141     if (const auto result = utils::validate(priority); !result.ok()) {
142         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
143         return toAStatus(ErrorStatus::INVALID_ARGUMENT, result.error());
144     }
145     if (deadline < -1) {
146         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
147         return toAStatus(ErrorStatus::INVALID_ARGUMENT,
148                          "Invalid deadline " + std::to_string(deadline));
149     }
150     if (modelCache.size() != static_cast<size_t>(kNumberOfCacheFiles.numModelCache)) {
151         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
152         return toAStatus(ErrorStatus::INVALID_ARGUMENT,
153                          "Invalid modelCache, size = " + std::to_string(modelCache.size()));
154     }
155     if (dataCache.size() != static_cast<size_t>(kNumberOfCacheFiles.numDataCache)) {
156         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
157         return toAStatus(ErrorStatus::INVALID_ARGUMENT,
158                          "Invalid modelCache, size = " + std::to_string(dataCache.size()));
159     }
160     if (token.size() != IDevice::BYTE_SIZE_OF_CACHE_TOKEN) {
161         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
162         return toAStatus(
163                 ErrorStatus::INVALID_ARGUMENT,
164                 "Invalid cache token, size = " + std::to_string(IDevice::BYTE_SIZE_OF_CACHE_TOKEN));
165     }
166     callback->notify(ErrorStatus::GENERAL_FAILURE, nullptr);
167     return ndk::ScopedAStatus::ok();
168 }
169 
prepareModelWithConfig(const Model & model,const PrepareModelConfig & config,const std::shared_ptr<IPreparedModelCallback> & callback)170 ndk::ScopedAStatus InvalidDevice::prepareModelWithConfig(
171         const Model& model, const PrepareModelConfig& config,
172         const std::shared_ptr<IPreparedModelCallback>& callback) {
173     if (!utils::valid(config.extensionNameToPrefix)) {
174         callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
175         return toAStatus(ErrorStatus::INVALID_ARGUMENT, "Invalid extensionNameToPrefix");
176     }
177     for (const auto& hint : config.compilationHints) {
178         auto result = std::find_if(config.extensionNameToPrefix.begin(),
179                                    config.extensionNameToPrefix.end(),
180                                    [&hint](const ExtensionNameAndPrefix& extension) {
181                                        uint16_t prefix = static_cast<uint32_t>(hint.token) >>
182                                                          IDevice::EXTENSION_TYPE_LOW_BITS_TYPE;
183                                        return prefix == extension.prefix;
184                                    });
185         if (result == config.extensionNameToPrefix.end()) {
186             callback->notify(ErrorStatus::INVALID_ARGUMENT, nullptr);
187             return toAStatus(ErrorStatus::INVALID_ARGUMENT,
188                              "Invalid token for compilation hints: " + std::to_string(hint.token));
189         }
190     }
191     return prepareModel(model, config.preference, config.priority, config.deadlineNs,
192                         config.modelCache, config.dataCache, utils::toVec(config.cacheToken),
193                         callback);
194 }
195 
prepareModelFromCache(int64_t,const std::vector<ndk::ScopedFileDescriptor> &,const std::vector<ndk::ScopedFileDescriptor> &,const std::vector<uint8_t> &,const std::shared_ptr<IPreparedModelCallback> & callback)196 ndk::ScopedAStatus InvalidDevice::prepareModelFromCache(
197         int64_t /*deadline*/, const std::vector<ndk::ScopedFileDescriptor>& /*modelCache*/,
198         const std::vector<ndk::ScopedFileDescriptor>& /*dataCache*/,
199         const std::vector<uint8_t>& /*token*/,
200         const std::shared_ptr<IPreparedModelCallback>& callback) {
201     callback->notify(ErrorStatus::GENERAL_FAILURE, nullptr);
202     return toAStatus(ErrorStatus::GENERAL_FAILURE, "InvalidDevice");
203 }
204 
205 }  // namespace aidl::android::hardware::neuralnetworks
206