/* * Copyright (c) 2022 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "nnrt_utils.h" #include "const.h" #include namespace OHOS { namespace NeuralNetworkRuntime { namespace Test { OH_NN_UInt32Array TransformUInt32Array(const std::vector& vector) { uint32_t* data = (vector.empty()) ? nullptr : const_cast(vector.data()); return {data, vector.size()}; } int BuildMultiOpGraph(OH_NNModel *model, const OHNNGraphArgsMulti &graphArgs) { int ret = 0; int opCnt = 0; for (int j = 0; j < graphArgs.operationTypes.size(); j++) { for (int i = 0; i < graphArgs.operands[j].size(); i++) { const OHNNOperandTest &operandTem = graphArgs.operands[j][i]; auto quantParam = operandTem.quantParam; OH_NN_Tensor operand = {operandTem.dataType, (uint32_t) operandTem.shape.size(), operandTem.shape.data(), quantParam, operandTem.type}; ret = OH_NNModel_AddTensor(model, &operand); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_AddTensor failed! ret=%d\n", ret); return ret; } if (std::find(graphArgs.paramIndices[j].begin(), graphArgs.paramIndices[j].end(), opCnt) != graphArgs.paramIndices[j].end()) { ret = OH_NNModel_SetTensorData(model, opCnt, operandTem.data, operandTem.length); } if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_SetTensorData failed! ret=%d\n", ret); return ret; } opCnt += 1; } auto paramIndices = TransformUInt32Array(graphArgs.paramIndices[j]); auto inputIndices = TransformUInt32Array(graphArgs.inputIndices[j]); auto outputIndices = TransformUInt32Array(graphArgs.outputIndices[j]); ret = OH_NNModel_AddOperation(model, graphArgs.operationTypes[j], ¶mIndices, &inputIndices, &outputIndices); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_AddOperation failed! ret=%d\n", ret); return ret; } } auto graphInputs = TransformUInt32Array(graphArgs.graphInput); auto graphOutputs = TransformUInt32Array(graphArgs.graphOutput); ret = OH_NNModel_SpecifyInputsAndOutputs(model, &graphInputs, &graphOutputs); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_SpecifyInputsAndOutputs failed! ret=%d\n", ret); return ret; } ret = OH_NNModel_Finish(model); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_Finish failed! ret=%d\n", ret); return ret; } return ret; } int BuildSingleOpGraph(OH_NNModel *model, const OHNNGraphArgs &graphArgs) { int ret = 0; for (int i = 0; i < graphArgs.operands.size(); i++) { const OHNNOperandTest &operandTem = graphArgs.operands[i]; auto quantParam = operandTem.quantParam; OH_NN_Tensor operand = {operandTem.dataType, (uint32_t) operandTem.shape.size(), operandTem.shape.data(), quantParam, operandTem.type}; ret = OH_NNModel_AddTensor(model, &operand); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_AddTensor failed! ret=%d\n", ret); return ret; } if (std::find(graphArgs.paramIndices.begin(), graphArgs.paramIndices.end(), i) != graphArgs.paramIndices.end()) { ret = OH_NNModel_SetTensorData(model, i, operandTem.data, operandTem.length); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_SetTensorData failed! ret=%d\n", ret); return ret; } } } auto paramIndices = TransformUInt32Array(graphArgs.paramIndices); auto inputIndices = TransformUInt32Array(graphArgs.inputIndices); auto outputIndices = TransformUInt32Array(graphArgs.outputIndices); if (graphArgs.addOperation) { ret = OH_NNModel_AddOperation(model, graphArgs.operationType, ¶mIndices, &inputIndices, &outputIndices); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_AddOperation failed! ret=%d\n", ret); return ret; } } if (graphArgs.specifyIO) { ret = OH_NNModel_SpecifyInputsAndOutputs(model, &inputIndices, &outputIndices); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_SpecifyInputsAndOutputs failed! ret=%d\n", ret); return ret; } } if (graphArgs.build) { ret = OH_NNModel_Finish(model); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNModel_Finish failed! ret=%d\n", ret); return ret; } } return ret; } int SetDevice(OH_NNCompilation *compilation) { int ret = 0; const size_t *devicesID{nullptr}; uint32_t devicesCount{0}; ret = OH_NNDevice_GetAllDevicesID(&devicesID, &devicesCount); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNDevice_GetAllDevicesID failed! ret=%d\n", ret); return ret; } if (devicesCount <= NO_DEVICE_COUNT) { return OH_NN_FAILED; } size_t targetDevice = devicesID[0]; // Use the first device in system test. ret = OH_NNCompilation_SetDevice(compilation, targetDevice); return ret; } int CompileGraphMock(OH_NNCompilation *compilation, const OHNNCompileParam &compileParam) { int ret = 0; ret = SetDevice(compilation); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNCompilation_SetDevice failed! ret=%d\n", ret); return ret; } // set cache if (!compileParam.cacheDir.empty()) { ret = OH_NNCompilation_SetCache(compilation, compileParam.cacheDir.c_str(), compileParam.cacheVersion); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNCompilation_SetCache failed! ret=%d\n", ret); return ret; } } // set performance if (compileParam.performanceMode != OH_NN_PERFORMANCE_NONE) { ret = OH_NNCompilation_SetPerformanceMode(compilation, compileParam.performanceMode); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNCompilation_SetPerformanceMode failed! ret=%d\n", ret); return ret; } } // set priority if (compileParam.priority != OH_NN_PRIORITY_NONE) { ret = OH_NNCompilation_SetPriority(compilation, compileParam.priority); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNCompilation_SetPriority failed! ret=%d\n", ret); return ret; } } // enable fp16 if (compileParam.enableFp16) { ret = OH_NNCompilation_EnableFloat16(compilation, compileParam.enableFp16); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNCompilation_EnableFloat16 failed! ret=%d\n", ret); return ret; } } // build ret = OH_NNCompilation_Build(compilation); return ret; } int ExecuteGraphMock(OH_NNExecutor *executor, const OHNNGraphArgs &graphArgs, float* expect) { OHOS::sptr device = V1_0::MockIDevice::GetInstance(); int ret = 0; uint32_t inputIndex = 0; uint32_t outputIndex = 0; for (auto i = 0; i < graphArgs.operands.size(); i++) { const OHNNOperandTest &operandTem = graphArgs.operands[i]; auto quantParam = operandTem.quantParam; OH_NN_Tensor operand = {operandTem.dataType, (uint32_t) operandTem.shape.size(), operandTem.shape.data(), quantParam, operandTem.type}; if (std::find(graphArgs.inputIndices.begin(), graphArgs.inputIndices.end(), i) != graphArgs.inputIndices.end()) { ret = OH_NNExecutor_SetInput(executor, inputIndex, &operand, operandTem.data, operandTem.length); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNExecutor_SetInput failed! ret=%d\n", ret); return ret; } inputIndex += 1; } else if (std::find(graphArgs.outputIndices.begin(), graphArgs.outputIndices.end(), i) != graphArgs.outputIndices.end()) { ret = OH_NNExecutor_SetOutput(executor, outputIndex, operandTem.data, operandTem.length); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNExecutor_SetOutput failed! ret=%d\n", ret); return ret; } if(expect!=nullptr){ ret = device->MemoryCopy(expect, operandTem.length); } if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] device set expect output failed! ret=%d\n", ret); return ret; } outputIndex += 1; } } ret = OH_NNExecutor_Run(executor); return ret; } int ExecutorWithMemory(OH_NNExecutor *executor, const OHNNGraphArgs &graphArgs, OH_NN_Memory *OHNNMemory[], float* expect) { OHOS::sptr device = V1_0::MockIDevice::GetInstance(); int ret = 0; uint32_t inputIndex = 0; uint32_t outputIndex = 0; for (auto i = 0; i < graphArgs.operands.size(); i++) { const OHNNOperandTest &operandTem = graphArgs.operands[i]; auto quantParam = operandTem.quantParam; OH_NN_Tensor operand = {operandTem.dataType, (uint32_t) operandTem.shape.size(), operandTem.shape.data(), quantParam, operandTem.type}; if (std::find(graphArgs.inputIndices.begin(), graphArgs.inputIndices.end(), i) != graphArgs.inputIndices.end()) { OH_NN_Memory *inputMemory = OH_NNExecutor_AllocateInputMemory(executor, inputIndex, operandTem.length); ret = OH_NNExecutor_SetInputWithMemory(executor, inputIndex, &operand, inputMemory); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNExecutor_SetInputWithMemory failed! ret=%d\n", ret); return ret; } memcpy_s(inputMemory->data, operandTem.length, (void *) operandTem.data, operandTem.length); OHNNMemory[inputIndex] = inputMemory; inputIndex += 1; } else if (std::find(graphArgs.outputIndices.begin(), graphArgs.outputIndices.end(), i) != graphArgs.outputIndices.end()) { OH_NN_Memory *outputMemory = OH_NNExecutor_AllocateOutputMemory(executor, outputIndex, operandTem.length); ret = OH_NNExecutor_SetOutputWithMemory(executor, outputIndex, outputMemory); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] OH_NNExecutor_SetOutputWithMemory failed! ret=%d\n", ret); return ret; } ret = device->MemoryCopy(expect, operandTem.length); if (ret != OH_NN_SUCCESS) { LOGE("[NNRtTest] device set expect output failed! ret=%d\n", ret); return ret; } OHNNMemory[inputIndex + outputIndex] = outputMemory; outputIndex += 1; } } ret = OH_NNExecutor_Run(executor); return ret; } void Free(OH_NNModel *model, OH_NNCompilation *compilation, OH_NNExecutor *executor) { if (model != nullptr) { OH_NNModel_Destroy(&model); ASSERT_EQ(nullptr, model); } if (compilation != nullptr) { OH_NNCompilation_Destroy(&compilation); ASSERT_EQ(nullptr, compilation); } if (executor != nullptr) { OH_NNExecutor_Destroy(&executor); ASSERT_EQ(nullptr, executor); } } PathType CheckPath(const std::string &path) { if (path.empty()) { LOGI("CheckPath: path is null"); return PathType::NOT_FOUND; } struct stat buf{}; if (stat(path.c_str(), &buf) == 0) { if (buf.st_mode & S_IFDIR) { return PathType::DIR; } else if (buf.st_mode & S_IFREG) { return PathType::FILE; } else { return PathType::UNKNOWN; } } LOGI("%s not found", path.c_str()); return PathType::NOT_FOUND; } bool DeleteFile(const std::string &path) { if (path.empty()) { LOGI("DeleteFile: path is null"); return false; } if (CheckPath(path) == PathType::NOT_FOUND) { LOGI("not found: %s", path.c_str()); return true; } if (remove(path.c_str()) == 0) { LOGI("deleted: %s", path.c_str()); return true; } LOGI("delete failed: %s", path.c_str()); return false; } void CopyFile(const std::string &srcPath, const std::string &dstPath) { std::ifstream src(srcPath, std::ios::binary); std::ofstream dst(dstPath, std::ios::binary); dst << src.rdbuf(); } std::string ConcatPath(const std::string &str1, const std::string &str2) { // boundary if (str2.empty()) { return str1; } if (str1.empty()) { return str2; } // concat char end = str1[str1.size() - 1]; if (end == '\\' or end == '/') { return str1 + str2; } else { return str1 + '/' + str2; } } void DeleteFolder(const std::string &path) { if (path.empty()) { LOGI("DeletePath: path is null"); return; } DIR *dir = opendir(path.c_str()); // check is dir ? if (dir == nullptr) { LOGE("[NNRtTest] Can not open dir. Check path or permission! path: %s", path.c_str()); return; } struct dirent *file; // read all the files in dir std::vector pathList; while ((file = readdir(dir)) != nullptr) { // skip "." and ".." if (strcmp(file->d_name, ".") == 0 || strcmp(file->d_name, "..") == 0) { continue; } if (file->d_type == DT_DIR) { std::string filePath = path + "/" + file->d_name; DeleteFolder(filePath); // 递归执行 } else { pathList.emplace_back(ConcatPath(path, file->d_name)); } } closedir(dir); pathList.emplace_back(path); LOGI("[Common] Delete folder %s", path.c_str()); for (auto &i : pathList) { DeleteFile(i); } } bool CreateFolder(const std::string &path) { if (path.empty()) { LOGI("CreateFolder: path is empty"); return false; } LOGI("CreateFolder:%s", path.c_str()); mode_t mode = 0700; for (int i = 1; i < path.size() - 1; i++) { if (path[i] != '/') { continue; } PathType ret = CheckPath(path.substr(0, i)); switch (ret) { case PathType::DIR: continue; case PathType::NOT_FOUND: LOGI("mkdir: %s", path.substr(0, i).c_str()); mkdir(path.substr(0, i).c_str(), mode); break; default: LOGI("error: %s", path.substr(0, i).c_str()); return false; } } mkdir(path.c_str(), mode); return CheckPath(path) == PathType::DIR; } bool CheckOutput(const float* output, const float* expect) { if (output == nullptr || expect == nullptr) { LOGE("[NNRtTest] output or expect is nullptr\n"); return false; } for (int i = 0; i < ELEMENT_COUNT; i++) { if (std::abs(float(output[i]) - float(expect[i])) > 1e-8) { for (int j = 0; j < ELEMENT_COUNT; j++) { LOGE("[NNRtTest] output %d not match: expect:%f, actual:%f\n", j, float(expect[j]), float(output[j])); } return false; } } return true; } } // namespace Test } // namespace NeuralNetworkRuntime } // namespace OHOS