From 6666e0c6c85be6a9aa0d60929ed3cfc2966072b2 Mon Sep 17 00:00:00 2001 From: Ravi Kumar Neti Date: Mon, 22 Jan 2024 06:08:17 +0530 Subject: [PATCH] Adding Object Detection solution Signed-off-by: Ravi Kumar Neti --- .../03-ObjectDetection/GenerateDLC.ipynb | 376 ++++++ .../android/03-ObjectDetection/README.md | 215 ++++ .../03-ObjectDetection/app/build.gradle | 68 + .../03-ObjectDetection/app/local.properties | 8 + .../03-ObjectDetection/app/proguard-rules.pro | 21 + .../app/src/main/AndroidManifest.xml | 38 + .../app/src/main/assets/ReadMe.txt | 1 + .../app/src/main/cpp/CMakeLists.txt | 66 + .../app/src/main/cpp/Model.cpp | 14 + .../app/src/main/cpp/Model.h | 60 + .../app/src/main/cpp/SSDMobileNetV2_Model.cpp | 111 ++ .../app/src/main/cpp/SSDMobileNetV2_Model.h | 133 ++ .../app/src/main/cpp/YOLONAS_Model.cpp | 90 ++ .../app/src/main/cpp/YOLONAS_Model.h | 142 +++ .../app/src/main/cpp/YOLO_X_Model.cpp | 145 +++ .../app/src/main/cpp/YOLO_X_Model.h | 165 +++ .../app/src/main/cpp/inc/hpp/CheckRuntime.hpp | 17 + .../src/main/cpp/inc/hpp/CreateUserBuffer.hpp | 43 + .../src/main/cpp/inc/hpp/LoadContainer.hpp | 19 + .../src/main/cpp/inc/hpp/LoadInputTensor.hpp | 27 + .../main/cpp/inc/hpp/SetBuilderOptions.hpp | 26 + .../app/src/main/cpp/inc/hpp/Util.hpp | 41 + .../app/src/main/cpp/inc/hpp/inference.h | 74 ++ .../src/main/cpp/inc/zdl/DiagLog/IDiagLog.h | 102 ++ .../src/main/cpp/inc/zdl/DiagLog/IDiagLog.hpp | 133 ++ .../src/main/cpp/inc/zdl/DiagLog/Options.h | 164 +++ .../src/main/cpp/inc/zdl/DiagLog/Options.hpp | 50 + .../cpp/inc/zdl/DlContainer/DlContainer.h | 185 +++ .../cpp/inc/zdl/DlContainer/IDlContainer.hpp | 146 +++ .../src/main/cpp/inc/zdl/DlSystem/DlEnums.h | 267 ++++ .../src/main/cpp/inc/zdl/DlSystem/DlEnums.hpp | 266 ++++ .../src/main/cpp/inc/zdl/DlSystem/DlError.h | 299 +++++ .../src/main/cpp/inc/zdl/DlSystem/DlError.hpp | 261 ++++ .../main/cpp/inc/zdl/DlSystem/DlOptional.hpp | 244 ++++ .../src/main/cpp/inc/zdl/DlSystem/DlVersion.h | 122 ++ .../main/cpp/inc/zdl/DlSystem/DlVersion.hpp | 118 ++ .../cpp/inc/zdl/DlSystem/IBufferAttributes.h | 117 ++ .../inc/zdl/DlSystem/IBufferAttributes.hpp | 85 ++ .../inc/zdl/DlSystem/IOBufferDataTypeMap.h | 156 +++ .../inc/zdl/DlSystem/IOBufferDataTypeMap.hpp | 69 + .../src/main/cpp/inc/zdl/DlSystem/ITensor.h | 118 ++ .../src/main/cpp/inc/zdl/DlSystem/ITensor.hpp | 95 ++ .../cpp/inc/zdl/DlSystem/ITensorFactory.hpp | 52 + .../main/cpp/inc/zdl/DlSystem/ITensorItr.hpp | 199 +++ .../cpp/inc/zdl/DlSystem/ITensorItrImpl.hpp | 32 + .../main/cpp/inc/zdl/DlSystem/IUserBuffer.h | 714 +++++++++++ .../main/cpp/inc/zdl/DlSystem/IUserBuffer.hpp | 390 ++++++ .../inc/zdl/DlSystem/IUserBufferFactory.hpp | 68 + .../cpp/inc/zdl/DlSystem/PlatformConfig.h | 329 +++++ .../cpp/inc/zdl/DlSystem/PlatformConfig.hpp | 265 ++++ .../main/cpp/inc/zdl/DlSystem/RuntimeList.h | 203 +++ .../main/cpp/inc/zdl/DlSystem/RuntimeList.hpp | 115 ++ .../inc/zdl/DlSystem/SnpeApiExportDefine.h | 34 + .../src/main/cpp/inc/zdl/DlSystem/String.hpp | 70 + .../main/cpp/inc/zdl/DlSystem/StringList.h | 154 +++ .../main/cpp/inc/zdl/DlSystem/StringList.hpp | 73 ++ .../src/main/cpp/inc/zdl/DlSystem/TensorMap.h | 154 +++ .../main/cpp/inc/zdl/DlSystem/TensorMap.hpp | 81 ++ .../main/cpp/inc/zdl/DlSystem/TensorShape.h | 174 +++ .../main/cpp/inc/zdl/DlSystem/TensorShape.hpp | 104 ++ .../cpp/inc/zdl/DlSystem/TensorShapeMap.h | 163 +++ .../cpp/inc/zdl/DlSystem/TensorShapeMap.hpp | 77 ++ .../main/cpp/inc/zdl/DlSystem/UserBufferMap.h | 151 +++ .../cpp/inc/zdl/DlSystem/UserBufferMap.hpp | 80 ++ .../main/cpp/inc/zdl/DlSystem/UserMemoryMap.h | 156 +++ .../cpp/inc/zdl/DlSystem/UserMemoryMap.hpp | 76 ++ .../zdl/PlatformValidator/PlatformValidator.h | 107 ++ .../PlatformValidator/PlatformValidator.hpp | 57 + .../cpp/inc/zdl/SNPE/ApplicationBufferMap.h | 85 ++ .../cpp/inc/zdl/SNPE/ApplicationBufferMap.hpp | 90 ++ .../app/src/main/cpp/inc/zdl/SNPE/PSNPE.h | 898 +++++++++++++ .../app/src/main/cpp/inc/zdl/SNPE/PSNPE.hpp | 537 ++++++++ .../main/cpp/inc/zdl/SNPE/RuntimeConfigList.h | 118 ++ .../cpp/inc/zdl/SNPE/RuntimeConfigList.hpp | 153 +++ .../app/src/main/cpp/inc/zdl/SNPE/SNPE.h | 336 +++++ .../app/src/main/cpp/inc/zdl/SNPE/SNPE.hpp | 125 ++ .../src/main/cpp/inc/zdl/SNPE/SNPEBuilder.h | 334 +++++ .../src/main/cpp/inc/zdl/SNPE/SNPEBuilder.hpp | 136 ++ .../src/main/cpp/inc/zdl/SNPE/SNPEFactory.hpp | 88 ++ .../app/src/main/cpp/inc/zdl/SNPE/SNPEUtil.h | 354 ++++++ .../main/cpp/inc/zdl/SNPE/UserBufferList.h | 77 ++ .../main/cpp/inc/zdl/SNPE/UserBufferList.hpp | 76 ++ .../src/main/cpp/inc/zdl/SnpeUdo/UdoBase.h | 546 ++++++++ .../app/src/main/cpp/inc/zdl/SnpeUdo/UdoReg.h | 117 ++ .../src/main/cpp/inc/zdl/SnpeUdo/UdoShared.h | 57 + .../app/src/main/cpp/inc/zdl/Wrapper.hpp | 449 +++++++ .../app/src/main/cpp/inference.cpp | 194 +++ .../app/src/main/cpp/inference_helper.cpp | 288 +++++ .../src/main/cpp/objectdetectionYoloNas.cpp | 195 +++ .../app/src/main/ic_launcher-playstore.png | Bin 0 -> 136308 bytes .../aistack_objdetect/CameraFragment.java | 1131 +++++++++++++++++ .../aistack_objdetect/FragmentRender.java | 104 ++ .../aistack_objdetect/HomeScreenActivity.java | 112 ++ .../qcom/aistack_objdetect/MainActivity.java | 138 ++ .../qcom/aistack_objdetect/RectangleBox.java | 36 + .../qcom/aistack_objdetect/SNPEHelper.java | 104 ++ .../app/src/main/jniLibs/arm64-v8a/ReadMe.txt | 2 + .../drawable-v24/ic_launcher_foreground.xml | 34 + .../res/drawable/ic_launcher_background.xml | 74 ++ .../main/res/layout/activity_home_screen.xml | 78 ++ .../src/main/res/layout/fragment_camera.xml | 22 + .../app/src/main/res/layout/main_activity.xml | 76 ++ .../res/mipmap-anydpi-v26/ic_launcher.xml | 5 + .../mipmap-anydpi-v26/ic_launcher_round.xml | 5 + .../src/main/res/mipmap-hdpi/ic_launcher.png | Bin 0 -> 5768 bytes .../mipmap-hdpi/ic_launcher_background.png | Bin 0 -> 557 bytes .../mipmap-hdpi/ic_launcher_foreground.png | Bin 0 -> 18124 bytes .../res/mipmap-hdpi/ic_launcher_round.png | Bin 0 -> 7500 bytes .../src/main/res/mipmap-mdpi/ic_launcher.png | Bin 0 -> 3348 bytes .../mipmap-mdpi/ic_launcher_background.png | Bin 0 -> 388 bytes .../mipmap-mdpi/ic_launcher_foreground.png | Bin 0 -> 9872 bytes .../res/mipmap-mdpi/ic_launcher_round.png | Bin 0 -> 4201 bytes .../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin 0 -> 8856 bytes .../mipmap-xhdpi/ic_launcher_background.png | Bin 0 -> 751 bytes .../mipmap-xhdpi/ic_launcher_foreground.png | Bin 0 -> 27948 bytes .../res/mipmap-xhdpi/ic_launcher_round.png | Bin 0 -> 11393 bytes .../main/res/mipmap-xxhdpi/ic_launcher.png | Bin 0 -> 15532 bytes .../mipmap-xxhdpi/ic_launcher_background.png | Bin 0 -> 1157 bytes .../mipmap-xxhdpi/ic_launcher_foreground.png | Bin 0 -> 52202 bytes .../res/mipmap-xxhdpi/ic_launcher_round.png | Bin 0 -> 19925 bytes .../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin 0 -> 23403 bytes .../mipmap-xxxhdpi/ic_launcher_background.png | Bin 0 -> 1748 bytes .../mipmap-xxxhdpi/ic_launcher_foreground.png | Bin 0 -> 82311 bytes .../res/mipmap-xxxhdpi/ic_launcher_round.png | Bin 0 -> 30175 bytes .../app/src/main/res/values/colors.xml | 6 + .../app/src/main/res/values/dimen.xml | 3 + .../app/src/main/res/values/strings.xml | 6 + .../app/src/main/res/values/styles.xml | 11 + .../android/03-ObjectDetection/build.gradle | 28 + .../demo/ObjectDetectYoloNAS.gif | Bin 0 -> 7532385 bytes .../03-ObjectDetection/gradle.properties | 16 + .../android/03-ObjectDetection/gradlew | 234 ++++ .../android/03-ObjectDetection/gradlew.bat | 89 ++ .../03-ObjectDetection/resolveDependencies.sh | 35 + .../android/03-ObjectDetection/sdk/ReadMe.txt | 2 + .../03-ObjectDetection/settings.gradle | 3 + 136 files changed, 16582 insertions(+) create mode 100644 ai-solutions/android/03-ObjectDetection/GenerateDLC.ipynb create mode 100644 ai-solutions/android/03-ObjectDetection/README.md create mode 100644 ai-solutions/android/03-ObjectDetection/app/build.gradle create mode 100644 ai-solutions/android/03-ObjectDetection/app/local.properties create mode 100644 ai-solutions/android/03-ObjectDetection/app/proguard-rules.pro create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/AndroidManifest.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/assets/ReadMe.txt create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/CMakeLists.txt create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CheckRuntime.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CreateUserBuffer.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadContainer.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadInputTensor.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/SetBuilderOptions.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/Util.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/inference.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/DlContainer.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/IDlContainer.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlOptional.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorFactory.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItr.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItrImpl.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBufferFactory.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/SnpeApiExportDefine.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/String.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEFactory.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEUtil.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoBase.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoReg.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoShared.h create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/Wrapper.hpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference_helper.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/cpp/objectdetectionYoloNas.cpp create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/ic_launcher-playstore.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/CameraFragment.java create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/FragmentRender.java create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/HomeScreenActivity.java create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/MainActivity.java create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/RectangleBox.java create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/SNPEHelper.java create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/jniLibs/arm64-v8a/ReadMe.txt create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/drawable-v24/ic_launcher_foreground.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/drawable/ic_launcher_background.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/layout/activity_home_screen.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/layout/fragment_camera.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/layout/main_activity.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-hdpi/ic_launcher.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-hdpi/ic_launcher_background.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-hdpi/ic_launcher_foreground.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-hdpi/ic_launcher_round.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-mdpi/ic_launcher.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-mdpi/ic_launcher_background.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-mdpi/ic_launcher_foreground.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-mdpi/ic_launcher_round.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xhdpi/ic_launcher.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xhdpi/ic_launcher_background.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xhdpi/ic_launcher_foreground.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxhdpi/ic_launcher.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxhdpi/ic_launcher_background.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxhdpi/ic_launcher_foreground.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxxhdpi/ic_launcher_background.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxxhdpi/ic_launcher_foreground.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/values/colors.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/values/dimen.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/values/strings.xml create mode 100644 ai-solutions/android/03-ObjectDetection/app/src/main/res/values/styles.xml create mode 100644 ai-solutions/android/03-ObjectDetection/build.gradle create mode 100644 ai-solutions/android/03-ObjectDetection/demo/ObjectDetectYoloNAS.gif create mode 100644 ai-solutions/android/03-ObjectDetection/gradle.properties create mode 100644 ai-solutions/android/03-ObjectDetection/gradlew create mode 100644 ai-solutions/android/03-ObjectDetection/gradlew.bat create mode 100644 ai-solutions/android/03-ObjectDetection/resolveDependencies.sh create mode 100644 ai-solutions/android/03-ObjectDetection/sdk/ReadMe.txt create mode 100644 ai-solutions/android/03-ObjectDetection/settings.gradle diff --git a/ai-solutions/android/03-ObjectDetection/GenerateDLC.ipynb b/ai-solutions/android/03-ObjectDetection/GenerateDLC.ipynb new file mode 100644 index 00000000..cfc14630 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/GenerateDLC.ipynb @@ -0,0 +1,376 @@ +{ + "cells": [ + { + "attachments": {}, + "cell_type": "markdown", + "id": "721491e1", + "metadata": {}, + "source": [ + "## Steps for generating YoloNAS dlc\n", + "#### Note->Use python3.8 or above for generating onnx and python3.6 for generating dlc" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "d1d3b4eb", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Requirement already satisfied: super-gradients==3.1.2 in /usr/local/lib/python3.8/site-packages (3.1.2)\n", + "Requirement already satisfied: torch>=1.9.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.13.1)\n", + "Requirement already satisfied: tqdm>=4.57.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (4.65.0)\n", + "Requirement already satisfied: boto3>=1.17.15 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.28.2)\n", + "Requirement already satisfied: jsonschema>=3.2.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (4.17.3)\n", + "Requirement already satisfied: Deprecated>=1.2.11 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.2.14)\n", + "Requirement already satisfied: opencv-python>=4.5.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (4.5.2.52)\n", + "Requirement already satisfied: scipy>=1.6.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.10.1)\n", + "Requirement already satisfied: matplotlib>=3.3.4 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (3.3.4)\n", + "Requirement already satisfied: psutil>=5.8.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (5.9.5)\n", + "Requirement already satisfied: tensorboard>=2.4.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (2.13.0)\n", + "Requirement already satisfied: setuptools>=21.0.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (57.5.0)\n", + "Requirement already satisfied: coverage~=5.3.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (5.3.1)\n", + "Requirement already satisfied: torchvision>=0.10.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (0.14.1)\n", + "Requirement already satisfied: sphinx~=4.0.2 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (4.0.3)\n", + "Requirement already satisfied: sphinx-rtd-theme in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.2.2)\n", + "Requirement already satisfied: torchmetrics==0.8 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (0.8.0)\n", + "Requirement already satisfied: hydra-core>=1.2.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.3.2)\n", + "Requirement already satisfied: omegaconf in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (2.3.0)\n", + "Requirement already satisfied: onnxruntime==1.13.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.13.1)\n", + "Requirement already satisfied: onnx==1.13.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.13.0)\n", + "Requirement already satisfied: pillow!=8.3,>=5.3.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (10.0.0)\n", + "Requirement already satisfied: pip-tools>=6.12.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (6.14.0)\n", + "Requirement already satisfied: pyparsing==2.4.5 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (2.4.5)\n", + "Requirement already satisfied: einops==0.3.2 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (0.3.2)\n", + "Requirement already satisfied: pycocotools==2.0.6 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (2.0.6)\n", + "Requirement already satisfied: protobuf==3.20.3 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (3.20.3)\n", + "Requirement already satisfied: treelib==1.6.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.6.1)\n", + "Requirement already satisfied: termcolor==1.1.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.1.0)\n", + "Requirement already satisfied: packaging>=20.4 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (21.0)\n", + "Requirement already satisfied: wheel>=0.38.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (0.40.0)\n", + "Requirement already satisfied: pygments>=2.7.4 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (2.15.1)\n", + "Requirement already satisfied: stringcase>=1.2.0 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.2.0)\n", + "Requirement already satisfied: numpy<=1.23 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (1.23.0)\n", + "Requirement already satisfied: rapidfuzz in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (3.1.1)\n", + "Requirement already satisfied: json-tricks==3.16.1 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (3.16.1)\n", + "Requirement already satisfied: onnx-simplifier<1.0,>=0.3.6 in /usr/local/lib/python3.8/site-packages (from super-gradients==3.1.2) (0.4.33)\n", + "Requirement already satisfied: typing-extensions>=3.6.2.1 in /usr/local/lib/python3.8/site-packages (from onnx==1.13.0->super-gradients==3.1.2) (4.7.1)\n", + "Requirement already satisfied: coloredlogs in /usr/local/lib/python3.8/site-packages (from onnxruntime==1.13.1->super-gradients==3.1.2) (15.0.1)\n", + "Requirement already satisfied: flatbuffers in /usr/local/lib/python3.8/site-packages (from onnxruntime==1.13.1->super-gradients==3.1.2) (23.5.26)\n", + "Requirement already satisfied: sympy in /usr/local/lib/python3.8/site-packages (from onnxruntime==1.13.1->super-gradients==3.1.2) (1.12)\n", + "Requirement already satisfied: pyDeprecate==0.3.* in /usr/local/lib/python3.8/site-packages (from torchmetrics==0.8->super-gradients==3.1.2) (0.3.2)\n", + "Requirement already satisfied: future in /usr/local/lib/python3.8/site-packages (from treelib==1.6.1->super-gradients==3.1.2) (0.18.3)\n", + "Requirement already satisfied: botocore<1.32.0,>=1.31.2 in /usr/local/lib/python3.8/site-packages (from boto3>=1.17.15->super-gradients==3.1.2) (1.31.2)\n", + "Requirement already satisfied: jmespath<2.0.0,>=0.7.1 in /usr/local/lib/python3.8/site-packages (from boto3>=1.17.15->super-gradients==3.1.2) (1.0.1)\n", + "Requirement already satisfied: s3transfer<0.7.0,>=0.6.0 in /usr/local/lib/python3.8/site-packages (from boto3>=1.17.15->super-gradients==3.1.2) (0.6.1)\n", + "Requirement already satisfied: wrapt<2,>=1.10 in /usr/local/lib/python3.8/site-packages (from Deprecated>=1.2.11->super-gradients==3.1.2) (1.15.0)\n", + "Requirement already satisfied: antlr4-python3-runtime==4.9.* in /usr/local/lib/python3.8/site-packages (from hydra-core>=1.2.0->super-gradients==3.1.2) (4.9.3)\n", + "Requirement already satisfied: importlib-resources in /usr/local/lib/python3.8/site-packages (from hydra-core>=1.2.0->super-gradients==3.1.2) (5.12.0)\n", + "Requirement already satisfied: attrs>=17.4.0 in /usr/local/lib/python3.8/site-packages (from jsonschema>=3.2.0->super-gradients==3.1.2) (23.1.0)\n", + "Requirement already satisfied: pkgutil-resolve-name>=1.3.10 in /usr/local/lib/python3.8/site-packages (from jsonschema>=3.2.0->super-gradients==3.1.2) (1.3.10)\n", + "Requirement already satisfied: pyrsistent!=0.17.0,!=0.17.1,!=0.17.2,>=0.14.0 in /usr/local/lib/python3.8/site-packages (from jsonschema>=3.2.0->super-gradients==3.1.2) (0.19.3)\n", + "Requirement already satisfied: cycler>=0.10 in /usr/local/lib/python3.8/site-packages (from matplotlib>=3.3.4->super-gradients==3.1.2) (0.11.0)\n", + "Requirement already satisfied: kiwisolver>=1.0.1 in /usr/local/lib/python3.8/site-packages (from matplotlib>=3.3.4->super-gradients==3.1.2) (1.4.4)\n", + "Requirement already satisfied: python-dateutil>=2.1 in /usr/local/lib/python3.8/site-packages (from matplotlib>=3.3.4->super-gradients==3.1.2) (2.8.2)\n", + "Requirement already satisfied: PyYAML>=5.1.0 in /usr/local/lib/python3.8/site-packages (from omegaconf->super-gradients==3.1.2) (6.0)\n", + "Requirement already satisfied: rich in /usr/local/lib/python3.8/site-packages (from onnx-simplifier<1.0,>=0.3.6->super-gradients==3.1.2) (13.4.2)\n", + "Requirement already satisfied: build in /usr/local/lib/python3.8/site-packages (from pip-tools>=6.12.1->super-gradients==3.1.2) (0.10.0)\n", + "Requirement already satisfied: click>=8 in /usr/local/lib/python3.8/site-packages (from pip-tools>=6.12.1->super-gradients==3.1.2) (8.1.5)\n", + "Requirement already satisfied: pip>=22.2 in /usr/local/lib/python3.8/site-packages (from pip-tools>=6.12.1->super-gradients==3.1.2) (23.1.2)\n", + "Requirement already satisfied: tomli in /usr/local/lib/python3.8/site-packages (from pip-tools>=6.12.1->super-gradients==3.1.2) (2.0.1)\n" + ] + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Requirement already satisfied: sphinxcontrib-applehelp in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (1.0.4)\n", + "Requirement already satisfied: sphinxcontrib-devhelp in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (1.0.2)\n", + "Requirement already satisfied: sphinxcontrib-jsmath in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (1.0.1)\n", + "Requirement already satisfied: sphinxcontrib-htmlhelp in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (2.0.1)\n", + "Requirement already satisfied: sphinxcontrib-serializinghtml in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (1.1.5)\n", + "Requirement already satisfied: sphinxcontrib-qthelp in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (1.0.3)\n", + "Requirement already satisfied: Jinja2>=2.3 in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (3.1.2)\n", + "Requirement already satisfied: docutils<0.18,>=0.14 in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (0.17.1)\n", + "Requirement already satisfied: snowballstemmer>=1.1 in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (2.2.0)\n", + "Requirement already satisfied: babel>=1.3 in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (2.12.1)\n", + "Requirement already satisfied: alabaster<0.8,>=0.7 in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (0.7.13)\n", + "Requirement already satisfied: imagesize in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (1.4.1)\n", + "Requirement already satisfied: requests>=2.5.0 in /usr/local/lib/python3.8/site-packages (from sphinx~=4.0.2->super-gradients==3.1.2) (2.31.0)\n", + "Requirement already satisfied: absl-py>=0.4 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (1.0.0)\n", + "Requirement already satisfied: grpcio>=1.48.2 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (1.56.0)\n", + "Requirement already satisfied: google-auth<3,>=1.6.3 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (2.21.0)\n", + "Requirement already satisfied: google-auth-oauthlib<1.1,>=0.5 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (1.0.0)\n", + "Requirement already satisfied: markdown>=2.6.8 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (3.4.3)\n", + "Requirement already satisfied: tensorboard-data-server<0.8.0,>=0.7.0 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (0.7.1)\n", + "Requirement already satisfied: werkzeug>=1.0.1 in /usr/local/lib/python3.8/site-packages (from tensorboard>=2.4.1->super-gradients==3.1.2) (2.3.6)\n", + "Requirement already satisfied: nvidia-cuda-runtime-cu11==11.7.99 in /usr/local/lib/python3.8/site-packages (from torch>=1.9.0->super-gradients==3.1.2) (11.7.99)\n", + "Requirement already satisfied: nvidia-cudnn-cu11==8.5.0.96 in /usr/local/lib/python3.8/site-packages (from torch>=1.9.0->super-gradients==3.1.2) (8.5.0.96)\n", + "Requirement already satisfied: nvidia-cublas-cu11==11.10.3.66 in /usr/local/lib/python3.8/site-packages (from torch>=1.9.0->super-gradients==3.1.2) (11.10.3.66)\n", + "Requirement already satisfied: nvidia-cuda-nvrtc-cu11==11.7.99 in /usr/local/lib/python3.8/site-packages (from torch>=1.9.0->super-gradients==3.1.2) (11.7.99)\n", + "Requirement already satisfied: sphinxcontrib-jquery<5,>=4 in /usr/local/lib/python3.8/site-packages (from sphinx-rtd-theme->super-gradients==3.1.2) (4.1)\n", + "Requirement already satisfied: six in /usr/local/lib/python3.8/site-packages (from absl-py>=0.4->tensorboard>=2.4.1->super-gradients==3.1.2) (1.16.0)\n", + "Requirement already satisfied: pytz>=2015.7 in /usr/local/lib/python3.8/site-packages (from babel>=1.3->sphinx~=4.0.2->super-gradients==3.1.2) (2023.3)\n", + "Requirement already satisfied: urllib3<1.27,>=1.25.4 in /usr/local/lib/python3.8/site-packages (from botocore<1.32.0,>=1.31.2->boto3>=1.17.15->super-gradients==3.1.2) (1.26.16)\n", + "Requirement already satisfied: cachetools<6.0,>=2.0.0 in /usr/local/lib/python3.8/site-packages (from google-auth<3,>=1.6.3->tensorboard>=2.4.1->super-gradients==3.1.2) (5.3.1)\n", + "Requirement already satisfied: pyasn1-modules>=0.2.1 in /usr/local/lib/python3.8/site-packages (from google-auth<3,>=1.6.3->tensorboard>=2.4.1->super-gradients==3.1.2) (0.3.0)\n", + "Requirement already satisfied: rsa<5,>=3.1.4 in /usr/local/lib/python3.8/site-packages (from google-auth<3,>=1.6.3->tensorboard>=2.4.1->super-gradients==3.1.2) (4.9)\n", + "Requirement already satisfied: requests-oauthlib>=0.7.0 in /usr/local/lib/python3.8/site-packages (from google-auth-oauthlib<1.1,>=0.5->tensorboard>=2.4.1->super-gradients==3.1.2) (1.3.1)\n", + "Requirement already satisfied: zipp>=3.1.0 in /usr/local/lib/python3.8/site-packages (from importlib-resources->hydra-core>=1.2.0->super-gradients==3.1.2) (3.15.0)\n", + "Requirement already satisfied: MarkupSafe>=2.0 in /usr/local/lib/python3.8/site-packages (from Jinja2>=2.3->sphinx~=4.0.2->super-gradients==3.1.2) (2.1.3)\n", + "Requirement already satisfied: importlib-metadata>=4.4 in /usr/local/lib/python3.8/site-packages (from markdown>=2.6.8->tensorboard>=2.4.1->super-gradients==3.1.2) (6.7.0)\n", + "Requirement already satisfied: charset-normalizer<4,>=2 in /usr/local/lib/python3.8/site-packages (from requests>=2.5.0->sphinx~=4.0.2->super-gradients==3.1.2) (3.1.0)\n", + "Requirement already satisfied: idna<4,>=2.5 in /usr/local/lib/python3.8/site-packages (from requests>=2.5.0->sphinx~=4.0.2->super-gradients==3.1.2) (3.4)\n", + "Requirement already satisfied: certifi>=2017.4.17 in /usr/local/lib/python3.8/site-packages (from requests>=2.5.0->sphinx~=4.0.2->super-gradients==3.1.2) (2023.5.7)\n", + "Requirement already satisfied: pyproject_hooks in /usr/local/lib/python3.8/site-packages (from build->pip-tools>=6.12.1->super-gradients==3.1.2) (1.0.0)\n", + "Requirement already satisfied: humanfriendly>=9.1 in /usr/local/lib/python3.8/site-packages (from coloredlogs->onnxruntime==1.13.1->super-gradients==3.1.2) (10.0)\n", + "Requirement already satisfied: markdown-it-py>=2.2.0 in /usr/local/lib/python3.8/site-packages (from rich->onnx-simplifier<1.0,>=0.3.6->super-gradients==3.1.2) (3.0.0)\n", + "Requirement already satisfied: mpmath>=0.19 in /usr/local/lib/python3.8/site-packages (from sympy->onnxruntime==1.13.1->super-gradients==3.1.2) (1.3.0)\n", + "Requirement already satisfied: mdurl~=0.1 in /usr/local/lib/python3.8/site-packages (from markdown-it-py>=2.2.0->rich->onnx-simplifier<1.0,>=0.3.6->super-gradients==3.1.2) (0.1.2)\n", + "Requirement already satisfied: pyasn1<0.6.0,>=0.4.6 in /usr/local/lib/python3.8/site-packages (from pyasn1-modules>=0.2.1->google-auth<3,>=1.6.3->tensorboard>=2.4.1->super-gradients==3.1.2) (0.5.0)\n", + "Requirement already satisfied: oauthlib>=3.0.0 in /usr/local/lib/python3.8/site-packages (from requests-oauthlib>=0.7.0->google-auth-oauthlib<1.1,>=0.5->tensorboard>=2.4.1->super-gradients==3.1.2) (3.2.2)\n", + "\u001b[33mWARNING: Running pip as the 'root' user can result in broken permissions and conflicting behaviour with the system package manager. It is recommended to use a virtual environment instead: https://pip.pypa.io/warnings/venv\u001b[0m\u001b[33m\n", + "\u001b[0m" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "/usr/local/lib/python3.8/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n", + " from .autonotebook import tqdm as notebook_tqdm\n", + "[2023-07-13 17:15:54] INFO - crash_tips_setup.py - Crash tips is enabled. You can set your environment variable to CRASH_HANDLER=FALSE to disable it\n" + ] + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + "The console stream is logged into /root/sg_logs/console.log\n" + ] + }, + { + "name": "stderr", + "output_type": "stream", + "text": [ + "[2023-07-13 17:15:57] WARNING - __init__.py - Failed to import pytorch_quantization\n", + "[2023-07-13 17:15:57] WARNING - calibrator.py - Failed to import pytorch_quantization\n", + "[2023-07-13 17:15:57] WARNING - export.py - Failed to import pytorch_quantization\n", + "[2023-07-13 17:15:57] WARNING - selective_quantization_utils.py - Failed to import pytorch_quantization\n", + "[2023-07-13 17:15:58] INFO - checkpoint_utils.py - License Notification: YOLO-NAS pre-trained weights are subjected to the specific license terms and conditions detailed in \n", + "https://github.com/Deci-AI/super-gradients/blob/master/LICENSE.YOLONAS.md\n", + "By downloading the pre-trained weight files you agree to comply with these terms.\n" + ] + } + ], + "source": [ + "## Note- Use python3.8 or above for generating onnx\n", + "\n", + "!pip install super-gradients==3.1.2\n", + "\n", + "\n", + "## Downloading Model from git repo\n", + "import torch\n", + "# Load model with pretrained weights\n", + "from super_gradients.training import models\n", + "from super_gradients.common.object_names import Models\n", + "\n", + "model = models.get(Models.YOLO_NAS_S, pretrained_weights=\"coco\")\n", + "\n", + "# Prepare model for conversion\n", + "# Input size is in format of [Batch x Channels x Width x Height] where 640 is the standard COCO dataset dimensions\n", + "model.eval()\n", + "model.prep_model_for_conversion(input_size=[1, 3, 320, 320])\n", + "\n", + "# Create dummy_input\n", + "dummy_input = torch.randn([1, 3, 320, 320], device=\"cpu\")\n", + "\n", + "# Convert model to onnx\n", + "torch.onnx.export(model, dummy_input, \"yolo_nas_s.onnx\", opset_version=11)" + ] + }, + { + "attachments": {}, + "cell_type": "markdown", + "id": "bb97b534", + "metadata": {}, + "source": [ + "#### Enable python3.6 environment, to use SNPE SDK and then convert onnx to dlc" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "c466b9aa", + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "2023-07-13 17:16:03,073 - 235 - INFO - Successfully simplified the onnx model in child process\n", + "2023-07-13 17:16:03,547 - 235 - INFO - Successfully receive the simplified onnx model in main process\n", + "2023-07-13 17:16:06,272 - 235 - INFO - INFO_INITIALIZATION_SUCCESS: \n" + ] + } + ], + "source": [ + "%%bash\n", + "snpe-onnx-to-dlc -i yolo_nas_s.onnx -o app/src/main/assets/yolo_nas_s.dlc" + ] + }, + { + "attachments": {}, + "cell_type": "markdown", + "id": "b2675610", + "metadata": {}, + "source": [ + "## Quantizing MobileNetSSD" + ] + }, + { + "cell_type": "code", + "execution_count": 27, + "id": "b01ac9cf", + "metadata": {}, + "outputs": [], + "source": [ + "##STEPS to preprocess images\n", + "\n", + "def preprocess(original_image):\n", + " resized_image = cv2.resize(original_image, (320, 320))\n", + " resized_image = resized_image/255\n", + " return resized_image\n", + "\n", + "import cv2\n", + "import numpy as np\n", + "import os\n", + "\n", + "##Please download Coco2014 dataset and give the path here\n", + "dataset_path = \"/workspace/val2014/\"\n", + "\n", + "!mkdir -p rawYoloNAS\n", + "\n", + "filenames=[]\n", + "for path in os.listdir(dataset_path)[:5]:\n", + " # check if current path is a file\n", + " if os.path.isfile(os.path.join(dataset_path, path)):\n", + " filenames.append(os.path.join(dataset_path, path))\n", + "\n", + "for filename in filenames:\n", + " original_image = cv2.imread(filename)\n", + " img = preprocess(original_image)\n", + " img = img.astype(np.float32)\n", + " img.tofile(\"rawYoloNAS/\"+filename.split(\"/\")[-1].split(\".\")[0]+\".raw\")" + ] + }, + { + "cell_type": "code", + "execution_count": 19, + "id": "7370c51c", + "metadata": {}, + "outputs": [], + "source": [ + "%%bash\n", + "find rawYoloNAS -name *.raw > YoloInputlist.txt" + ] + }, + { + "cell_type": "code", + "execution_count": 20, + "id": "40b37c70", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "rawYoloNAS/COCO_val2014_000000000400.raw\n", + "rawYoloNAS/COCO_val2014_000000000042.raw\n", + "rawYoloNAS/COCO_val2014_000000000073.raw\n", + "rawYoloNAS/COCO_val2014_000000000074.raw\n", + "rawYoloNAS/COCO_val2014_000000000133.raw\n" + ] + } + ], + "source": [ + "%%bash\n", + "cat YoloInputlist.txt" + ] + }, + { + "cell_type": "code", + "execution_count": 21, + "id": "39b97591", + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "[INFO] InitializeStderr: DebugLog initialized.\n", + "[INFO] Processed command-line arguments\n", + "[INFO] Quantized parameters\n", + "[INFO] Generated activations\n", + "[INFO] Saved quantized dlc to: app/src/main/assets/Quant_yoloNas_s_320.dlc\n", + "[INFO] DebugLog shutting down.\n" + ] + }, + { + "name": "stdout", + "output_type": "stream", + "text": [ + " 0.1ms [ INFO ] Initializing logging in the backend. Callback: [0xc42410], Log Level: [3]\n", + " 0.1ms [ INFO ] No BackendExtensions lib provided;initializing NetRunBackend Interface\n", + " 875.4ms [ INFO ] cleaning up resources for input tensors\n", + " 875.5ms [ INFO ] cleaning up resources for output tensors\n", + " 1524.6ms [ INFO ] cleaning up resources for input tensors\n", + " 1524.6ms [ INFO ] cleaning up resources for output tensors\n", + " 2136.3ms [ INFO ] cleaning up resources for input tensors\n", + " 2136.3ms [ INFO ] cleaning up resources for output tensors\n", + " 2852.1ms [ INFO ] cleaning up resources for input tensors\n", + " 2852.1ms [ INFO ] cleaning up resources for output tensors\n", + " 3461.0ms [ INFO ] cleaning up resources for input tensors\n", + " 3461.0ms [ INFO ] cleaning up resources for output tensors\n" + ] + } + ], + "source": [ + "%%bash\n", + "snpe-dlc-quantize --input_dlc app/src/main/assets/yolo_nas_s.dlc --input_list YoloInputlist.txt --use_enhanced_quantizer --use_adjusted_weights_quantizer --axis_quant --output_dlc app/src/main/assets/Quant_yoloNas_s_320.dlc" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a13d7629", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.8.17" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/ai-solutions/android/03-ObjectDetection/README.md b/ai-solutions/android/03-ObjectDetection/README.md new file mode 100644 index 00000000..d9917ca0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/README.md @@ -0,0 +1,215 @@ +## Object Detection with YoloNAS / SSDMobilenetV2 / YoloX +The project is designed to utilize the [Qualcomm® Neural Processing SDK for AI ](https://developer.qualcomm.com/sites/default/files/docs/snpe/index.html), a deep learning software from Snapdragon platforms for Object Detection in Android. The Android application can be designed to use any built-in/connected camera to capture the objects and use Machine Learning model to get the prediction/inference and location of the respective objects. + +# Pre-requisites + +* Before starting the Android application, please follow the instructions for setting up Qualcomm Neural Processing SDK using the link provided. https://developer.qualcomm.com/sites/default/files/docs/snpe/setup.html +* Android device 6.0 and above which uses below mentioned Snapdragon processors/Snapdragon HDK with display can be used to test the application +* Download CocoDataset 2014 and give its path to Generate_DLC.ipynb. Change variable "dataset_path" in Quantization Section in notebook. + + +## List of Supported Devices + +- Snapdragon® SM8550 + +The above targets supports the application with CPU, GPU and DSP. For more information on the supported devices, please follow this link https://developer.qualcomm.com/docs/snpe/overview.html + +# Source Overview + +## Source Organization + +demo : Contains demo GIF + +app : Contains source files in standard Android app format + +app\src\main\assets : Contains Model binary DLC + +app\src\main\java\com\qc\objectdetectionYoloNas : Application java source code + +app\src\main\cpp : native source code + +sdk: Contains openCV sdk + +## DLC Generation + +Run jupyter notebook GenerateDLC.ipynb. This notebook will generate YoloNAS quantized dlc. + +YoloNAS model is trained on COCO dataset for 80 classes of everyday objects. +List of the classes can be found in dataset at : https://cocodataset.org/#explore + +## Code Implementation + +This application opens a camera preview, collects all the frames and converts them to bitmap. The network is built via Neural Network builder by passing model dlc name and runtime as the input. The bitmap is then given to the model for inference, which returns object prediction and localization of the respective object. + + +### Prerequisite for Camera Preview. + +Permission to obtain camera preview frames is granted in the following file: +```python +/app/src/main/AndroidManifest.xml + + ``` +In order to use camera2 APIs, add the below feature +```python + +``` +### Loading Model +Code snippet for neural network connection and loading model: +```java + snpe = snpeBuilder.setOutputLayers({}) + .setPerformanceProfile(zdl::DlSystem::PerformanceProfile_t::BURST) + .setExecutionPriorityHint( + zdl::DlSystem::ExecutionPriorityHint_t::HIGH) + .setRuntimeProcessorOrder(runtimeList) + .setUseUserSuppliedBuffers(useUserSuppliedBuffers) + .setPlatformConfig(platformConfig) + .setInitCacheMode(useCaching) + .setCPUFallbackMode(true) + .setUnconsumedTensorsAsOutputs(true) + .build(); +``` +### Preprocessing +The bitmap image is passed as openCV Mat to native and then converted to BGR Mat. DLC models can work with specific image sizes. +Therefore, we need to resize the input image to the size accepted by the corresponding selected model DLC before passing image to DLC. +Below code reference for YoloNAS preprocessing. Similarly for other models based on model requirements, the preprocessing may change. +```java + cv::Mat img320; + //Resize and get the size from model itself (320x320 for YOLONAS) + cv::resize(img,img320,cv::Size(dims[2],dims[1]),cv::INTER_LINEAR); + + float inputScale = 0.00392156862745f; + + float * accumulator = reinterpret_cast (&dest_buffer[0]); + + //opencv read in BGRA by default + cvtColor(img320, img320, CV_BGRA2BGR); + int lim = img320.rows*img320.cols*3; + for(int idx = 0; idx=0.5 ) + { + int x1 = BBout_boxcoords[i * 4 + 0]; + int y1 = BBout_boxcoords[i * 4 + 1]; + int x2 = BBout_boxcoords[i * 4 + 2]; + int y2 = BBout_boxcoords[i * 4 + 3]; + Boxlist.push_back(BoxCornerEncoding(x1, y1, x2, y2,*it,classname)); + } + } + + std::vector reslist = NonMaxSuppression(Boxlist,0.20); +``` +then we just scale the coords for original image + +```python + float top,bottom,left,right; + left = reslist[k].y1 * ratio_1; //y1 + right = reslist[k].y2 * ratio_1; //y2 + + bottom = reslist[k].x1 * ratio_2; //x1 + top = reslist[k].x2 * ratio_2; //x2 +``` + +## Drawing bounding boxes + +```python + RectangleBox rbox = boxlist.get(j); + float y = rbox.left; + float y1 = rbox.right; + float x = rbox.top; + float x1 = rbox.bottom; + + String fps_textLabel = "FPS: "+String.valueOf(rbox.fps); + canvas.drawText(fps_textLabel,10,70,mTextColor); + + String processingTimeTextLabel= rbox.processing_time+"ms"; + + canvas.drawRect(x1, y, x, y1, mBorderColor); + canvas.drawText(rbox.label,x1+10, y+40, mTextColor); + canvas.drawText(processingTimeTextLabel,x1+10, y+90, mTextColor); +``` + +# Build and run with Android Studio + +## Build APK file with Android Studio + +1. Clone QIDK repo. + +2. Run below script, from the directory where it is present, to resolve dependencies of this project. + +* This will copy snpe-release.aar file from $SNPE_ROOT to "snpe-release" directory in Android project. + + **NOTE - If you are using SNPE version 2.11 or greater, please change following line in resolveDependencies.sh.** + ``` + From: cp $SNPE_ROOT/android/snpe-release.aar snpe-release + To : cp $SNPE_ROOT/lib/android/snpe-release.aar snpe-release + ``` +* Download opencv and paste to sdk directory, to enable OpenCv for android Java. + +```java + bash resolveDependencies.sh +``` + + +3. Run jupyter notebook GenerateDLC.ipynb to generate DLC(s) for quantized YOLO_NAS DLC. Also, **change the dataset_path with Coco Dataset Path**. +* This script generates required dlc(s) and paste them to appropriate location. + + +4. Do gradle sync +5. Compile the project. +6. Output APK file should get generated : app-debug.apk +7. Prepare the Qualcomm Innovators development kit to install the application (Do not run APK on emulator) + +8. If Unsigned or Signed DSP runtime is not getting detected, then please check the logcat logs for the FastRPC error. DSP runtime may not get detected due to SE Linux security policy. Please try out following commands to set permissive SE Linux policy. + +It is recommended to run below commands. +```java +adb disable-verity +adb reboot +adb root +adb remount +adb shell setenforce 0 +``` + +9. Install and test application : app-debug.apk +```java +adb install -r -t app-debug.apk +``` + +10. launch the application + +Following is the basic "Pose Detection" Android App + +1. On launch of application, from home screen user can select the model and runtime and then press start camera button. +2. On first launch of camera, user needs to provide camera permissions. +3. After camera launched, the selected model with runtime starts loading in the background. User will see a dialogue box till model is being loaded. +4. Once the model is loaded, it will start detecting objects and box will be seen around the object if respective object is detected on the screen +5. User can go back to home screen by pressing back button and select appropriate model and run-time and observe performance difference. + +Same results for the application are : + +## Demo of the application +![Screenshot](.//demo/ObjectDetectYoloNAS.gif) + +# References +1. SSD - Single shot Multi box detector - https://arxiv.org/pdf/1512.02325.pdf +2. https://github.com/Deci-AI/super-gradients +3. https://zenodo.org/record/7789328 + + +###### *Snapdragon and Qualcomm Neural Processing SDK are products of Qualcomm Technologies, Inc. and/or its subsidiaries.* diff --git a/ai-solutions/android/03-ObjectDetection/app/build.gradle b/ai-solutions/android/03-ObjectDetection/app/build.gradle new file mode 100644 index 00000000..4f8293a0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/build.gradle @@ -0,0 +1,68 @@ +apply plugin: 'com.android.application' + +android { + compileSdkVersion 30 + buildToolsVersion "30.0.3" + + defaultConfig { + applicationId "com.qcom.aistack_objdetect" + minSdkVersion 24 + targetSdkVersion 30 + versionCode 1 + versionName "1.0" + + testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" + externalNativeBuild { + cmake { +// cppFlags '' + cppFlags "-std=c++11 -frtti -fexceptions" + arguments "-DOpenCV_DIR=" + project(':sdk').projectDir + "/native/jni", + "-DANDROID_TOOLCHAIN=clang" +// "-DANDROID_STL=c++_shared", +// "-DANDROID_ARM_NEON=TRUE" + targets "objectdetectionYoloNas" + } + ndk { + abiFilters 'arm64-v8a' + } + } + } + + packagingOptions { + pickFirst 'lib/x86/libc++_shared.so' + pickFirst 'lib/x86_64/libc++_shared.so' + pickFirst 'lib/arm64-v8a/libc++_shared.so' + pickFirst 'lib/armeabi-v7a/libc++_shared.so' + } + + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } + + compileOptions { + sourceCompatibility JavaVersion.VERSION_1_8 + targetCompatibility JavaVersion.VERSION_1_8 + } + ndkVersion '21.4.7075529' + externalNativeBuild { + cmake { + path file('src/main/cpp/CMakeLists.txt') + } + } +} + +dependencies { + implementation fileTree(dir: 'libs', include: ['*.jar']) + implementation project(path: ':sdk') + testImplementation 'junit:junit:4.12' + androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.1' + androidTestImplementation 'com.android.support.test.espresso:espresso-contrib:3.0.1' + implementation 'com.android.support:design:26.0.0' + implementation 'com.android.support:support-v4:26.0.0' + + + +} diff --git a/ai-solutions/android/03-ObjectDetection/app/local.properties b/ai-solutions/android/03-ObjectDetection/app/local.properties new file mode 100644 index 00000000..0a5b4775 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/local.properties @@ -0,0 +1,8 @@ +## This file must *NOT* be checked into Version Control Systems, +# as it contains information specific to your local configuration. +# +# Location of the SDK. This is only used by Gradle. +# For customization when using a Version Control System, please read the +# header note. +#Sat Jan 07 01:53:02 IST 2023 +sdk.dir=C\:\\Users\\shubgoya\\AppData\\Local\\Android\\Sdk diff --git a/ai-solutions/android/03-ObjectDetection/app/proguard-rules.pro b/ai-solutions/android/03-ObjectDetection/app/proguard-rules.pro new file mode 100644 index 00000000..6e7ffa99 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/proguard-rules.pro @@ -0,0 +1,21 @@ +# Add project specific ProGuard rules here. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/AndroidManifest.xml b/ai-solutions/android/03-ObjectDetection/app/src/main/AndroidManifest.xml new file mode 100644 index 00000000..e387bd86 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/AndroidManifest.xml @@ -0,0 +1,38 @@ + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/assets/ReadMe.txt b/ai-solutions/android/03-ObjectDetection/app/src/main/assets/ReadMe.txt new file mode 100644 index 00000000..aca5c44b --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/assets/ReadMe.txt @@ -0,0 +1 @@ +Generate model DLC and place here \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/CMakeLists.txt b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/CMakeLists.txt new file mode 100644 index 00000000..adeaf8e0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/CMakeLists.txt @@ -0,0 +1,66 @@ + +# For more information about using CMake with Android Studio, read the +# documentation: https://d.android.com/studio/projects/add-native-code.html + +# Sets the minimum version of CMake required to build the native library. + +cmake_minimum_required(VERSION 3.18.1) + +# Declares and names the project. + +project("objectdetectionYoloNas") + +# Creates and names a library, sets it as either STATIC +# or SHARED, and provides the relative paths to its source code. +# You can define multiple libraries, and CMake builds them for you. +# Gradle automatically packages shared libraries with your APK. + +###OPENCV +#find_package(OpenCV REQUIRED) ##FAILED, cannot find libcpufeatures.so +#set(OpenCV_STATIC on) +#set(OpenCV_DIR C:/Users/shubgoya/Desktop/SNPEworkspace/github_workspace/HRNET_posenet/opencv45/native/jni) +find_package(OpenCV REQUIRED) +#INCLUDE_DIRECTORIES(${OpenCV_INCLUDE_DIRS}) + + +###INCLUDE_DIRECTORIES +include_directories(${CMAKE_CURRENT_SOURCE_DIR}/inc/) +include_directories(${CMAKE_CURRENT_SOURCE_DIR}/inc/zdl) +include_directories(${CMAKE_CURRENT_SOURCE_DIR}/inc/hpp) +include_directories(${CMAKE_CURRENT_SOURCE_DIR}) + +add_library( # Sets the name of the library. + objectdetectionYoloNas + + # Sets the library as a shared library. + SHARED + + # Provides a relative path to your source file(s). + inference.cpp inference_helper.cpp objectdetectionYoloNas.cpp Model.h Model.cpp YOLONAS_Model.h YOLONAS_Model.cpp + SSDMobileNetV2_Model.h SSDMobileNetV2_Model.cpp YOLO_X_Model.h YOLO_X_Model.cpp) + +# Searches for a specified prebuilt library and stores the path as a +# variable. Because CMake includes system libraries in the search path by +# default, you only need to specify the name of the public NDK library +# you want to add. CMake verifies that the library exists before +# completing its build. + +find_library( # Sets the name of the path variable. + log-lib + + # Specifies the name of the NDK library that + # you want CMake to locate. + log ) + +# Specifies libraries CMake should link to your target library. You +# can link multiple libraries, such as libraries you define in this +# build script, prebuilt third-party libraries, or system libraries. + +target_link_libraries( # Specifies the target library. + objectdetectionYoloNas + + # Links the target library to the log library + # included in the NDK. + ${CMAKE_CURRENT_SOURCE_DIR}/../jniLibs/arm64-v8a/libSNPE.so + + ${log-lib} ${OpenCV_LIBS}) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.cpp new file mode 100644 index 00000000..c15814fc --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.cpp @@ -0,0 +1,14 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 8/30/2023. +// + +// #include "Model.h" diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.h new file mode 100644 index 00000000..06e5d491 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/Model.h @@ -0,0 +1,60 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 8/30/2023. +// + +#ifndef APP_MODEL_H +#define APP_MODEL_H + + +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include "android/log.h" + + +#include +#include +#include + +#define LOG_TAG "SNPE_INF" +#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) +#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) + +// List of All the supported models by the current application +enum ModelName +{ + YOLONAS, + SSDMobilenetV2, + YoloX +}; + +class Model { + +public: + virtual void preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims) = 0; + virtual void postprocess(int orig_width, int orig_height, int &numberofobj, std::vector> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time) = 0; + virtual void msg() = 0; + + ModelName model_name=YOLONAS; //initialized + +}; + + +#endif //APP_MODEL_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.cpp new file mode 100644 index 00000000..72750b9e --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.cpp @@ -0,0 +1,111 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 9/11/2023. +// + +#include "SSDMobileNetV2_Model.h" + + +void SSDMobileNetV2_Model::preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims) +{ + LOGI("SSDMobileNetV2_Model preprocess"); + cv::Mat img320; + //Resize and get the size from model itself (320x320 for SSDMobileNetV2) + cv::resize(img,img320,cv::Size(dims[2],dims[1]),0,0,cv::INTER_LINEAR); + + //float inputScale = 0.00392156862745f; //normalization value, this is 1/255 + float inputScale = 1.070312500000f; //normalization value, this is 1/255 + + float * accumulator = reinterpret_cast (&dest_buffer[0]); + + cvtColor(img320, img320, CV_BGRA2RGB); + //LOGI("num of channels: %d",img320.channels()); + int lim = img320.rows*img320.cols*3; + for(int idx = 0; idx> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time) { + LOGI("SSDMobileNetV2_Model postprocess"); + std::vector Boxlist; + std::vector Classlist; + + //sanjeev temp sanity check for sometimes stability issue in SSDMobileNetV2 Model + if (BBout_boxcoords.size() == 0) + { + numberofobj=-1; + LOGE("sanjeev BBout_boxcoords is zero. Returning Error.."); + return; + } + + //Post Processing + for(int i =1;i<(21);i++) // [21 classes supported by SSDMobileNetV2] + { + + int row_index; + float max_element; + + std::string classname = classnamemapping[i]; + + for (int j=i; j<(67914); j+=21) // [67914 = 21 (no of classes) x 3234 (total boxes output by model)] + { + if (BBout_class[j] > 0.4) + { + max_element = BBout_class[j]; + row_index = j/21; + + float x1 = BBout_boxcoords[row_index * 4 + 0]; + float y1 = BBout_boxcoords[row_index * 4 + 1]; + float x2 = BBout_boxcoords[row_index * 4 + 2]; + float y2 = BBout_boxcoords[row_index * 4 + 3]; + + Boxlist.push_back(SSDMobileNetV2BoxCornerEncoding(x1, y1, x2, y2,max_element,classname)); + } + } + + } + + //LOGI("Boxlist size:: %d",Boxlist.size()); + std::vector reslist = NonMaxSuppression(Boxlist,0.20); + //LOGI("reslist ssize %d", reslist.size()); + + numberofobj = reslist.size(); + + //LOGI("numberofobj detected = %d", numberofobj); + + float ratio_2 = orig_width; + float ratio_1 = orig_height; + + //LOGI("ratio1 %f :: ratio_2 %f",ratio_1,ratio_2); + + for(int k=0;k singleboxcoords{top, bottom, left, right, milli_time}; + BB_coords.push_back(singleboxcoords); + BB_names.push_back(reslist[k].objlabel); + } + +} + +void SSDMobileNetV2_Model::msg() +{ + LOGI("SSDMobileNetV2_Model Class msg model_name = %d", model_name); +} \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.h new file mode 100644 index 00000000..379f2a70 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/SSDMobileNetV2_Model.h @@ -0,0 +1,133 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 9/11/2023. +// + +#ifndef APP_SSDMOBILENETV2_MODEL_H +#define APP_SSDMOBILENETV2_MODEL_H + +#include "Model.h" + +#include + +class SSDMobileNetV2BoxCornerEncoding { + +public: + float x1; + float y1; + float x2; + float y2; + float score; + std::string objlabel; + + SSDMobileNetV2BoxCornerEncoding(float a, float b, float c, float d,float sc, std::string name="person") + { + x1 = a; + y1 = b; + x2 = c; + y2 = d; + score = sc; + objlabel = name; + } +}; + + + +class SSDMobileNetV2_Model: public Model{ + + std::map classnamemapping = + { + {0, "BACKGROUND"},{ 1, "aeroplane"},{ 2, "bicycle"},{ 3, "bird"},{ 4, "boat"},{ 5, "bottle"},{ + 6, "bus"},{ 7, "car"},{ 8, "cat"},{ 9, "chair"},{ 10, "cow"},{ 11, "diningtable"},{ 12, "dog"},{ + 13, "horse"},{ 14, "motorbike"},{ 15, "person"},{ 16, "pottedplant"},{ 17, "sheep"},{ 18, "sofa"},{ 19, "train"},{ + 20, "tvmonitor"} + }; + +public: + + SSDMobileNetV2_Model() + { + model_name=SSDMobilenetV2; + } + + void preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims); + void postprocess(int orig_width, int orig_height, int &numberofobj, std::vector> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time); + void msg(); + + inline float ComputeIntersectionOverUnion(const SSDMobileNetV2BoxCornerEncoding &box_i,const SSDMobileNetV2BoxCornerEncoding &box_j) + { + const float box_i_y_min = std::min(box_i.y1, box_i.y2); + const float box_i_y_max = std::max(box_i.y1, box_i.y2); + const float box_i_x_min = std::min(box_i.x1, box_i.x2); + const float box_i_x_max = std::max(box_i.x1, box_i.x2); + const float box_j_y_min = std::min(box_j.y1, box_j.y2); + const float box_j_y_max = std::max(box_j.y1, box_j.y2); + const float box_j_x_min = std::min(box_j.x1, box_j.x2); + const float box_j_x_max = std::max(box_j.x1, box_j.x2); + + const float area_i = + (box_i_y_max - box_i_y_min) * (box_i_x_max - box_i_x_min); + const float area_j = + (box_j_y_max - box_j_y_min) * (box_j_x_max - box_j_x_min); + if (area_i <= 0 || area_j <= 0) return 0.0; + const float intersection_ymax = std::min(box_i_y_max, box_j_y_max); + const float intersection_xmax = std::min(box_i_x_max, box_j_x_max); + const float intersection_ymin = std::max(box_i_y_min, box_j_y_min); + const float intersection_xmin = std::max(box_i_x_min, box_j_x_min); + const float intersection_area = + std::max(intersection_ymax - intersection_ymin, 0.0) * + std::max(intersection_xmax - intersection_xmin, 0.0); + return intersection_area / (area_i + area_j - intersection_area); + } + + std::vector NonMaxSuppression(std::vector boxes, + const float iou_threshold) + { + + if (boxes.size()==0) { + return boxes; + } + + std::sort(boxes.begin(), boxes.end(), [] (const SSDMobileNetV2BoxCornerEncoding& left, const SSDMobileNetV2BoxCornerEncoding& right) { + if (left.score > right.score) { + return true; + } else { + return false; + } + }); + + + std::vector flag(boxes.size(), false); + for (unsigned int i = 0; i < boxes.size(); i++) { + if (flag[i]) { + continue; + } + + for (unsigned int j = i + 1; j < boxes.size(); j++) { + if (ComputeIntersectionOverUnion(boxes[i],boxes[j]) > iou_threshold) { + flag[j] = true; + } + } + } + + std::vector ret; + for (unsigned int i = 0; i < boxes.size(); i++) { + if (!flag[i]) + ret.push_back(boxes[i]); + } + + return ret; + } + +}; + + +#endif //APP_SSDMOBILENETV2_MODEL_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.cpp new file mode 100644 index 00000000..7fa9b630 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.cpp @@ -0,0 +1,90 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 8/30/2023. +// + +#include "YOLONAS_Model.h" + +void YOLONAS_Model::preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims) +{ + LOGI("YOLONAS_Model preprocess"); + cv::Mat img320; + //Resize and get the size from model itself (320x320 for YOLONAS) + cv::resize(img,img320,cv::Size(dims[2],dims[1]),cv::INTER_LINEAR); + + float inputScale = 0.00392156862745f; //normalization value, this is 1/255 + + float * accumulator = reinterpret_cast (&dest_buffer[0]); + + //opencv read in BGRA by default + cvtColor(img320, img320, CV_BGRA2BGR); + //LOGI("num of channels: %d",img320.channels()); + int lim = img320.rows*img320.cols*3; + for(int idx = 0; idx> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time) { + LOGI("YOLONAS_Model postprocess"); + std::vector Boxlist; + std::vector Classlist; + + //Post Processing + for(int i =0;i<(2100);i++) //TODO change value of 2100 to soft value + { + int start = i*80; + int end = (i+1)*80; + + auto it = max_element (BBout_class.begin()+start, BBout_class.begin()+end); + int index = distance(BBout_class.begin()+start, it); + + std::string classname = classnamemapping[index]; + if(*it>=0.5 ) + { + int x1 = BBout_boxcoords[i * 4 + 0]; + int y1 = BBout_boxcoords[i * 4 + 1]; + int x2 = BBout_boxcoords[i * 4 + 2]; + int y2 = BBout_boxcoords[i * 4 + 3]; + Boxlist.push_back(BoxCornerEncoding(x1, y1, x2, y2,*it,classname)); + } + } + + //LOGI("Boxlist size:: %d",Boxlist.size()); + std::vector reslist = NonMaxSuppression(Boxlist,0.20); + //LOGI("reslist ssize %d", reslist.size()); + + numberofobj = reslist.size(); + float ratio_2 = orig_width/320.0f; + float ratio_1 = orig_height/320.0f; + //LOGI("ratio1 %f :: ratio_2 %f",ratio_1,ratio_2); + + for(int k=0;k singleboxcoords{top, bottom, left, right, milli_time}; + BB_coords.push_back(singleboxcoords); + BB_names.push_back(reslist[k].objlabel); + } +} + +void YOLONAS_Model::msg() +{ + LOGI("YOLONAS_Model Class msg model_name = %d",model_name); +} \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.h new file mode 100644 index 00000000..853af371 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLONAS_Model.h @@ -0,0 +1,142 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 8/30/2023. +// + +#ifndef APP_YOLONAS_MODEL_H +#define APP_YOLONAS_MODEL_H + +#include "Model.h" + +#include + +class BoxCornerEncoding { + +public: + int x1; + int y1; + int x2; + int y2; + float score; + std::string objlabel; + + BoxCornerEncoding(int a, int b, int c, int d,int sc, std::string name="person") + { + x1 = a; + y1 = b; + x2 = c; + y2 = d; + score = sc; + objlabel = name; + } +}; + + +class YOLONAS_Model: public Model{ + + + std::map classnamemapping = + { + {0, "person"},{ 1, "bicycle"},{ 2, "car"},{ 3, "motorcycle"},{ 4, "airplane"},{ 5, "bus"},{ + 6, "train"},{ 7, "truck"},{ 8, "boat"},{ 9, "traffic"},{ 10, "fire"},{ 11, "stop"},{ 12, "parking"},{ + 13, "bench"},{ 14, "bird"},{ 15, "cat"},{ 16, "dog"},{ 17, "horse"},{ 18, "sheep"},{ 19, "cow"},{ + 20, "elephant"},{ 21, "bear"},{ 22, "zebra"},{ 23, "giraffe"},{ 24, "backpack"},{ 25, "umbrella"},{ + 26, "handbag"},{ 27, "tie"},{ 28, "suitcase"},{ 29, "frisbee"},{ 30, "skis"},{ 31, "snowboard"},{ + 32, "sports"},{ 33, "kite"},{ 34, "baseball"},{ 35, "baseball"},{ 36, "skateboard"},{ 37, "surfboard"},{ + 38, "tennis"},{ 39, "bottle"},{ 40, "wine"},{ 41, "cup"},{ 42, "fork"},{ 43, "knife"},{ 44, "spoon"},{ + 45, "bowl"},{ 46, "banana"},{ 47, "apple"},{ 48, "sandwich"},{ 49, "orange"},{ 50, "broccoli"},{ + 51, "carrot"},{ 52, "hot"},{ 53, "pizza"},{ 54, "donut"},{ 55, "cake"},{ 56, "chair"},{ 57, "couch"},{ + 58, "potted"},{ 59, "bed"},{ 60, "dining"},{ 61, "toilet"},{ 62, "tv"},{ 63, "laptop"},{ 64, "mouse"},{ + 65, "remote"},{ 66, "keyboard"},{ 67, "cell"},{ 68, "microwave"},{ 69, "oven"},{ 70, "toaster"},{ + 71, "sink"},{ 72, "refrigerator"},{ 73, "book"},{ 74, "clock"},{ 75, "vase"},{ 76, "scissors"},{ + 77, "teddy"},{ 78, "hair"},{ 79, "toothbrush"} + }; + +public: + + YOLONAS_Model() + { + model_name=YOLONAS; + } + + void preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims); + void postprocess(int orig_width, int orig_height, int &numberofobj, std::vector> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time); + void msg(); + + inline float ComputeIntersectionOverUnion(const BoxCornerEncoding &box_i,const BoxCornerEncoding &box_j) + { + const float box_i_y_min = std::min(box_i.y1, box_i.y2); + const float box_i_y_max = std::max(box_i.y1, box_i.y2); + const float box_i_x_min = std::min(box_i.x1, box_i.x2); + const float box_i_x_max = std::max(box_i.x1, box_i.x2); + const float box_j_y_min = std::min(box_j.y1, box_j.y2); + const float box_j_y_max = std::max(box_j.y1, box_j.y2); + const float box_j_x_min = std::min(box_j.x1, box_j.x2); + const float box_j_x_max = std::max(box_j.x1, box_j.x2); + + const float area_i = + (box_i_y_max - box_i_y_min) * (box_i_x_max - box_i_x_min); + const float area_j = + (box_j_y_max - box_j_y_min) * (box_j_x_max - box_j_x_min); + if (area_i <= 0 || area_j <= 0) return 0.0; + const float intersection_ymax = std::min(box_i_y_max, box_j_y_max); + const float intersection_xmax = std::min(box_i_x_max, box_j_x_max); + const float intersection_ymin = std::max(box_i_y_min, box_j_y_min); + const float intersection_xmin = std::max(box_i_x_min, box_j_x_min); + const float intersection_area = + std::max(intersection_ymax - intersection_ymin, 0.0) * + std::max(intersection_xmax - intersection_xmin, 0.0); + return intersection_area / (area_i + area_j - intersection_area); + } + + std::vector NonMaxSuppression(std::vector boxes, + const float iou_threshold) + { + + if (boxes.size()==0) { + return boxes; + } + + std::sort(boxes.begin(), boxes.end(), [] (const BoxCornerEncoding& left, const BoxCornerEncoding& right) { + if (left.score > right.score) { + return true; + } else { + return false; + } + }); + + + std::vector flag(boxes.size(), false); + for (unsigned int i = 0; i < boxes.size(); i++) { + if (flag[i]) { + continue; + } + + for (unsigned int j = i + 1; j < boxes.size(); j++) { + if (ComputeIntersectionOverUnion(boxes[i],boxes[j]) > iou_threshold) { + flag[j] = true; + } + } + } + + std::vector ret; + for (unsigned int i = 0; i < boxes.size(); i++) { + if (!flag[i]) + ret.push_back(boxes[i]); + } + + return ret; + } + +}; + + +#endif //APP_YOLONAS_MODEL_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.cpp new file mode 100644 index 00000000..315a0919 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.cpp @@ -0,0 +1,145 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 9/20/2023. +// + +#include "YOLO_X_Model.h" + + +void YOLO_X_Model::preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims) +{ + LOGI("YOLO_X_Model preprocess"); + cv::Mat img640; + //Resize and get the size from model itself (640x640 for YoloX) + cv::resize(img,img640,cv::Size(dims[2],dims[1]),cv::INTER_LINEAR); + + //float inputScale = 0.00392156862745f; //normalization value, this is 1/255 + float inputScale = 0.998046875000f; //normalization value, this is 1/255 + + float * accumulator = reinterpret_cast (&dest_buffer[0]); + + //opencv read in BGRA by default + cvtColor(img640, img640, CV_BGRA2BGR); + //LOGI("num of channels: %d",img640.channels()); + int lim = img640.rows*img640.cols*3; + for(int idx = 0; idx> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time) { + + //YoloX Model has only single tensor output (BBout_class) that includes classes, boxes as well as score + + LOGI("YOLO_X_Model postprocess"); + std::vector Boxlist; + std::vector Classlist; + + std::vector img_size = {640, 640}; + bool p6=false; + + struct Point { + float32_t x,y; + }; + + std::vector grids; + std::vector expanded_strides; + std::vector strides = !p6 ? std::vector{8, 16, 32} : std::vector{8, 16, 32, 64}; + + std::vector hsizes, wsizes; + for (const int stride : strides) { + hsizes.push_back(img_size[0] / stride); + wsizes.push_back(img_size[1] / stride); + } + + for (size_t i=0; i(x), static_cast(y)}; + grids.push_back(point); + expanded_strides.push_back(stride); + } + } + } + + for (size_t i=0; i<8400; i++) { + + BBout_class[i*85+0] = (BBout_class[i*85+0] + grids[i].x) * expanded_strides[i]; + BBout_class[i*85+1] = (BBout_class[i*85+1] + grids[i].y) * expanded_strides[i]; + BBout_class[i*85+2] = std::exp(BBout_class[i*85+2]) * expanded_strides[i]; + BBout_class[i*85+3] = std::exp(BBout_class[i*85+3]) * expanded_strides[i]; + + } + + for(int i =0;i<(8400);i++) // Total tensor output rows + { + int start = i*85+5; // each row contains classes from indexes 5 to 85 + int end = (i+1)*85; + + float score = BBout_class[start-1]; // each row contains score at index 4 + + if(score>=0.2 ) + { + + auto it = max_element (BBout_class.begin()+start, BBout_class.begin()+end); + int index = distance(BBout_class.begin()+start, it); + + std::string classname = classnamemapping[index]; + + // each row contains box co-ordinates from index 0 to 3 + float x1 = BBout_class[start-5]; + float y1 = BBout_class[start-4]; + float x2 = BBout_class[start-3]; + float y2 = BBout_class[start-2]; + Boxlist.push_back(YOLO_X_BoxCornerEncoding(x1, y1, x2, y2,score,classname)); + } + } + + //LOGI("Boxlist size:: %d",Boxlist.size()); + std::vector reslist_temp = Yolo_X_Rescale_boxes(Boxlist,orig_width,orig_height); + + std::vector reslist = NonMaxSuppression(reslist_temp,0.20); + //LOGI("reslist ssize %d", reslist.size()); + + + numberofobj = reslist.size(); + float ratio_2 = orig_width/640.0f; + float ratio_1 = orig_height/640.0f; + //LOGI("ratio1 %f :: ratio_2 %f",ratio_1,ratio_2); + + for(int k=0;k singleboxcoords{top, bottom, left, right, milli_time}; + BB_coords.push_back(singleboxcoords); + BB_names.push_back(reslist[k].objlabel); + } +} + +void YOLO_X_Model::msg() +{ + LOGI("YOLO_X_Model Class msg model_name = %d", model_name); +} \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.h new file mode 100644 index 00000000..63acc3fc --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/YOLO_X_Model.h @@ -0,0 +1,165 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by gsanjeev on 9/20/2023. +// + +#ifndef APP_YOLO_X_MODEL_H +#define APP_YOLO_X_MODEL_H + +#include "Model.h" + +#include + +class YOLO_X_BoxCornerEncoding { + +public: + float x1; + float y1; + float x2; + float y2; + float score; + std::string objlabel; + + YOLO_X_BoxCornerEncoding(float a, float b, float c, float d,float sc, std::string name="person") + { + x1 = a; + y1 = b; + x2 = c; + y2 = d; + score = sc; + objlabel = name; + } +}; + + +class YOLO_X_Model: public Model{ + + + std::map classnamemapping = + { + {0, "person"},{ 1, "bicycle"},{ 2, "car"},{ 3, "motorcycle"},{ 4, "airplane"},{ 5, "bus"},{ + 6, "train"},{ 7, "truck"},{ 8, "boat"},{ 9, "traffic"},{ 10, "fire"},{ 11, "stop"},{ 12, "parking"},{ + 13, "bench"},{ 14, "bird"},{ 15, "cat"},{ 16, "dog"},{ 17, "horse"},{ 18, "sheep"},{ 19, "cow"},{ + 20, "elephant"},{ 21, "bear"},{ 22, "zebra"},{ 23, "giraffe"},{ 24, "backpack"},{ 25, "umbrella"},{ + 26, "handbag"},{ 27, "tie"},{ 28, "suitcase"},{ 29, "frisbee"},{ 30, "skis"},{ 31, "snowboard"},{ + 32, "sports"},{ 33, "kite"},{ 34, "baseball"},{ 35, "baseball"},{ 36, "skateboard"},{ 37, "surfboard"},{ + 38, "tennis"},{ 39, "bottle"},{ 40, "wine"},{ 41, "cup"},{ 42, "fork"},{ 43, "knife"},{ 44, "spoon"},{ + 45, "bowl"},{ 46, "banana"},{ 47, "apple"},{ 48, "sandwich"},{ 49, "orange"},{ 50, "broccoli"},{ + 51, "carrot"},{ 52, "hot"},{ 53, "pizza"},{ 54, "donut"},{ 55, "cake"},{ 56, "chair"},{ 57, "couch"},{ + 58, "potted"},{ 59, "bed"},{ 60, "dining"},{ 61, "toilet"},{ 62, "tv"},{ 63, "laptop"},{ 64, "mouse"},{ + 65, "remote"},{ 66, "keyboard"},{ 67, "cell"},{ 68, "microwave"},{ 69, "oven"},{ 70, "toaster"},{ + 71, "sink"},{ 72, "refrigerator"},{ 73, "book"},{ 74, "clock"},{ 75, "vase"},{ 76, "scissors"},{ + 77, "teddy"},{ 78, "hair"},{ 79, "toothbrush"} + }; + +public: + + YOLO_X_Model() + { + model_name=YoloX; + } + + void preprocess(std::vector &dest_buffer, cv::Mat &img, std::vector dims); + void postprocess(int orig_width, int orig_height, int &numberofobj, std::vector> &BB_coords, std::vector &BB_names, std::vector &BBout_boxcoords, std::vector &BBout_class, float milli_time); + void msg(); + + inline float ComputeIntersectionOverUnion(const YOLO_X_BoxCornerEncoding &box_i,const YOLO_X_BoxCornerEncoding &box_j) + { + const float box_i_y_min = std::min(box_i.y1, box_i.y2); + const float box_i_y_max = std::max(box_i.y1, box_i.y2); + const float box_i_x_min = std::min(box_i.x1, box_i.x2); + const float box_i_x_max = std::max(box_i.x1, box_i.x2); + const float box_j_y_min = std::min(box_j.y1, box_j.y2); + const float box_j_y_max = std::max(box_j.y1, box_j.y2); + const float box_j_x_min = std::min(box_j.x1, box_j.x2); + const float box_j_x_max = std::max(box_j.x1, box_j.x2); + + const float area_i = + (box_i_y_max - box_i_y_min) * (box_i_x_max - box_i_x_min); + const float area_j = + (box_j_y_max - box_j_y_min) * (box_j_x_max - box_j_x_min); + if (area_i <= 0 || area_j <= 0) return 0.0; + const float intersection_ymax = std::min(box_i_y_max, box_j_y_max); + const float intersection_xmax = std::min(box_i_x_max, box_j_x_max); + const float intersection_ymin = std::max(box_i_y_min, box_j_y_min); + const float intersection_xmin = std::max(box_i_x_min, box_j_x_min); + const float intersection_area = + std::max(intersection_ymax - intersection_ymin, 0.0) * + std::max(intersection_xmax - intersection_xmin, 0.0); + return intersection_area / (area_i + area_j - intersection_area); + } + + std::vector NonMaxSuppression(std::vector boxes, + const float iou_threshold) + { + + if (boxes.size()==0) { + return boxes; + } + + std::sort(boxes.begin(), boxes.end(), [] (const YOLO_X_BoxCornerEncoding& left, const YOLO_X_BoxCornerEncoding& right) { + if (left.score > right.score) { + return true; + } else { + return false; + } + }); + + + std::vector flag(boxes.size(), false); + for (unsigned int i = 0; i < boxes.size(); i++) { + if (flag[i]) { + continue; + } + + for (unsigned int j = i + 1; j < boxes.size(); j++) { + if (ComputeIntersectionOverUnion(boxes[i],boxes[j]) > iou_threshold) { + flag[j] = true; + } + } + } + + std::vector ret; + for (unsigned int i = 0; i < boxes.size(); i++) { + if (!flag[i]) + ret.push_back(boxes[i]); + } + + return ret; + } + + std::vector Yolo_X_Rescale_boxes(std::vector boxes, + int img_width, int img_height){ + + std::vector ret; + for (unsigned int i = 0; i < boxes.size(); i++) { + + float x1=boxes[i].x1 - 0.5f * boxes[i].x2; + float y1=boxes[i].y1 - 0.5f * boxes[i].y2; + float x2=boxes[i].x1 + 0.5f * boxes[i].x2; + float y2=boxes[i].y1 + 0.5f * boxes[i].y2; + + boxes[i].x1 =x1; + boxes[i].y1 =y1; + boxes[i].x2 =x2; + boxes[i].y2 =y2; + + ret.push_back(boxes[i]); + } + + return ret; + + } + +}; + + +#endif //APP_YOLO_X_MODEL_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CheckRuntime.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CheckRuntime.hpp new file mode 100644 index 00000000..07538cd0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CheckRuntime.hpp @@ -0,0 +1,17 @@ +//============================================================================== +// +// Copyright (c) 2017-2019 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef CHECKRUNTIME_H +#define CHECKRUNTIME_H + +#include "SNPE/SNPEFactory.hpp" + +zdl::DlSystem::Runtime_t checkRuntime(zdl::DlSystem::Runtime_t runtime); +bool checkGLCLInteropSupport(); + +#endif diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CreateUserBuffer.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CreateUserBuffer.hpp new file mode 100644 index 00000000..68f25407 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/CreateUserBuffer.hpp @@ -0,0 +1,43 @@ +//============================================================================== +// +// Copyright (c) 2017-2020 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#include"inference.h" +#include +#include +#include +#include "SNPE/SNPE.hpp" +#include "DlSystem/IUserBuffer.hpp" +#include "DlSystem/UserBufferMap.hpp" + +typedef unsigned int GLuint; + +// Helper function to fill a single entry of the UserBufferMap with the given user-backed buffer +void createUserBuffer(zdl::DlSystem::UserBufferMap& userBufferMap, + std::unordered_map>& applicationBuffers, + std::vector>& snpeUserBackedBuffers, + std::unique_ptr& snpe, + const char * name, + const bool isTfNBuffer, + int bitWidth); + + +// Create a UserBufferMap of the SNPE network outputs +void createOutputBufferMap(zdl::DlSystem::UserBufferMap& outputMap, + std::unordered_map>& applicationBuffers, + std::vector>& snpeUserBackedBuffers, + std::unique_ptr& snpe, + const bool isTfNBuffer, + int bitWidth); + +// Create a UserBufferMap of the SNPE network inputs +void createInputBufferMap(zdl::DlSystem::UserBufferMap& inputMap, + std::unordered_map>& applicationBuffers, + std::vector>& snpeUserBackedBuffers, + std::unique_ptr& snpe, + const bool isTfNBuffer, + int bitWidth); diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadContainer.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadContainer.hpp new file mode 100644 index 00000000..85bf622a --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadContainer.hpp @@ -0,0 +1,19 @@ +//============================================================================== +// +// Copyright (c) 2019 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef LOADCONTAINER_H +#define LOADCONTAINER_H + +#include + +#include "DlContainer/IDlContainer.hpp" + +std::unique_ptr loadContainerFromFile(std::string containerPath); +std::unique_ptr loadContainerFromBuffer(const uint8_t * buffer, const size_t size); + +#endif diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadInputTensor.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadInputTensor.hpp new file mode 100644 index 00000000..7aec3b24 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/LoadInputTensor.hpp @@ -0,0 +1,27 @@ +//============================================================================== +// +// Copyright (c) 2017-2019 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef LOADINPUTTENSOR_H +#define LOADINPUTTENSOR_H + +#include +#include +#include + +#include "SNPE/SNPE.hpp" +#include "DlSystem/ITensorFactory.hpp" +#include "DlSystem/TensorMap.hpp" +#include "../../Model.h" + + +bool loadInputUserBuffer(std::unordered_map>& applicationBuffers, + std::unique_ptr& snpe, + cv::Mat &model_input, + zdl::DlSystem::UserBufferMap& inputMap, + int bitWidth, Model *modelobj); +#endif diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/SetBuilderOptions.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/SetBuilderOptions.hpp new file mode 100644 index 00000000..e3db3df0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/SetBuilderOptions.hpp @@ -0,0 +1,26 @@ +//============================================================================== +// +// Copyright (c) 2017-2019 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef SETBUILDEROPTIONS_H +#define SETBUILDEROPTIONS_H + +#include "DlSystem/RuntimeList.hpp" +#include "SNPE/SNPE.hpp" +#include "DlSystem/DlEnums.hpp" +//#include "DlSystem/UDLFunc.hpp" +#include "DlContainer/IDlContainer.hpp" +#include "DlSystem/PlatformConfig.hpp" + +std::unique_ptr setBuilderOptions(std::unique_ptr & container, + zdl::DlSystem::Runtime_t runtime, + zdl::DlSystem::RuntimeList runtimeList, + bool useUserSuppliedBuffers, + bool useCaching, + ModelName modelName); + +#endif //SETBUILDEROPTIONS_H \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/Util.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/Util.hpp new file mode 100644 index 00000000..346e7ac0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/Util.hpp @@ -0,0 +1,41 @@ +//============================================================================== +// +// Copyright (c) 2017-2019 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef UTIL_H +#define UTIL_H + +#include +#include +#include +#include + +#include "DlSystem/ITensorFactory.hpp" +#include "DlSystem/TensorShape.hpp" + +template Container& split(Container& result, const typename Container::value_type & s, typename Container::value_type::value_type delimiter ) +{ + result.clear(); + std::istringstream ss( s ); + while (!ss.eof()) + { + typename Container::value_type field; + getline( ss, field, delimiter ); + if (field.empty()) continue; + result.push_back( field ); + } + return result; +} + + +cv::Mat get_affine_transform(int dst_w, int dst_h, int inv, double center[], double scale[]); +//void getcenterscale(int image_width, int image_height, double center[2], double scale[2]); +void getcenterscale(int image_width, int image_height, double center[2], double scale[2],float bottom, float left, float top, float right); +float** getCoords(std::vector buff, double center[], double scale[]); + +#endif + diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/inference.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/inference.h new file mode 100644 index 00000000..a74eaea3 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/hpp/inference.h @@ -0,0 +1,74 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +// +// Created by shubpate on 12/11/2021. +// + +#ifndef NATIVEINFERENCE_INFERENCE_H +#define NATIVEINFERENCE_INFERENCE_H + +#include "zdl/DlSystem/TensorShape.hpp" +#include "zdl/DlSystem/TensorMap.hpp" +#include "zdl/DlSystem/TensorShapeMap.hpp" +#include "zdl/DlSystem/IUserBufferFactory.hpp" +#include "zdl/DlSystem/IUserBuffer.hpp" +#include "zdl/DlSystem/UserBufferMap.hpp" +#include "zdl/DlSystem/IBufferAttributes.hpp" + +#include "zdl/DlSystem/StringList.hpp" + +#include "zdl/SNPE/SNPE.hpp" +#include "zdl/SNPE/SNPEFactory.hpp" +#include "zdl/DlSystem/DlVersion.hpp" +#include "zdl/DlSystem/DlEnums.hpp" +#include "zdl/DlSystem/String.hpp" +#include "zdl/DlContainer/IDlContainer.hpp" +#include "zdl/SNPE/SNPEBuilder.hpp" + +#include "zdl/DlSystem/ITensor.hpp" +#include "zdl/DlSystem/ITensorFactory.hpp" + +#include +#include "android/log.h" +#include + +#include "../../Model.h" + +#define LOG_TAG "SNPE_INF" +#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) +#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) + +/*class BoxCornerEncoding { + +public: + int x1; + int y1; + int x2; + int y2; + float score; + std::string objlabel; + + BoxCornerEncoding(int a, int b, int c, int d,int sc, std::string name="person") + { + x1 = a; + y1 = b; + x2 = c; + y2 = d; + score = sc; + objlabel = name; + } +};*/ + +std::string build_network_BB(const uint8_t * dlc_buffer, const size_t dlc_size, const char runtime_arg, ModelName modelName); +bool SetAdspLibraryPath(std::string nativeLibPath); + +bool executeDLC(cv::Mat &img, int orig_width, int orig_height, int &numberofhuman, std::vector> &BB_coords, std::vector &BB_names, Model *modelobj); + +#endif //NATIVEINFERENCE_INFERENCE_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.h new file mode 100644 index 00000000..9a084071 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.h @@ -0,0 +1,102 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef _DIAGLOG_IDIAGLOG_H_ +#define _DIAGLOG_IDIAGLOG_H_ + +#include "DiagLog/Options.h" +#include "DlSystem/SnpeApiExportDefine.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE IDiagLog handle + */ +typedef void* Snpe_IDiagLog_Handle_t; + +/** + * @brief . + * + * Sets the options after initialization occurs. + * + * @param[in] handle : Handle to access IDiagLog + * @param[in] loggingOptions : The options to set up diagnostic logging. + * + * @return Error code if the options could not be set. Ensure logging is not started/ + * SNPE_SUCCESS otherwise + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IDiagLog_SetOptions(Snpe_IDiagLog_Handle_t handle, Snpe_Options_Handle_t loggingOptionsHandle); + +/** + * @brief . + * + * Gets the curent options for the diag logger. + * + * @param[in] handle : Handle to access IDiagLog + * @return Handle to access DiagLog options. + */ +SNPE_API +Snpe_Options_Handle_t Snpe_IDiagLog_GetOptions(Snpe_IDiagLog_Handle_t handle); + +/** + * @brief . + * + * @param[in] handle : Handle to access IDiagLog + * @param[in] mask : Allows for setting the log mask once diag logging has started + * @return SNPE_SUCCESS if the level was set successfully. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IDiagLog_SetDiagLogMask(Snpe_IDiagLog_Handle_t handle, const char* mask) ; + +/** + * @brief . + * + * Enables logging. + * + * Logging should be started prior to the instantiation of other SNPE_APIs + * to ensure all events are captured. + * + * @param[in] handle : Handle to access IDiagLog + * @return SNPE_SUCCESS if diagnostic logging started successfully. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IDiagLog_Start(Snpe_IDiagLog_Handle_t handle); + +/** + * @brief Disables logging. + * + * @param[in] handle : Handle to access IDiagLog + * + * @return SNPE_SUCCESS if logging stopped successfully. Error code otherwise. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IDiagLog_Stop(Snpe_IDiagLog_Handle_t handle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _DIAGLOG_IDIAGLOG_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.hpp new file mode 100644 index 00000000..64b81eba --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/IDiagLog.hpp @@ -0,0 +1,133 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include +#include + +#include "Options.hpp" +#include "DlSystem/String.hpp" + +#include "DiagLog/IDiagLog.h" + + +namespace DiagLog{ +class IDiagLog : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static Snpe_ErrorCode_t InvalidDeleteCall(Snpe_IDiagLog_Handle_t ){ + return SNPE_ERRORCODE_CAPI_DELETE_FAILURE; + } + + static constexpr DeleteFunctionType DeleteFunction{InvalidDeleteCall}; + + class OptionsInternal : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_Options_Delete}; + public: + OptionsInternal() + : BaseType(Snpe_Options_Create()) + { } + + explicit OptionsInternal(const Options& options) + : BaseType(Snpe_Options_Create()) + { + setDiagLogMask(options.DiagLogMask.c_str()); + setLogFileDirectory(options.LogFileDirectory.c_str()); + setLogFileName(options.LogFileName.c_str()); + setLogFileRotateCount(options.LogFileRotateCount); + setLogFileReplace(options.LogFileReplace); + } + + const char* getDiagLogMask() const{ + return Snpe_Options_GetDiagLogMask(handle()); + } + void setDiagLogMask(const char* diagLogMask){ + Snpe_Options_SetDiagLogMask(handle(), diagLogMask); + } + + const char* getLogFileDirectory() const{ + return Snpe_Options_GetLogFileDirectory(handle()); + } + void setLogFileDirectory(const char* logFileDirectory){ + Snpe_Options_SetLogFileDirectory(handle(), logFileDirectory); + } + + const char* getLogFileName() const{ + return Snpe_Options_GetLogFileName(handle()); + } + void setLogFileName(const char* logFileName){ + Snpe_Options_SetLogFileName(handle(), logFileName); + } + + uint32_t getLogFileRotateCount() const{ + return Snpe_Options_GetLogFileRotateCount(handle()); + } + void setLogFileRotateCount(uint32_t logFileRotateCount){ + Snpe_Options_SetLogFileRotateCount(handle(), logFileRotateCount); + } + + bool getLogFileReplace() const{ + return Snpe_Options_GetLogFileReplace(handle()); + } + void setLogFileReplace(bool logFileReplace){ + Snpe_Options_SetLogFileReplace(handle(), logFileReplace); + } + + explicit operator Options() const{ + return { + getDiagLogMask(), + getLogFileDirectory(), + getLogFileName(), + getLogFileRotateCount(), + getLogFileReplace() + }; + } + + }; + + + +public: + bool setOptions(const Options& loggingOptions){ + OptionsInternal optionsInternal(loggingOptions); + return SNPE_SUCCESS == Snpe_IDiagLog_SetOptions(handle(), getHandle(optionsInternal)); + } + Options getOptions() const{ + OptionsInternal optionsInternal(moveHandle(Snpe_IDiagLog_GetOptions(handle()))); + return Options(optionsInternal); + } + + bool setDiagLogMask(const std::string& mask){ + return SNPE_SUCCESS == Snpe_IDiagLog_SetDiagLogMask(handle(), mask.c_str()); + } + bool setDiagLogMask(const DlSystem::String& mask){ + return setDiagLogMask(static_cast(mask)); + } + + bool start(void){ + return SNPE_SUCCESS == Snpe_IDiagLog_Start(handle()); + } + bool stop(void){ + return SNPE_SUCCESS == Snpe_IDiagLog_Stop(handle()); + } + +}; + +} // ns DiagLog + +ALIAS_IN_ZDL_NAMESPACE(DiagLog, IDiagLog) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.h new file mode 100644 index 00000000..ad641cca --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.h @@ -0,0 +1,164 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef _DIAGLOG_OPTIONS_H_ +#define _DIAGLOG_OPTIONS_H_ + +#include + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE Options handle + */ +typedef void* Snpe_Options_Handle_t; + + +SNPE_API +Snpe_Options_Handle_t Snpe_Options_Create(); + +/** + * Destroys/frees a Options + * + * @param[in] handle : Handle to access Options object + * @return indication of success/failures + */ +SNPE_API +Snpe_ErrorCode_t Snpe_Options_Delete(Snpe_Options_Handle_t handle); + +/** + * Gets DiagLogMask + * diagLogMask: Enables diag logging only on the specified area mask + * + * @param[in] handle : Handle to access Options object + * @return diagLogMask as a const char* + */ +SNPE_API +const char* Snpe_Options_GetDiagLogMask(Snpe_Options_Handle_t handle); + +/** + * Sets DiagLogMask + * diagLogMask: Enables diag logging only on the specified area mask + * + * @param[in] handle : Handle to access Options object + * @param[in] diagLogMask : specific area where logging needs to be enabed + */ +SNPE_API +void Snpe_Options_SetDiagLogMask(Snpe_Options_Handle_t handle, const char* diagLogMask); + +/** + * Gets logFileDirectory + * logFileDirectory: The path to the directory where log files will be written. + * The path may be relative or absolute. Relative paths are interpreted + * + * @param[in] handle : Handle to access Options object + * @return logFileDirectory as a const char* + */ +SNPE_API +const char* Snpe_Options_GetLogFileDirectory(Snpe_Options_Handle_t handle); + +/** + * Sets logFileDirectory + * logFileDirectory: The path to the directory where log files will be written. + * The path may be relative or absolute. Relative paths are interpreted + * + * @param[in] handle : Handle to access Options object + * @param[in] logFileDirectory : path for saving the log files + */ +SNPE_API +void Snpe_Options_SetLogFileDirectory(Snpe_Options_Handle_t handle, const char* logFileDirectory); + + +/** + * Gets logFileName + * logFileName: The name used for log files. If this value is empty then BaseName will be + * used as the default file name. + * + * @param[in] handle : Handle to access Options object + * @return logFileName as a const char* + */ +SNPE_API +const char* Snpe_Options_GetLogFileName(Snpe_Options_Handle_t handle); + +/** + * Sets logFileName + * logFileName: The name used for log files. If this value is empty then BaseName will be + * used as the default file name. + * + * @param[in] handle : Handle to access Options object + * @param[in] logFileName : name of log file + */ +SNPE_API +void Snpe_Options_SetLogFileName(Snpe_Options_Handle_t handle, const char* logFileName); + +/** + * Gets the maximum number of log files to create. If set to 0 no log rotation + * will be used and the log file name specified will be used each time, overwriting + * any existing log file that may exist. + * + * @param[in] handle : Handle to access options object. + * @return max log files to create + */ +SNPE_API +uint32_t Snpe_Options_GetLogFileRotateCount(Snpe_Options_Handle_t handle); + +/** + * Sets the maximum number of log files to create. If set to 0 no log rotation + * will be used and the log file name specified will be used each time, overwriting + * any existing log file that may exist. + * + * @param[in] handle : Handle to access options object. + * @param[in] logFileRotateCount : max log files to create + */ +SNPE_API +void Snpe_Options_SetLogFileRotateCount(Snpe_Options_Handle_t handle, uint32_t logFileRotateCount); + +/** + * If the log file already exists, control whether it will be replaced + * + * @param[in] handle : Handle to access options object + * @return 1 if log file will be replaced, 0 otherwise + */ +SNPE_API +int Snpe_Options_GetLogFileReplace(Snpe_Options_Handle_t handle); + +/** + * If the log file already exists, control whether it will be replaced + * + * @param[in] handle : Handle to access options object + * @param[in] logFileReplace : 1 if log file to be replaced, 0 otherwise + */ +SNPE_API +void Snpe_Options_SetLogFileReplace(Snpe_Options_Handle_t handle, int logFileReplace); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _DIAGLOG_OPTIONS_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.hpp new file mode 100644 index 00000000..c9ad48b6 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DiagLog/Options.hpp @@ -0,0 +1,50 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include +#include + +#include "DiagLog/IDiagLog.h" + + +namespace DiagLog { + +class Options +{ +public: + Options( + std::string diagLogMask = "", + std::string logFileDirectory = "diaglogs", + std::string logFileName = "DiagLog", + uint32_t logFileRotateCount = 20, + bool logFileReplace = true + ) + : DiagLogMask(std::move(diagLogMask)), + LogFileDirectory(std::move(logFileDirectory)), + LogFileName(std::move(logFileName)), + LogFileRotateCount(logFileRotateCount), + LogFileReplace(logFileReplace) + { + // Solves the empty string problem with multiple std libs + DiagLogMask.reserve(1); + } + + std::string DiagLogMask; + std::string LogFileDirectory; + std::string LogFileName; + uint32_t LogFileRotateCount; + + bool LogFileReplace; +}; + +} // ns DiagLog + +ALIAS_IN_ZDL_NAMESPACE(DiagLog, Options) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/DlContainer.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/DlContainer.h new file mode 100644 index 00000000..6ce7cd25 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/DlContainer.h @@ -0,0 +1,185 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_CONTAINER_DLCONTAINER_H +#define DL_CONTAINER_DLCONTAINER_H + +#ifdef __cplusplus +#include // uint8_t +#include // size_t +#else +#include +#include +#endif + +#include "DlSystem/DlError.h" +#include "DlSystem/StringList.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE DlcRecord handle + */ +typedef void* Snpe_DlcRecord_Handle_t; + +/** + * Constructs a DlcRecord and returns a handle to it + * + * @return the handle to the created DlcRecord + */ +SNPE_API +Snpe_DlcRecord_Handle_t Snpe_DlcRecord_Create(); + +/** + * Constructs a DlcRecord with a provided name and returns a handle to it + * + * @param[in] name : the name of the record + * + * @return the handle to the created DlcRecord + */ +SNPE_API +Snpe_DlcRecord_Handle_t Snpe_DlcRecord_CreateName(const char* name); + + +/** + * Destroys/frees a DlcRecord + * + * @param[in] dlcRecordHandle : Handle to access DlcRecord + * + * @return indication of success/failures + */ +SNPE_API +Snpe_ErrorCode_t Snpe_DlcRecord_Delete(Snpe_DlcRecord_Handle_t dlcRecordHandle); + +/** + * Gets the size of a DlcRecord in bytes + * + * @param[in] dlcRecordHandle : Handle to access DlcRecord + * + * @return the size of the DlcRecord in bytes + */ +SNPE_API +size_t Snpe_DlcRecord_Size(Snpe_DlcRecord_Handle_t dlcRecordHandle); + +/** + * Gets a pointer to the start of the DlcRecord's data + * + * @param[in] dlcRecordHandle : Handle to access DlcRecord + * + * @return uint8_t pointer to the DlcRecord's data + */ +SNPE_API +uint8_t* Snpe_DlcRecord_Data(Snpe_DlcRecord_Handle_t dlcRecordHandle); + +/** + * Gets the name of the DlcRecord + * + * @param[in] dlcRecordHandle : Handle to access DlcRecord + * + * @return the record's name + */ +SNPE_API +const char* Snpe_DlcRecord_Name(Snpe_DlcRecord_Handle_t dlcRecordHandle); + +/** + * A typedef to indicate a SNPE DlContainer handle + */ +typedef void* Snpe_DlContainer_Handle_t; + +/** + * Destroys/frees a DlContainer + * + * @param[in] dlContainerHandle : Handle to access DlContainer + * + * @return indication of success/failures + */ +SNPE_API +Snpe_ErrorCode_t Snpe_DlContainer_Delete(Snpe_DlContainer_Handle_t dlContainerHandle); + + +/** + * Initializes a container from a container archive file. + * + * @param[in] filename Container archive file path. + * + * @return Status of container open call + */ +SNPE_API +Snpe_DlContainer_Handle_t Snpe_DlContainer_Open(const char* filename); + +/** + * Initializes a container from a byte buffer. + * + * @param[in] buffer Byte buffer holding the contents of an archive + * file. + * + * @param[in] size Size of the byte buffer. + * + * @return A Snpe_DlContainer_Handle_t to access the dlContainer + */ +SNPE_API +Snpe_DlContainer_Handle_t Snpe_DlContainer_OpenBuffer(const uint8_t* buffer, const size_t size); + +/** + * Get the record catalog for a container. + * + * @param[in] dlContainerHandle : Handle to access DlContainer + * + * @return A Snpe_StringListHandle_t that holds the record names of the DlContainer + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_DlContainer_GetCatalog(Snpe_DlContainer_Handle_t dlContainerHandle); + +/** + * Get a record from a container by name. + * + * @param[in] dlContainerHandle : Handle to access DlContainer + * @param[in] recordName : Name of the record to fetch. + * + * @return A Snpe_DlcRecordHandle_t that owns the record read from the DlContainer + */ +SNPE_API +Snpe_DlcRecord_Handle_t Snpe_DlContainer_GetRecord(Snpe_DlContainer_Handle_t dlContainerHandle, const char* recordName); + +/** + * Save the container to an archive on disk. This function will save the + * container if the filename is different from the file that it was opened + * from, or if at least one record was modified since the container was + * opened. + * + * It will truncate any existing file at the target path. + * + * @param[in] dlContainerHandle : Handle to access DlContainer + * @param[in] filename : Container archive file path. + * + * @return indication of success/failure + */ +SNPE_API +Snpe_ErrorCode_t Snpe_DlContainer_Save(Snpe_DlContainer_Handle_t dlContainerHandle, const char* filename); + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_CONTAINER_DLCONTAINER_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/IDlContainer.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/IDlContainer.hpp new file mode 100644 index 00000000..482dbd02 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlContainer/IDlContainer.hpp @@ -0,0 +1,146 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include +#include +#include +#include +#include + +#include "Wrapper.hpp" +#include "DlSystem/String.hpp" + +#include "DlContainer/DlContainer.h" +#include "DlSystem/StringList.hpp" + + + +namespace DlContainer { + +struct DlcRecord +{ + std::string name; + std::vector data; + + DlcRecord() + : name{}, + data{} + { } + + DlcRecord( DlcRecord&& other ) noexcept + : name(std::move(other.name)), + data(std::move(other.data)) + { } + DlcRecord(const std::string& new_name) + : name(new_name), + data() + { + if(name.empty()) { + name.reserve(1); + } + } + DlcRecord(const DlcRecord&) = delete; +}; + + +class IDlContainer : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_DlContainer_Delete}; + + template + void getCatalog_(std::set& catalog) const{ + DlSystem::StringList sl(moveHandle(Snpe_DlContainer_GetCatalog(handle()))); + for(auto s : sl){ + catalog.emplace(s); + } + } + + + class DlcRecordInternal : public Wrapper { + friend BaseType; + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_DlcRecord_Delete}; + public: + DlcRecordInternal() + : BaseType(Snpe_DlcRecord_Create()) + { } + explicit DlcRecordInternal(const std::string& name) + : BaseType(Snpe_DlcRecord_CreateName(name.c_str())) + { } + + uint8_t* getData(){ + return Snpe_DlcRecord_Data(handle()); + } + size_t size() const{ + return Snpe_DlcRecord_Size(handle()); + } + const char* getName(){ + return Snpe_DlcRecord_Name(handle()); + } + }; + + +public: + static std::unique_ptr open(const std::string& filename) noexcept{ + return makeUnique(Snpe_DlContainer_Open(filename.c_str())); + } + + static std::unique_ptr open(const uint8_t* buffer, const size_t size) noexcept{ + return makeUnique(Snpe_DlContainer_OpenBuffer(buffer, size)); + + } + static std::unique_ptr open(const std::vector& buffer) noexcept{ + return open(buffer.data(), buffer.size()); + } + static std::unique_ptr open(const DlSystem::String &filename) noexcept{ + return open(static_cast(filename)); + } + + + void getCatalog(std::set& catalog) const{ + return getCatalog_(catalog); + } + void getCatalog(std::set& catalog) const{ + return getCatalog_(catalog); + } + + bool getRecord(const std::string& name, DlcRecord& record) const{ + auto h = Snpe_DlContainer_GetRecord(handle(), name.c_str()); + if(!h) return false; + DlcRecordInternal internal(moveHandle(h)); + auto data = internal.getData(); + + record.name.assign(internal.getName()); + record.data.assign(data, data+internal.size()); + return true; + } + + bool getRecord(const DlSystem::String& name, DlcRecord& record) const{ + return getRecord(static_cast(name), record); + } + + bool save(const std::string& filename){ + return Snpe_DlContainer_Save(handle(), filename.c_str()); + } + + bool save(const DlSystem::String& filename){ + return save(static_cast(filename)); + } +}; + + +} // ns DlContainer + +ALIAS_IN_ZDL_NAMESPACE(DlContainer, DlcRecord) +ALIAS_IN_ZDL_NAMESPACE(DlContainer, IDlContainer) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.h new file mode 100644 index 00000000..85a0f4d3 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.h @@ -0,0 +1,267 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _DL_ENUMS_H_ +#define _DL_ENUMS_H_ + +#include "DlSystem/SnpeApiExportDefine.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * Enumeration of supported target runtimes. + */ +typedef enum +{ + /// Special value indicating the property is unset. + SNPE_RUNTIME_UNSET = -1, + /// Run the processing on Snapdragon CPU. + /// Data: float 32bit + /// Math: float 32bit + SNPE_RUNTIME_CPU_FLOAT32 = 0, + /// Default legacy enum to retain backward compatibility. + /// CPU = CPU_FLOAT32 + SNPE_RUNTIME_CPU = SNPE_RUNTIME_CPU_FLOAT32, + + /// Run the processing on the Adreno GPU. + /// Data: float 16bit + /// Math: float 32bit + SNPE_RUNTIME_GPU_FLOAT32_16_HYBRID = 1, + /// Default legacy enum to retain backward compatibility. + /// GPU = GPU_FLOAT32_16_HYBRID + SNPE_RUNTIME_GPU = SNPE_RUNTIME_GPU_FLOAT32_16_HYBRID, + + /// Run the processing on the Hexagon DSP. + /// Data: 8bit fixed point Tensorflow style format + /// Math: 8bit fixed point Tensorflow style format + SNPE_RUNTIME_DSP_FIXED8_TF = 2, + /// Default legacy enum to retain backward compatibility. + /// DSP = DSP_FIXED8_TF + SNPE_RUNTIME_DSP = SNPE_RUNTIME_DSP_FIXED8_TF, + + /// Run the processing on the Adreno GPU. + /// Data: float 16bit + /// Math: float 16bit + SNPE_RUNTIME_GPU_FLOAT16 = 3, + + /// Run the processing on Snapdragon AIX+HVX. + /// Data: 8bit fixed point Tensorflow style format + /// Math: 8bit fixed point Tensorflow style format + SNPE_RUNTIME_AIP_FIXED8_TF = 5, + SNPE_RUNTIME_AIP_FIXED_TF = SNPE_RUNTIME_AIP_FIXED8_TF +} Snpe_Runtime_t; + +/** + * Enumeration of runtime available check options. + */ +typedef enum +{ + /// Perform standard runtime available check + SNPE_RUNTIME_CHECK_OPTION_DEFAULT = 2, + /// Perform standard runtime available check + SNPE_RUNTIME_CHECK_OPTION_NORMAL_CHECK = 0, + /// Perform basic runtime available check, may be runtime specific + SNPE_RUNTIME_CHECK_OPTION_BASIC_CHECK = 1, + /// Perform unsignedPD runtime available check + SNPE_RUNTIME_CHECK_OPTION_UNSIGNEDPD_CHECK = 2, +} Snpe_RuntimeCheckOption_t; + +/** + * Enumeration of various performance profiles that can be requested. + */ +typedef enum +{ + /// Run in a standard mode. + /// This mode will be deprecated in the future and replaced with BALANCED. + SNPE_PERFORMANCE_PROFILE_DEFAULT = 0, + /// Run in a balanced mode. + SNPE_PERFORMANCE_PROFILE_BALANCED = 0, + + /// Run in high performance mode + SNPE_PERFORMANCE_PROFILE_HIGH_PERFORMANCE = 1, + + /// Run in a power sensitive mode, at the expense of performance. + SNPE_PERFORMANCE_PROFILE_POWER_SAVER = 2, + + /// Use system settings. SNPE makes no calls to any performance related APIs. + SNPE_PERFORMANCE_PROFILE_SYSTEM_SETTINGS = 3, + + /// Run in sustained high performance mode + SNPE_PERFORMANCE_PROFILE_SUSTAINED_HIGH_PERFORMANCE = 4, + + /// Run in burst mode + SNPE_PERFORMANCE_PROFILE_BURST = 5, + + /// Run in lower clock than POWER_SAVER, at the expense of performance. + SNPE_PERFORMANCE_PROFILE_LOW_POWER_SAVER = 6, + + /// Run in higher clock and provides better performance than POWER_SAVER. + SNPE_PERFORMANCE_PROFILE_HIGH_POWER_SAVER = 7, + + /// Run in lower balanced mode + SNPE_PERFORMANCE_PROFILE_LOW_BALANCED = 8, + + /// Run in lowest clock at the expense of performance + SNPE_PERFORMANCE_PROFILE_EXTREME_POWER_SAVER = 9, + +} Snpe_PerformanceProfile_t; + +/** + * Enumeration of various profilngLevels that can be requested. + */ +typedef enum +{ + /// No profiling. + /// Collects no runtime stats in the DiagLog + SNPE_PROFILING_LEVEL_OFF = 0, + + /// Basic profiling + /// Collects some runtime stats in the DiagLog + SNPE_PROFILING_LEVEL_BASIC = 1, + + /// Detailed profiling + /// Collects more runtime stats in the DiagLog, including per-layer statistics + /// Performance may be impacted + SNPE_PROFILING_LEVEL_DETAILED = 2, + + /// Moderate profiling + /// Collects more runtime stats in the DiagLog, no per-layer statistics + SNPE_PROFILING_LEVEL_MODERATE = 3, + + /// Linting profiling + /// HTP exclusive profiling level that collects in-depth performance metrics + /// for each op in the graph including main thread execution time and time spent + /// on parallel background ops + SNPE_PROFILING_LEVEL_LINTING = 4 + +} Snpe_ProfilingLevel_t; + +/** + * Enumeration of various execution priority hints. + */ +typedef enum +{ + /// Normal priority + SNPE_EXECUTION_PRIORITY_NORMAL = 0, + + /// Higher than normal priority + SNPE_EXECUTION_PRIORITY_HIGH = 1, + + /// Lower priority + SNPE_EXECUTION_PRIORITY_LOW = 2, + + /// Between Normal and High priority + SNPE_EXECUTION_PRIORITY_NORMAL_HIGH = 3 + +} Snpe_ExecutionPriorityHint_t; + +/** + * Enumeration that lists the supported image encoding formats. + */ +typedef enum +{ + /// For unknown image type. Also used as a default value for ImageEncoding_t. + SNPE_IMAGE_ENCODING_UNKNOWN = 0, + + /// The RGB format consists of 3 bytes per pixel: one byte for + /// Red, one for Green, and one for Blue. The byte ordering is + /// endian independent and is always in RGB byte order. + SNPE_IMAGE_ENCODING_RGB = 1, + + /// The ARGB32 format consists of 4 bytes per pixel: one byte for + /// Red, one for Green, one for Blue, and one for the alpha channel. + /// The alpha channel is ignored. The byte ordering depends on the + /// underlying CPU. For little endian CPUs, the byte order is BGRA. + /// For big endian CPUs, the byte order is ARGB. + SNPE_IMAGE_ENCODING_ARGB32 = 2, + + /// The RGBA format consists of 4 bytes per pixel: one byte for + /// Red, one for Green, one for Blue, and one for the alpha channel. + /// The alpha channel is ignored. The byte ordering is endian independent + /// and is always in RGBA byte order. + SNPE_IMAGE_ENCODING_RGBA = 3, + + /// The GRAYSCALE format is for 8-bit grayscale. + SNPE_IMAGE_ENCODING_GRAYSCALE = 4, + + /// NV21 is the Android version of YUV. The Chrominance is down + /// sampled and has a subsampling ratio of 4:2:0. Note that this + /// image format has 3 channels, but the U and V channels + /// are subsampled. For every four Y pixels there is one U and one V pixel. @newpage + SNPE_IMAGE_ENCODING_NV21 = 5, + + /// The BGR format consists of 3 bytes per pixel: one byte for + /// Red, one for Green and one for Blue. The byte ordering is + /// endian independent and is always BGR byte order. + SNPE_IMAGE_ENCODING_BGR = 6 +} Snpe_ImageEncoding_t; + +/** + * Enumeration that lists the supported LogLevels that can be set by users. + */ +typedef enum +{ + /// Enumeration variable to be used by user to set logging level to FATAL. + SNPE_LOG_LEVEL_FATAL = 0, + + /// Enumeration variable to be used by user to set logging level to ERROR. + SNPE_LOG_LEVEL_ERROR = 1, + + /// Enumeration variable to be used by user to set logging level to WARN. + SNPE_LOG_LEVEL_WARN = 2, + + /// Enumeration variable to be used by user to set logging level to INFO. + SNPE_LOG_LEVEL_INFO = 3, + + /// Enumeration variable to be used by user to set logging level to VERBOSE. + SNPE_LOG_LEVEL_VERBOSE = 4 +} Snpe_LogLevel_t; + +/** + * Enumeration that list the supported data types for buffers + */ +typedef enum +{ + /// Unspecified + SNPE_IO_BUFFER_DATATYPE_UNSPECIFIED = 0, + + /// 32-bit floating point + SNPE_IO_BUFFER_DATATYPE_FLOATING_POINT_32 = 1, + + /// 16-bit floating point + SNPE_IO_BUFFER_DATATYPE_FLOATING_POINT_16 = 2, + + /// 8-bit fixed point + SNPE_IO_BUFFER_DATATYPE_FIXED_POINT_8 = 3, + + /// 16-bit fixed point + SNPE_IO_BUFFER_DATATYPE_FIXED_POINT_16 = 4 +} Snpe_IOBufferDataType_t; + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _DL_ENUMS_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.hpp new file mode 100644 index 00000000..9158f594 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlEnums.hpp @@ -0,0 +1,266 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +namespace DlSystem { +/** @addtogroup c_plus_plus_apis C++ +@{ */ + +/** + * Enumeration of supported target runtimes. + */ +enum class Runtime_t +{ + /// Special value indicating the property is unset. + UNSET = -1, + /// Run the processing on Snapdragon CPU. + /// Data: float 32bit + /// Math: float 32bit + CPU_FLOAT32 = 0, + /// Default legacy enum to retain backward compatibility. + /// CPU = CPU_FLOAT32 + CPU = CPU_FLOAT32, + + /// Run the processing on the Adreno GPU. + /// Data: float 16bit + /// Math: float 32bit + GPU_FLOAT32_16_HYBRID = 1, + /// Default legacy enum to retain backward compatibility. + /// GPU = GPU_FLOAT32_16_HYBRID + GPU = GPU_FLOAT32_16_HYBRID, + + /// Run the processing on the Hexagon DSP. + /// Data: 8bit fixed point Tensorflow style format + /// Math: 8bit fixed point Tensorflow style format + DSP_FIXED8_TF = 2, + /// Default legacy enum to retain backward compatibility. + /// DSP = DSP_FIXED8_TF + DSP = DSP_FIXED8_TF, + + /// Run the processing on the Adreno GPU. + /// Data: float 16bit + /// Math: float 16bit + GPU_FLOAT16 = 3, + + /// Run the processing on Snapdragon AIX+HVX. + /// Data: 8bit fixed point Tensorflow style format + /// Math: 8bit fixed point Tensorflow style format + AIP_FIXED8_TF = 5, + AIP_FIXED_TF = AIP_FIXED8_TF, + + /// Any new enums should be added above this line + NUM_RUNTIME_TARGETS +}; + +/** + * Enumeration of runtime available check options. + */ +enum class RuntimeCheckOption_t +{ + /// Perform standard runtime available check + NORMAL_CHECK = 0, + /// Perform basic runtime available check, may be runtime specific + BASIC_CHECK = 1, + /// Perform unsignedPD runtime available check + UNSIGNEDPD_CHECK = 2, + /// Perform standard runtime available check + DEFAULT = 2, + /// Any new enums should be added above this line + NUM_RUNTIMECHECK_OPTIONS +}; + +/** + * Enumeration of various performance profiles that can be requested. + */ +enum class PerformanceProfile_t +{ + /// Run in a standard mode. + /// This mode will be deprecated in the future and replaced with BALANCED. + DEFAULT = 0, + /// Run in a balanced mode. + BALANCED = 0, + + /// Run in high performance mode + HIGH_PERFORMANCE = 1, + + /// Run in a power sensitive mode, at the expense of performance. + POWER_SAVER = 2, + + /// Use system settings. SNPE makes no calls to any performance related APIs. + SYSTEM_SETTINGS = 3, + + /// Run in sustained high performance mode + SUSTAINED_HIGH_PERFORMANCE = 4, + + /// Run in burst mode + BURST = 5, + + /// Run in lower clock than POWER_SAVER, at the expense of performance. + LOW_POWER_SAVER = 6, + + /// Run in higher clock and provides better performance than POWER_SAVER. + HIGH_POWER_SAVER = 7, + + /// Run in lower balanced mode + LOW_BALANCED = 8, + + /// Run in lowest clock at the expense of performance + EXTREME_POWER_SAVER = 9, + + /// Any new enums should be added above this line + NUM_PERF_PROFILES +}; + +/** + * Enumeration of various profilngLevels that can be requested. + */ +enum class ProfilingLevel_t +{ + /// No profiling. + /// Collects no runtime stats in the DiagLog + OFF = 0, + + /// Basic profiling + /// Collects some runtime stats in the DiagLog + BASIC = 1, + + /// Detailed profiling + /// Collects more runtime stats in the DiagLog, including per-layer statistics + /// Performance may be impacted + DETAILED = 2, + + /// Moderate profiling + /// Collects more runtime stats in the DiagLog, no per-layer statistics + MODERATE = 3, + + /// Linting profiling + /// HTP exclusive profiling level that collects in-depth performance metrics + /// for each op in the graph including main thread execution time and time spent + /// on parallel background ops + LINTING = 4 +}; + +/** + * Enumeration of various execution priority hints. + */ +enum class ExecutionPriorityHint_t +{ + /// Normal priority + NORMAL = 0, + + /// Higher than normal priority + HIGH = 1, + + /// Lower priority + LOW = 2, + + /// Between Normal and High priority + NORMAL_HIGH = 3, + + /// Any new enums should be added above this line + NUM_EXECUTION_PRIORITY_HINTS +}; + +/** @} */ /* end_addtogroup c_plus_plus_apis C++*/ + +/** + * Enumeration that lists the supported image encoding formats. + */ +enum class ImageEncoding_t +{ + /// For unknown image type. Also used as a default value for ImageEncoding_t. + UNKNOWN = 0, + + /// The RGB format consists of 3 bytes per pixel: one byte for + /// Red, one for Green, and one for Blue. The byte ordering is + /// endian independent and is always in RGB byte order. + RGB = 1, + + /// The ARGB32 format consists of 4 bytes per pixel: one byte for + /// Red, one for Green, one for Blue, and one for the alpha channel. + /// The alpha channel is ignored. The byte ordering depends on the + /// underlying CPU. For little endian CPUs, the byte order is BGRA. + /// For big endian CPUs, the byte order is ARGB. + ARGB32 = 2, + + /// The RGBA format consists of 4 bytes per pixel: one byte for + /// Red, one for Green, one for Blue, and one for the alpha channel. + /// The alpha channel is ignored. The byte ordering is endian independent + /// and is always in RGBA byte order. + RGBA = 3, + + /// The GRAYSCALE format is for 8-bit grayscale. + GRAYSCALE = 4, + + /// NV21 is the Android version of YUV. The Chrominance is down + /// sampled and has a subsampling ratio of 4:2:0. Note that this + /// image format has 3 channels, but the U and V channels + /// are subsampled. For every four Y pixels there is one U and one V pixel. @newpage + NV21 = 5, + + /// The BGR format consists of 3 bytes per pixel: one byte for + /// Red, one for Green and one for Blue. The byte ordering is + /// endian independent and is always BGR byte order. + BGR = 6 +}; + +/** + * Enumeration that lists the supported LogLevels that can be set by users. + */ +enum class LogLevel_t +{ + /// Enumeration variable to be used by user to set logging level to FATAL. + LOG_FATAL = 0, + + /// Enumeration variable to be used by user to set logging level to ERROR. + LOG_ERROR = 1, + + /// Enumeration variable to be used by user to set logging level to WARN. + LOG_WARN = 2, + + /// Enumeration variable to be used by user to set logging level to INFO. + LOG_INFO = 3, + + /// Enumeration variable to be used by user to set logging level to VERBOSE. + LOG_VERBOSE = 4, + + /// Any new enums should be added above this line + NUM_LOG_LEVELS +}; + +enum class IOBufferDataType_t : int +{ + UNSPECIFIED = 0, + FLOATING_POINT_32 = 1, + FLOATING_POINT_16 = 2, + FIXED_POINT_8 = 3, + FIXED_POINT_16 = 4, + INT_32 = 5, + UINT_32 = 6, + INT_8 = 7, + UINT_8 = 8, + INT_16 = 9, + UINT_16 = 10, + BOOL_8 = 11, + INT_64 = 12, + UINT_64 = 13 +}; + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, Runtime_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, RuntimeCheckOption_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, PerformanceProfile_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, ProfilingLevel_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, ExecutionPriorityHint_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, ImageEncoding_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, LogLevel_t) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, IOBufferDataType_t) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.h new file mode 100644 index 00000000..f8c216ea --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.h @@ -0,0 +1,299 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _DL_ERROR_H_ +#define _DL_ERROR_H_ + +#ifdef __cplusplus +#include +#else +#include +#endif + +#include "SnpeApiExportDefine.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * Enumeration of error codes + */ +typedef enum +{ + /// Indicate success: SNPE_SUCCESS = 0 + SNPE_SUCCESS = 0, + + // C API Error Codes + // This is a temporary place for them. We still have to figure out how to manage + // passing error codes from the C API to C++ if we want to use things like SetLastError + SNPE_ERRORCODE_CAPI_CREATE_FAILURE = 10, + SNPE_ERRORCODE_CAPI_HANDLEGEN_FAILURE = 11, + SNPE_ERRORCODE_CAPI_DELETE_FAILURE = 12, + SNPE_ERRORCODE_CAPI_BAD_HANDLE = 13, + SNPE_ERRORCODE_CAPI_BAD_ARGUMENT = 14, + SNPE_ERRORCODE_CAPI_BAD_ALLOC = 15, + + // System config errors + SNPE_ERRORCODE_CONFIG_MISSING_PARAM = 100, + SNPE_ERRORCODE_CONFIG_INVALID_PARAM = 101, + SNPE_ERRORCODE_CONFIG_MISSING_FILE = 102, + SNPE_ERRORCODE_CONFIG_NNCONFIG_NOT_SET = 103, + SNPE_ERRORCODE_CONFIG_NNCONFIG_INVALID = 104, + SNPE_ERRORCODE_CONFIG_WRONG_INPUT_NAME = 105, + SNPE_ERRORCODE_CONFIG_INCORRECT_INPUT_DIMENSIONS = 106, + SNPE_ERRORCODE_CONFIG_DIMENSIONS_MODIFICATION_NOT_SUPPORTED = 107, + SNPE_ERRORCODE_CONFIG_BOTH_OUTPUT_LAYER_TENSOR_NAMES_SET = 108, + + SNPE_ERRORCODE_CONFIG_NNCONFIG_ONLY_TENSOR_SUPPORTED = 120, + SNPE_ERRORCODE_CONFIG_NNCONFIG_ONLY_USER_BUFFER_SUPPORTED = 121, + + // DlSystem errors + SNPE_ERRORCODE_DLSYSTEM_MISSING_BUFFER = 200, + SNPE_ERRORCODE_DLSYSTEM_TENSOR_CAST_FAILED = 201, + SNPE_ERRORCODE_DLSYSTEM_FIXED_POINT_PARAM_INVALID = 202, + SNPE_ERRORCODE_DLSYSTEM_SIZE_MISMATCH = 203, + SNPE_ERRORCODE_DLSYSTEM_NAME_NOT_FOUND = 204, + SNPE_ERRORCODE_DLSYSTEM_VALUE_MISMATCH = 205, + SNPE_ERRORCODE_DLSYSTEM_INSERT_FAILED = 206, + SNPE_ERRORCODE_DLSYSTEM_TENSOR_FILE_READ_FAILED = 207, + SNPE_ERRORCODE_DLSYSTEM_DIAGLOG_FAILURE = 208, + SNPE_ERRORCODE_DLSYSTEM_LAYER_NOT_SET = 209, + SNPE_ERRORCODE_DLSYSTEM_WRONG_NUMBER_INPUT_BUFFERS = 210, + SNPE_ERRORCODE_DLSYSTEM_RUNTIME_TENSOR_SHAPE_MISMATCH = 211, + SNPE_ERRORCODE_DLSYSTEM_TENSOR_MISSING = 212, + SNPE_ERRORCODE_DLSYSTEM_TENSOR_ITERATION_UNSUPPORTED = 213, + SNPE_ERRORCODE_DLSYSTEM_BUFFER_MANAGER_MISSING = 214, + SNPE_ERRORCODE_DLSYSTEM_RUNTIME_BUFFER_SOURCE_UNSUPPORTED = 215, + SNPE_ERRORCODE_DLSYSTEM_BUFFER_CAST_FAILED = 216, + SNPE_ERRORCODE_DLSYSTEM_WRONG_TRANSITION_TYPE = 217, + SNPE_ERRORCODE_DLSYSTEM_LAYER_ALREADY_REGISTERED = 218, + SNPE_ERRORCODE_DLSYSTEM_TENSOR_DIM_INVALID = 219, + + SNPE_ERRORCODE_DLSYSTEM_BUFFERENCODING_UNKNOWN = 240, + SNPE_ERRORCODE_DLSYSTEM_BUFFER_INVALID_PARAM = 241, + + // DlContainer errors + SNPE_ERRORCODE_DLCONTAINER_MODEL_PARSING_FAILED = 300, + SNPE_ERRORCODE_DLCONTAINER_UNKNOWN_LAYER_CODE = 301, + SNPE_ERRORCODE_DLCONTAINER_MISSING_LAYER_PARAM = 302, + SNPE_ERRORCODE_DLCONTAINER_LAYER_PARAM_NOT_SUPPORTED = 303, + SNPE_ERRORCODE_DLCONTAINER_LAYER_PARAM_INVALID = 304, + SNPE_ERRORCODE_DLCONTAINER_TENSOR_DATA_MISSING = 305, + SNPE_ERRORCODE_DLCONTAINER_MODEL_LOAD_FAILED = 306, + SNPE_ERRORCODE_DLCONTAINER_MISSING_RECORDS = 307, + SNPE_ERRORCODE_DLCONTAINER_INVALID_RECORD = 308, + SNPE_ERRORCODE_DLCONTAINER_WRITE_FAILURE = 309, + SNPE_ERRORCODE_DLCONTAINER_READ_FAILURE = 310, + SNPE_ERRORCODE_DLCONTAINER_BAD_CONTAINER = 311, + SNPE_ERRORCODE_DLCONTAINER_BAD_DNN_FORMAT_VERSION = 312, + SNPE_ERRORCODE_DLCONTAINER_UNKNOWN_AXIS_ANNOTATION = 313, + SNPE_ERRORCODE_DLCONTAINER_UNKNOWN_SHUFFLE_TYPE = 314, + SNPE_ERRORCODE_DLCONTAINER_TEMP_FILE_FAILURE = 315, + + // Network errors + SNPE_ERRORCODE_NETWORK_EMPTY_NETWORK = 400, + SNPE_ERRORCODE_NETWORK_CREATION_FAILED = 401, + SNPE_ERRORCODE_NETWORK_PARTITION_FAILED = 402, + SNPE_ERRORCODE_NETWORK_NO_OUTPUT_DEFINED = 403, + SNPE_ERRORCODE_NETWORK_MISMATCH_BETWEEN_NAMES_AND_DIMS = 404, + SNPE_ERRORCODE_NETWORK_MISSING_INPUT_NAMES = 405, + SNPE_ERRORCODE_NETWORK_MISSING_OUTPUT_NAMES = 406, + SNPE_ERRORCODE_NETWORK_EXECUTION_FAILED = 407, + + // Host runtime errors + SNPE_ERRORCODE_HOST_RUNTIME_TARGET_UNAVAILABLE = 500, + + // CPU runtime errors + SNPE_ERRORCODE_CPU_LAYER_NOT_SUPPORTED = 600, + SNPE_ERRORCODE_CPU_LAYER_PARAM_NOT_SUPPORTED = 601, + SNPE_ERRORCODE_CPU_LAYER_PARAM_INVALID = 602, + SNPE_ERRORCODE_CPU_LAYER_PARAM_COMBINATION_INVALID = 603, + SNPE_ERRORCODE_CPU_BUFFER_NOT_FOUND = 604, + SNPE_ERRORCODE_CPU_NETWORK_NOT_SUPPORTED = 605, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_ERRORCODE_CPU_UDO_OPERATION_FAILED = 606, +#endif //DNN_RUNTIME_HAVE_UDO_CAPABILITY + + // CPU fixed-point runtime errors + SNPE_ERRORCODE_CPU_FXP_LAYER_NOT_SUPPORTED = 700, + SNPE_ERRORCODE_CPU_FXP_LAYER_PARAM_NOT_SUPPORTED = 701, + SNPE_ERRORCODE_CPU_FXP_LAYER_PARAM_INVALID = 702, + SNPE_ERRORCODE_CPU_FXP_OPTION_INVALID = 703, + + // GPU runtime errors + SNPE_ERRORCODE_GPU_LAYER_NOT_SUPPORTED = 800, + SNPE_ERRORCODE_GPU_LAYER_PARAM_NOT_SUPPORTED = 801, + SNPE_ERRORCODE_GPU_LAYER_PARAM_INVALID = 802, + SNPE_ERRORCODE_GPU_LAYER_PARAM_COMBINATION_INVALID = 803, + SNPE_ERRORCODE_GPU_KERNEL_COMPILATION_FAILED = 804, + SNPE_ERRORCODE_GPU_CONTEXT_NOT_SET = 805, + SNPE_ERRORCODE_GPU_KERNEL_NOT_SET = 806, + SNPE_ERRORCODE_GPU_KERNEL_PARAM_INVALID = 807, + SNPE_ERRORCODE_GPU_OPENCL_CHECK_FAILED = 808, + SNPE_ERRORCODE_GPU_OPENCL_FUNCTION_ERROR = 809, + SNPE_ERRORCODE_GPU_BUFFER_NOT_FOUND = 810, + SNPE_ERRORCODE_GPU_TENSOR_DIM_INVALID = 811, + SNPE_ERRORCODE_GPU_MEMORY_FLAGS_INVALID = 812, + SNPE_ERRORCODE_GPU_UNEXPECTED_NUMBER_OF_IO = 813, + SNPE_ERRORCODE_GPU_LAYER_PROXY_ERROR = 814, + SNPE_ERRORCODE_GPU_BUFFER_IN_USE = 815, + SNPE_ERRORCODE_GPU_BUFFER_MODIFICATION_ERROR = 816, + SNPE_ERRORCODE_GPU_DATA_ARRANGEMENT_INVALID = 817, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_ERRORCODE_GPU_UDO_OPERATION_FAILED = 818, +#endif //DNN_RUNTIME_HAVE_UDO_CAPABILITY + // DSP runtime errors + SNPE_ERRORCODE_DSP_LAYER_NOT_SUPPORTED = 900, + SNPE_ERRORCODE_DSP_LAYER_PARAM_NOT_SUPPORTED = 901, + SNPE_ERRORCODE_DSP_LAYER_PARAM_INVALID = 902, + SNPE_ERRORCODE_DSP_LAYER_PARAM_COMBINATION_INVALID = 903, + SNPE_ERRORCODE_DSP_STUB_NOT_PRESENT = 904, + SNPE_ERRORCODE_DSP_LAYER_NAME_TRUNCATED = 905, + SNPE_ERRORCODE_DSP_LAYER_INPUT_BUFFER_NAME_TRUNCATED = 906, + SNPE_ERRORCODE_DSP_LAYER_OUTPUT_BUFFER_NAME_TRUNCATED = 907, + SNPE_ERRORCODE_DSP_RUNTIME_COMMUNICATION_ERROR = 908, + SNPE_ERRORCODE_DSP_RUNTIME_INVALID_PARAM_ERROR = 909, + SNPE_ERRORCODE_DSP_RUNTIME_SYSTEM_ERROR = 910, + SNPE_ERRORCODE_DSP_RUNTIME_CRASHED_ERROR = 911, + SNPE_ERRORCODE_DSP_BUFFER_SIZE_ERROR = 912, + SNPE_ERRORCODE_DSP_UDO_EXECUTE_ERROR = 913, + SNPE_ERRORCODE_DSP_UDO_LIB_NOT_REGISTERED_ERROR = 914, + SNPE_ERRORCODE_DSP_UDO_INVALID_QUANTIZATION_TYPE_ERROR = 915, + + // Model validataion errors + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_NOT_SUPPORTED = 1000, + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_PARAM_NOT_SUPPORTED = 1001, + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_PARAM_INVALID = 1002, + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_PARAM_MISSING = 1003, + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_PARAM_COMBINATION_INVALID = 1004, + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_ORDERING_INVALID = 1005, + SNPE_ERRORCODE_MODEL_VALIDATION_INVALID_CONSTRAINT = 1006, + SNPE_ERRORCODE_MODEL_VALIDATION_MISSING_BUFFER = 1007, + SNPE_ERRORCODE_MODEL_VALIDATION_BUFFER_REUSE_NOT_SUPPORTED = 1008, + SNPE_ERRORCODE_MODEL_VALIDATION_LAYER_COULD_NOT_BE_ASSIGNED = 1009, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_ERRORCODE_MODEL_VALIDATION_UDO_LAYER_FAILED = 1010, +#endif // DNN_RUNTIME_HAVE_UDO_CAPABILITY + + // UDL errors + SNPE_ERRORCODE_UDL_LAYER_EMPTY_UDL_NETWORK = 1100, + SNPE_ERRORCODE_UDL_LAYER_PARAM_INVALID = 1101, + SNPE_ERRORCODE_UDL_LAYER_INSTANCE_MISSING = 1102, + SNPE_ERRORCODE_UDL_LAYER_SETUP_FAILED = 1103, + SNPE_ERRORCODE_UDL_EXECUTE_FAILED = 1104, + SNPE_ERRORCODE_UDL_BUNDLE_INVALID = 1105, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_ERRORCODE_UDO_REGISTRATION_FAILED = 1106, + SNPE_ERRORCODE_UDO_GET_PACKAGE_FAILED = 1107, + SNPE_ERRORCODE_UDO_GET_IMPLEMENTATION_FAILED = 1108, +#endif // DNN_RUNTIME_HAVE_UDO_CAPABILITY + + // Dependent library errors + SNPE_ERRORCODE_STD_LIBRARY_ERROR = 1200, + + // Unknown exception (catch (...)), Has no component attached to this + SNPE_ERRORCODE_UNKNOWN_EXCEPTION = 1210, + + // Storage Errors + SNPE_ERRORCODE_STORAGE_INVALID_KERNEL_REPO = 1300, + +#ifdef DNN_RUNTIME_HAVE_AIP_RUNTIME + // AIP runtime errors + SNPE_ERRORCODE_AIP_LAYER_NOT_SUPPORTED = 1400, + SNPE_ERRORCODE_AIP_LAYER_PARAM_NOT_SUPPORTED = 1401, + SNPE_ERRORCODE_AIP_LAYER_PARAM_INVALID = 1402, + SNPE_ERRORCODE_AIP_LAYER_PARAM_COMBINATION_INVALID = 1403, + SNPE_ERRORCODE_AIP_STUB_NOT_PRESENT = 1404, + SNPE_ERRORCODE_AIP_LAYER_NAME_TRUNCATED = 1405, + SNPE_ERRORCODE_AIP_LAYER_INPUT_BUFFER_NAME_TRUNCATED = 1406, + SNPE_ERRORCODE_AIP_LAYER_OUTPUT_BUFFER_NAME_TRUNCATED = 1407, + SNPE_ERRORCODE_AIP_RUNTIME_COMMUNICATION_ERROR = 1408, + SNPE_ERRORCODE_AIP_RUNTIME_INVALID_PARAM_ERROR = 1409, + SNPE_ERRORCODE_AIP_RUNTIME_SYSTEM_ERROR = 1410, + SNPE_ERRORCODE_AIP_RUNTIME_TENSOR_MISSING = 1411, + SNPE_ERRORCODE_AIP_RUNTIME_TENSOR_SHAPE_MISMATCH = 1412, + SNPE_ERRORCODE_AIP_RUNTIME_BAD_AIX_RECORD = 1413, +#endif // DNN_RUNTIME_HAVE_AIP_RUNTIME + + // DlCaching errors + SNPE_ERRORCODE_DLCACHING_INVALID_METADATA = 1500, + SNPE_ERRORCODE_DLCACHING_INVALID_INITBLOB = 1501, + + // Infrastructure Errors + SNPE_ERRORCODE_INFRA_CLUSTERMGR_INSTANCE_INVALID = 1600, + SNPE_ERRORCODE_INFRA_CLUSTERMGR_EXECUTE_SYNC_FAILED = 1601, + + // Memory Errors + SNPE_ERRORCODE_MEMORY_CORRUPTION_ERROR = 1700 + +} Snpe_ErrorCode_t; + + + +/** + * Clear the last error code + */ +SNPE_API void Snpe_ErrorCode_clearLastErrorCode(); + +/** +* Returns the error code of the last error encountered. +* +* @return The error code. +* +* @note The returned error code is significant only when the return +* value of the call indicated an error. +*/ +SNPE_API Snpe_ErrorCode_t Snpe_ErrorCode_getLastErrorCode(); + +/** +* Returns the error string of the last error encountered. +* +* @return The error string. +* +* @note The returned error string is significant only when the return +* value of the call indicated an error. +*/ +SNPE_API const char* Snpe_ErrorCode_GetLastErrorString(); + +/** + * Returns the info string of the last error encountered. + */ +SNPE_API const char* Snpe_ErrorCode_getLastInfoString(); + +/** + * Returns the uint32_t representation of the error code enum. + * + * @param[in] code The error code to be converted. + * + * @return uint32_t representation of the error code. + */ +SNPE_API uint32_t Snpe_ErrorCode_enumToUInt32(Snpe_ErrorCode_t code); + + +#ifdef __cplusplus +} // extern "C" +#endif + + +#endif // _DL_ERROR_H_ + diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.hpp new file mode 100644 index 00000000..55dc2140 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlError.hpp @@ -0,0 +1,261 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include + +#include "DlSystem/DlError.h" + + +namespace DlSystem { + +enum class ErrorCode : uint32_t { + NONE = 0, + + // C API Error Codes + // This is a temporary place for them. We still have to figure out how to manage + // passing error codes from the C API to C++ if we want to use things like SetLastError + SNPE_CAPI_CREATE_FAILURE = 10, + SNPE_CAPI_HANDLEGEN_FAILURE = 11, + SNPE_CAPI_DELETE_FAILURE = 12, + SNPE_CAPI_BAD_HANDLE = 13, + SNPE_CAPI_BAD_ARGUMENT = 14, + SNPE_CAPI_BAD_ALLOC = 15, + + + // System config errors + SNPE_CONFIG_MISSING_PARAM = 100, + SNPE_CONFIG_INVALID_PARAM = 101, + SNPE_CONFIG_MISSING_FILE = 102, + SNPE_CONFIG_NNCONFIG_NOT_SET = 103, + SNPE_CONFIG_NNCONFIG_INVALID = 104, + SNPE_CONFIG_WRONG_INPUT_NAME = 105, + SNPE_CONFIG_INCORRECT_INPUT_DIMENSIONS = 106, + SNPE_CONFIG_DIMENSIONS_MODIFICATION_NOT_SUPPORTED = 107, + SNPE_CONFIG_BOTH_OUTPUT_LAYER_TENSOR_NAMES_SET = 108, + + SNPE_CONFIG_NNCONFIG_ONLY_TENSOR_SUPPORTED = 120, + SNPE_CONFIG_NNCONFIG_ONLY_USER_BUFFER_SUPPORTED = 121, + + // DlSystem errors + SNPE_DLSYSTEM_MISSING_BUFFER = 200, + SNPE_DLSYSTEM_TENSOR_CAST_FAILED = 201, + SNPE_DLSYSTEM_FIXED_POINT_PARAM_INVALID = 202, + SNPE_DLSYSTEM_SIZE_MISMATCH = 203, + SNPE_DLSYSTEM_NAME_NOT_FOUND = 204, + SNPE_DLSYSTEM_VALUE_MISMATCH = 205, + SNPE_DLSYSTEM_INSERT_FAILED = 206, + SNPE_DLSYSTEM_TENSOR_FILE_READ_FAILED = 207, + SNPE_DLSYSTEM_DIAGLOG_FAILURE = 208, + SNPE_DLSYSTEM_LAYER_NOT_SET = 209, + SNPE_DLSYSTEM_WRONG_NUMBER_INPUT_BUFFERS = 210, + SNPE_DLSYSTEM_RUNTIME_TENSOR_SHAPE_MISMATCH = 211, + SNPE_DLSYSTEM_TENSOR_MISSING = 212, + SNPE_DLSYSTEM_TENSOR_ITERATION_UNSUPPORTED = 213, + SNPE_DLSYSTEM_BUFFER_MANAGER_MISSING = 214, + SNPE_DLSYSTEM_RUNTIME_BUFFER_SOURCE_UNSUPPORTED = 215, + SNPE_DLSYSTEM_BUFFER_CAST_FAILED = 216, + SNPE_DLSYSTEM_WRONG_TRANSITION_TYPE = 217, + SNPE_DLSYSTEM_LAYER_ALREADY_REGISTERED = 218, + SNPE_DLSYSTEM_TENSOR_DIM_INVALID = 219, + + SNPE_DLSYSTEM_BUFFERENCODING_UNKNOWN = 240, + SNPE_DLSYSTEM_BUFFER_INVALID_PARAM = 241, + + // DlContainer errors + SNPE_DLCONTAINER_MODEL_PARSING_FAILED = 300, + SNPE_DLCONTAINER_UNKNOWN_LAYER_CODE = 301, + SNPE_DLCONTAINER_MISSING_LAYER_PARAM = 302, + SNPE_DLCONTAINER_LAYER_PARAM_NOT_SUPPORTED = 303, + SNPE_DLCONTAINER_LAYER_PARAM_INVALID = 304, + SNPE_DLCONTAINER_TENSOR_DATA_MISSING = 305, + SNPE_DLCONTAINER_MODEL_LOAD_FAILED = 306, + SNPE_DLCONTAINER_MISSING_RECORDS = 307, + SNPE_DLCONTAINER_INVALID_RECORD = 308, + SNPE_DLCONTAINER_WRITE_FAILURE = 309, + SNPE_DLCONTAINER_READ_FAILURE = 310, + SNPE_DLCONTAINER_BAD_CONTAINER = 311, + SNPE_DLCONTAINER_BAD_DNN_FORMAT_VERSION = 312, + SNPE_DLCONTAINER_UNKNOWN_AXIS_ANNOTATION = 313, + SNPE_DLCONTAINER_UNKNOWN_SHUFFLE_TYPE = 314, + SNPE_DLCONTAINER_TEMP_FILE_FAILURE = 315, + + // Network errors + SNPE_NETWORK_EMPTY_NETWORK = 400, + SNPE_NETWORK_CREATION_FAILED = 401, + SNPE_NETWORK_PARTITION_FAILED = 402, + SNPE_NETWORK_NO_OUTPUT_DEFINED = 403, + SNPE_NETWORK_MISMATCH_BETWEEN_NAMES_AND_DIMS = 404, + SNPE_NETWORK_MISSING_INPUT_NAMES = 405, + SNPE_NETWORK_MISSING_OUTPUT_NAMES = 406, + SNPE_NETWORK_EXECUTION_FAILED = 407, + + // Host runtime errors + SNPE_HOST_RUNTIME_TARGET_UNAVAILABLE = 500, + + // CPU runtime errors + SNPE_CPU_LAYER_NOT_SUPPORTED = 600, + SNPE_CPU_LAYER_PARAM_NOT_SUPPORTED = 601, + SNPE_CPU_LAYER_PARAM_INVALID = 602, + SNPE_CPU_LAYER_PARAM_COMBINATION_INVALID = 603, + SNPE_CPU_BUFFER_NOT_FOUND = 604, + SNPE_CPU_NETWORK_NOT_SUPPORTED = 605, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_CPU_UDO_OPERATION_FAILED = 606, +#endif //DNN_RUNTIME_HAVE_UDO_CAPABILITY + + // CPU fixed-point runtime errors + SNPE_CPU_FXP_LAYER_NOT_SUPPORTED = 700, + SNPE_CPU_FXP_LAYER_PARAM_NOT_SUPPORTED = 701, + SNPE_CPU_FXP_LAYER_PARAM_INVALID = 702, + SNPE_CPU_FXP_OPTION_INVALID = 703, + + // GPU runtime errors + SNPE_GPU_LAYER_NOT_SUPPORTED = 800, + SNPE_GPU_LAYER_PARAM_NOT_SUPPORTED = 801, + SNPE_GPU_LAYER_PARAM_INVALID = 802, + SNPE_GPU_LAYER_PARAM_COMBINATION_INVALID = 803, + SNPE_GPU_KERNEL_COMPILATION_FAILED = 804, + SNPE_GPU_CONTEXT_NOT_SET = 805, + SNPE_GPU_KERNEL_NOT_SET = 806, + SNPE_GPU_KERNEL_PARAM_INVALID = 807, + SNPE_GPU_OPENCL_CHECK_FAILED = 808, + SNPE_GPU_OPENCL_FUNCTION_ERROR = 809, + SNPE_GPU_BUFFER_NOT_FOUND = 810, + SNPE_GPU_TENSOR_DIM_INVALID = 811, + SNPE_GPU_MEMORY_FLAGS_INVALID = 812, + SNPE_GPU_UNEXPECTED_NUMBER_OF_IO = 813, + SNPE_GPU_LAYER_PROXY_ERROR = 814, + SNPE_GPU_BUFFER_IN_USE = 815, + SNPE_GPU_BUFFER_MODIFICATION_ERROR = 816, + SNPE_GPU_DATA_ARRANGEMENT_INVALID = 817, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_GPU_UDO_OPERATION_FAILED = 818, +#endif //DNN_RUNTIME_HAVE_UDO_CAPABILITY + // DSP runtime errors + SNPE_DSP_LAYER_NOT_SUPPORTED = 900, + SNPE_DSP_LAYER_PARAM_NOT_SUPPORTED = 901, + SNPE_DSP_LAYER_PARAM_INVALID = 902, + SNPE_DSP_LAYER_PARAM_COMBINATION_INVALID = 903, + SNPE_DSP_STUB_NOT_PRESENT = 904, + SNPE_DSP_LAYER_NAME_TRUNCATED = 905, + SNPE_DSP_LAYER_INPUT_BUFFER_NAME_TRUNCATED = 906, + SNPE_DSP_LAYER_OUTPUT_BUFFER_NAME_TRUNCATED = 907, + SNPE_DSP_RUNTIME_COMMUNICATION_ERROR = 908, + SNPE_DSP_RUNTIME_INVALID_PARAM_ERROR = 909, + SNPE_DSP_RUNTIME_SYSTEM_ERROR = 910, + SNPE_DSP_RUNTIME_CRASHED_ERROR = 911, + SNPE_DSP_BUFFER_SIZE_ERROR = 912, + SNPE_DSP_UDO_EXECUTE_ERROR = 913, + SNPE_DSP_UDO_LIB_NOT_REGISTERED_ERROR = 914, + SNPE_DSP_UDO_INVALID_QUANTIZATION_TYPE_ERROR = 915, + SNPE_DSP_RUNTIME_INVALID_RPC_DRIVER = 916, + SNPE_DSP_RUNTIME_RPC_PERMISSION_ERROR = 917, + SNPE_DSP_RUNTIME_DSP_FILE_OPEN_ERROR = 918, + + // Model validataion errors + SNPE_MODEL_VALIDATION_LAYER_NOT_SUPPORTED = 1000, + SNPE_MODEL_VALIDATION_LAYER_PARAM_NOT_SUPPORTED = 1001, + SNPE_MODEL_VALIDATION_LAYER_PARAM_INVALID = 1002, + SNPE_MODEL_VALIDATION_LAYER_PARAM_MISSING = 1003, + SNPE_MODEL_VALIDATION_LAYER_PARAM_COMBINATION_INVALID = 1004, + SNPE_MODEL_VALIDATION_LAYER_ORDERING_INVALID = 1005, + SNPE_MODEL_VALIDATION_INVALID_CONSTRAINT = 1006, + SNPE_MODEL_VALIDATION_MISSING_BUFFER = 1007, + SNPE_MODEL_VALIDATION_BUFFER_REUSE_NOT_SUPPORTED = 1008, + SNPE_MODEL_VALIDATION_LAYER_COULD_NOT_BE_ASSIGNED = 1009, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_MODEL_VALIDATION_UDO_LAYER_FAILED = 1010, +#endif // DNN_RUNTIME_HAVE_UDO_CAPABILITY + + // UDL errors + SNPE_UDL_LAYER_EMPTY_UDL_NETWORK = 1100, + SNPE_UDL_LAYER_PARAM_INVALID = 1101, + SNPE_UDL_LAYER_INSTANCE_MISSING = 1102, + SNPE_UDL_LAYER_SETUP_FAILED = 1103, + SNPE_UDL_EXECUTE_FAILED = 1104, + SNPE_UDL_BUNDLE_INVALID = 1105, +#ifdef DNN_RUNTIME_HAVE_UDO_CAPABILITY + SNPE_UDO_REGISTRATION_FAILED = 1106, + SNPE_UDO_GET_PACKAGE_FAILED = 1107, + SNPE_UDO_GET_IMPLEMENTATION_FAILED = 1108, +#endif // DNN_RUNTIME_HAVE_UDO_CAPABILITY + + // Dependent library errors + SNPE_STD_LIBRARY_ERROR = 1200, + + // Unknown exception (catch (...)), Has no component attached to this + SNPE_UNKNOWN_EXCEPTION = 1210, + + // Storage Errors + SNPE_STORAGE_INVALID_KERNEL_REPO = 1300, + +#ifdef DNN_RUNTIME_HAVE_AIP_RUNTIME + // AIP runtime errors + SNPE_AIP_LAYER_NOT_SUPPORTED = 1400, + SNPE_AIP_LAYER_PARAM_NOT_SUPPORTED = 1401, + SNPE_AIP_LAYER_PARAM_INVALID = 1402, + SNPE_AIP_LAYER_PARAM_COMBINATION_INVALID = 1403, + SNPE_AIP_STUB_NOT_PRESENT = 1404, + SNPE_AIP_LAYER_NAME_TRUNCATED = 1405, + SNPE_AIP_LAYER_INPUT_BUFFER_NAME_TRUNCATED = 1406, + SNPE_AIP_LAYER_OUTPUT_BUFFER_NAME_TRUNCATED = 1407, + SNPE_AIP_RUNTIME_COMMUNICATION_ERROR = 1408, + SNPE_AIP_RUNTIME_INVALID_PARAM_ERROR = 1409, + SNPE_AIP_RUNTIME_SYSTEM_ERROR = 1410, + SNPE_AIP_RUNTIME_TENSOR_MISSING = 1411, + SNPE_AIP_RUNTIME_TENSOR_SHAPE_MISMATCH = 1412, + SNPE_AIP_RUNTIME_BAD_AIX_RECORD = 1413, + SNPE_AIP_AXIS_QUANT_UNSUPPORTED = 1414, + +#endif // DNN_RUNTIME_HAVE_AIP_RUNTIME + + // DlCaching errors + SNPE_DLCACHING_INVALID_METADATA = 1500, + SNPE_DLCACHING_INVALID_INITBLOB = 1501, + + // Infrastructure Errors + SNPE_INFRA_CLUSTERMGR_INSTANCE_INVALID = 1600, + SNPE_INFRA_CLUSTERMGR_EXECUTE_SYNC_FAILED = 1601, + + // Memory Errors + SNPE_MEMORY_CORRUPTION_ERROR = 1700 + +}; + + +inline ErrorCode getLastErrorCode(){ + return static_cast(Snpe_ErrorCode_getLastErrorCode()); +} + +inline const char* getLastErrorString(){ + return Snpe_ErrorCode_GetLastErrorString(); +} + +inline const char* getLastInfoString(){ + return Snpe_ErrorCode_getLastInfoString(); +} + + +inline uint32_t enumToUInt32(ErrorCode code){ + return Snpe_ErrorCode_enumToUInt32(static_cast(code)); +} + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, ErrorCode); + + +namespace zdl{ namespace DlSystem { + inline ErrorCode getLastErrorCode() { return ::DlSystem::getLastErrorCode() ; } + inline const char* getLastErrorString() { return ::DlSystem::getLastErrorString() ; } + inline const char* getLastInfoString() { return ::DlSystem::getLastInfoString() ; } + inline uint32_t enumToUInt32(ErrorCode code){ return ::DlSystem::enumToUInt32(code); } +}} // ns zdl::DlSystem diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlOptional.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlOptional.hpp new file mode 100644 index 00000000..e7bbf666 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlOptional.hpp @@ -0,0 +1,244 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include +#include + + +//============================================================================== +// +// Copyright (c) 2016, 2020 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +//#include +#include +//#include + + +namespace DlSystem { + + +/** @addtogroup c_plus_plus_apis C++ +@{ */ + +/** + * @brief . + * + * Class to manage a value that may or may not exist. The boolean value + * of the Optional class is true if the object contains a value and false + * if it does not contain a value. + * + * The class must be evaluated and confirmed as true (containing a value) + * before being dereferenced. + */ +template +class Optional { +public: + enum class LIFECYCLE { + NONE = 0, + REFERENCE_OWNED = 1, + POINTER_OWNED = 2, + POINTER_NOT_OWNED = 3 + }; + + struct ReferenceCount { + size_t count = 0; + + void increment() { count++; } + + size_t decrement() { + if (count > 0) { + count--; + } + return count; + } + }; + + using U = typename std::remove_pointer::type; + + /** + * The default constructor is set to not have any value, and is + * therefore evaluated as false. + */ + // Do not explicit it so we can return {} + Optional() { + m_Type = LIFECYCLE::NONE; + } + + /** + * Construct an Optional class using an object. + * @param[in] Reference to an object v + * @param[out] Optional instance of object v + */ + template + Optional (const T& v, typename std::enable_if::value>::type* = 0) + : m_Type(LIFECYCLE::REFERENCE_OWNED) { + try { + m_StoragePtr = new T(v); + } catch (...) { + m_StoragePtr = nullptr; + m_Type = LIFECYCLE::NONE; + } + } + + template + Optional(U* v, LIFECYCLE type, typename std::enable_if::value>::type* = 0) + : m_Type(type) { + switch (m_Type) { + case LIFECYCLE::POINTER_OWNED: + m_StoragePtr = v; + m_Count = new ReferenceCount(); + m_Count->increment(); + break; + case LIFECYCLE::POINTER_NOT_OWNED: + m_StoragePtr = v; + break; + case LIFECYCLE::REFERENCE_OWNED: + throw std::bad_exception(); + case LIFECYCLE::NONE: + break; + } + } + + Optional(const Optional &other) : m_Type(other.m_Type), m_Count(other.m_Count) { + if (isReference()) { + m_StoragePtr = new U(*other.m_StoragePtr); + } else if (isPointer()) { + m_StoragePtr = other.m_StoragePtr; + if (isOwned()) { + m_Count->increment(); + } + } + } + + Optional& operator=(const Optional& other) noexcept { + Optional tmp(other); + swap(std::move(tmp)); + return *this; + } + + Optional(Optional&& other) noexcept { + swap(std::move(other)); + } + + Optional& operator=(Optional&& other) noexcept { + swap(std::move(other)); + return *this; + } + + ~Optional() { + if (isOwned()) { + if (isReference() || (isPointer() && m_Count->decrement() == 0)) { + delete m_StoragePtr; + delete m_Count; + } + } + } + + /** + * Boolean value of Optional class is only true when there exists a value. + */ + operator bool() const noexcept { return isValid(); } + + bool operator!() const noexcept { return !isValid(); } + + /** + * Get reference of Optional object + * @warning User must validate Optional has value before. + */ + const T& operator*() { return this->GetReference(); } + + /** + * Get reference of Optional object + * @warning User must validate Optional has value before. + */ + const T& operator*() const { return this->GetReference(); } + + operator T&() { return this->GetReference(); } + + T operator->() { + T self = this->GetReference(); + return self; + } + + void release(){ + if(isOwned() && isPointer()){ + m_Type = LIFECYCLE::POINTER_NOT_OWNED; + if(m_Count && m_Count->decrement() == 0){ + delete m_Count; + m_Count = nullptr; + } + } + } +private: + void swap(Optional&& other) { + m_Type = other.m_Type; + m_StoragePtr = other.m_StoragePtr; + m_Count = other.m_Count; + + other.m_Type = LIFECYCLE::NONE; + other.m_StoragePtr = nullptr; + other.m_Count = nullptr; + } + + template + typename std::enable_if::value, const Q&>::type GetReference() const noexcept { + if (!isReference()) std::terminate(); + return *static_cast(m_StoragePtr); + } + + template + typename std::enable_if::value, const Q&>::type GetReference() const noexcept { + if (!isPointer()) std::terminate(); + return static_cast(m_StoragePtr); + } + + template + typename std::enable_if::value, Q&>::type GetReference() noexcept { + if (!isReference()) std::terminate(); + return *m_StoragePtr; + } + + template + typename std::enable_if::value, Q&>::type GetReference() noexcept { + if (!isPointer()) std::terminate(); + return m_StoragePtr; + } + + bool isPointer() const { + return m_Type == LIFECYCLE::POINTER_OWNED || m_Type == LIFECYCLE::POINTER_NOT_OWNED; + } + + bool isOwned() const { + return m_Type == LIFECYCLE::REFERENCE_OWNED || m_Type == LIFECYCLE::POINTER_OWNED; + } + + bool isReference() const { + return m_Type == LIFECYCLE::REFERENCE_OWNED; + } + + bool isValid() const { + return m_Type != LIFECYCLE::NONE; + } + + U* m_StoragePtr = nullptr; + LIFECYCLE m_Type; + ReferenceCount *m_Count = nullptr; +}; + +} // ns DlSystem + + + +namespace zdl { namespace DlSystem { template using Optional = ::DlSystem::Optional; }} diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.h new file mode 100644 index 00000000..fac01d1c --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.h @@ -0,0 +1,122 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + + +/** + * @file + */ + +#ifndef _DL_VERSION_H_ +#define _DL_VERSION_H_ + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" +#include + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A class that contains the different portions of a version number. + * A typedef to indicate a SNPE DlVersion handle + */ +typedef void* Snpe_DlVersion_Handle_t; + +/** + * Construct a DlVersion + * + * @return a handle to the created DlVersion + */ +SNPE_API +Snpe_DlVersion_Handle_t Snpe_DlVersion_Create(); + + +/** + * Destroys/frees DlVersion + * + * @param[in] handle : Handle to access DlVersion + * + * @return SNPE_SUCCESS if Delete operation successful. +*/ +SNPE_API +Snpe_ErrorCode_t Snpe_DlVersion_Delete(Snpe_DlVersion_Handle_t handle); + +/** + * Get the major version number. + * @param[in] handle : Handle to access DlVersion + * @return Major version + */ +SNPE_API +int32_t Snpe_DlVersion_GetMajor(Snpe_DlVersion_Handle_t handle); + +/** + * Get the minor version number. + * @param[in] handle : Handle to access DlVersion + * @return Minor version + */ +SNPE_API +int32_t Snpe_DlVersion_GetMinor(Snpe_DlVersion_Handle_t handle); + +/** + * Get the teeny version number. + * @param[in] handle : Handle to access DlVersion + * @return Teeny version + */ +SNPE_API +int32_t Snpe_DlVersion_GetTeeny(Snpe_DlVersion_Handle_t handle); + +/** + * Get the string holding information about the build version. + * + * @param[in] handle : Handle to access DlVersion + * @return Build information + */ +SNPE_API +const char* Snpe_DlVersion_GetBuild(Snpe_DlVersion_Handle_t handle); + +/** + * @brief Returns a string in the form Major.Minor.Teeny.Build + * + * @param[in] handle : Handle to access DlVersion + * @return A formatted char* holding the version information. + * + * @note the returned string will be invalidated by subsequent calls to this function + */ +SNPE_API +const char* Snpe_DlVersion_ToString(Snpe_DlVersion_Handle_t handle); + +/** + * @brief Create a DlVersion from a string + * + * @param stringValue The formatted DlVersion string + * + * @return A handle to the created DlVersion + */ +SNPE_API +Snpe_DlVersion_Handle_t Snpe_DlVersion_FromString(const char* stringValue); + + + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _DL_VERSION_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.hpp new file mode 100644 index 00000000..7badab1f --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/DlVersion.hpp @@ -0,0 +1,118 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include +#include + +#include "Wrapper.hpp" +#include "String.hpp" + +#include "DlSystem/DlVersion.h" +#include "SNPE/SNPEUtil.h" + + +namespace DlSystem { + +class Version_t : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_DlVersion_Delete}; + + template + using MajorReference = WrapperDetail::GenericConstMemberReference; + + template + using MinorReference = WrapperDetail::GenericConstMemberReference; + + template + using TeenyReference = WrapperDetail::GenericConstMemberReference; + + + static std::string BuildGetter(Snpe_DlVersion_Handle_t handle){ + return Snpe_DlVersion_GetBuild(handle); + } + + template + using BuildReference = WrapperDetail::GenericConstMemberReference; + + + static const std::string& toString(int32_t Major, int32_t Minor, int32_t Teeny, const std::string& Build){ + thread_local std::string toret; + + toret = std::to_string(Major); + toret += '.'; + toret += std::to_string(Minor); + toret += '.'; + toret += std::to_string(Teeny); + if(!Build.empty()){ + toret += '.'; + toret += Build; + } + + return toret; + } + +public: + Version_t() + : BaseType(Snpe_DlVersion_Create()) + { } + + Version_t(int32_t Major, int32_t Minor, int32_t Teeny, const std::string& Build) + : BaseType(Snpe_DlVersion_FromString(toString(Major, Minor, Teeny, Build).c_str())) + { } + + + /// Holds the major version number. Changes in this value indicate + /// major changes that break backward compatibility. + MajorReference Major{*this}; + + /// Holds the minor version number. Changes in this value indicate + /// minor changes made to library that are backwards compatible + /// (such as additions to the interface). + MinorReference Minor{*this}; + + /// Holds the teeny version number. Changes in this value indicate + /// changes such as bug fixes and patches made to the library that + /// do not affect the interface. + TeenyReference Teeny{*this}; + + /// This string holds information about the build version. + BuildReference Build{*this}; + + + static Version_t fromString(const std::string& stringValue){ + return moveHandle(Snpe_DlVersion_FromString(stringValue.c_str())); + } + + /** + * @brief Returns a string in the form Major.Minor.Teeny.Build + * + * @return A formatted string holding the version information. + */ + std::string toString() const{ + return Snpe_DlVersion_ToString(handle()); + } + + /** + * @brief Returns a string in the form Major.Minor.Teeny.Build + * + * @return A formatted string holding the version information. + */ + String asString() const{ + return String(toString()); + } +}; + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, Version_t) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.h new file mode 100644 index 00000000..96453ef9 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.h @@ -0,0 +1,117 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _IBUFFER_ATTRIBUTES_H +#define _IBUFFER_ATTRIBUTES_H + +#include "DlSystem/IUserBuffer.h" +#include "DlSystem/TensorShape.h" +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE IBufferAttributes handle + */ +typedef void* Snpe_IBufferAttributes_Handle_t; + + +/** + * @brief Gets the buffer's element size, in bytes + * + * This can be used to compute the memory size required + * to back this buffer. + * + * @param[in] handle : Handle to access IBufferAttributes + * + * @return Element size, in bytes + */ +SNPE_API +size_t Snpe_IBufferAttributes_GetElementSize(Snpe_IBufferAttributes_Handle_t handle); + +/** + * @brief Gets the element's encoding type + * + * @param[in] handle : Handle to access IBufferAttributes + * + * @return encoding type + */ +SNPE_API +Snpe_UserBufferEncoding_ElementType_t Snpe_IBufferAttributes_GetEncodingType(Snpe_IBufferAttributes_Handle_t handle); + +/** + * @brief Gets the number of elements in each dimension + * + * @param[in] handle : Handle to access IBufferAttributes + * + * @return Dimension size, in terms of number of elements + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_IBufferAttributes_GetDims(Snpe_IBufferAttributes_Handle_t handle); + +/** + * @brief Gets the alignment requirement of each dimension + * + * Alignment per each dimension is expressed as an multiple, for + * example, if one particular dimension can accept multiples of 8, + * the alignment will be 8. + * + * @param[in] handle : Handle to access IBufferAttributes + * + * @return Alignment in each dimension, in terms of multiple of + * number of elements + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_IBufferAttributes_GetAlignments(Snpe_IBufferAttributes_Handle_t handle); + +/** + * @brief Gets the buffer encoding returned from the network responsible + * for generating this buffer. Depending on the encoding type, this will + * be an instance of an encoding type specific derived class. + * + * @param[in] handle : Handle to access IBufferAttributes + * + * @return Derived user buffer encoding object. + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_IBufferAttributes_GetEncoding_Ref(Snpe_IBufferAttributes_Handle_t handle); + +/** + * @brief Destroys the IBufferAttributes object + * + * @param[handle] handle : Handle to access IBufferAttributes + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IBufferAttributes_Delete(Snpe_IBufferAttributes_Handle_t handle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _IBUFFER_ATTRIBUTES_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.hpp new file mode 100644 index 00000000..2a86fcec --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IBufferAttributes.hpp @@ -0,0 +1,85 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include +#include "TensorShape.hpp" + +#include "DlSystem/IBufferAttributes.h" +#include "IUserBuffer.hpp" + +namespace DlSystem { + + +class IBufferAttributes : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_IBufferAttributes_Delete}; +public: + + size_t getElementSize() const noexcept{ + return Snpe_IBufferAttributes_GetElementSize(handle()); + } + + UserBufferEncoding::ElementType_t getEncodingType() const noexcept{ + return static_cast(Snpe_IBufferAttributes_GetEncodingType(handle())); + } + + TensorShape getDims() const{ + return moveHandle(Snpe_IBufferAttributes_GetDims(handle())); + } + + TensorShape getAlignments() const{ + return moveHandle(Snpe_IBufferAttributes_GetAlignments(handle())); + } + + UserBufferEncoding* getEncoding() const{ + auto h = Snpe_IBufferAttributes_GetEncoding_Ref(handle()); + switch(Snpe_UserBufferEncoding_GetElementType(h)){ + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT: + return makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNSIGNED8BIT: + return makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT8: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT32: + return makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT8: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT32: + return makeReference(h); + + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT16: + return makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF8: + return makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF16: + return makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_BOOL8: + return makeReference(h); + + default: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNKNOWN: + return makeReference(h); + } + } + +}; + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, IBufferAttributes) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.h new file mode 100644 index 00000000..a3c3c623 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.h @@ -0,0 +1,156 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_IOBUFFER_DATATYPE_MAP_H +#define DL_SYSTEM_IOBUFFER_DATATYPE_MAP_H + +#include + +#include "DlSystem/DlError.h" +#include "DlSystem/DlEnums.h" +#include "DlSystem/SnpeApiExportDefine.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE IOBufferDataTypeMap handle + */ +typedef void* Snpe_IOBufferDataTypeMap_Handle_t; + +/** + * @brief . + * + * Creates a new Buffer Data type map + * + */ +SNPE_API +Snpe_IOBufferDataTypeMap_Handle_t Snpe_IOBufferDataTypeMap_Create(); + +/** + * @brief Destroys the map + * + * @param[in] handle : Handle to access the IOBufferDataType map + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IOBufferDataTypeMap_Delete(Snpe_IOBufferDataTypeMap_Handle_t handle); +/** + * @brief Adds a name and the corresponding buffer data type + * to the map + * + * @param[in] handle : Handle to access the IOBufferDataType map + * + * @param[in] name : The name of the buffer + * + * @param[in] bufferDataType : data type of the buffer + * + * @note If a buffer with the same name already exists, no new + * buffer is added. + */ +SNPE_API +Snpe_ErrorCode_t +Snpe_IOBufferDataTypeMap_Add(Snpe_IOBufferDataTypeMap_Handle_t handle, const char* name, Snpe_IOBufferDataType_t bufferDataType); + +/** + * @brief Removes a buffer name from the map + * + * @param[in] handle : Handle to access the IOBufferDataType map + * + * @param[in] name : The name of the buffer + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IOBufferDataTypeMap_Remove(Snpe_IOBufferDataTypeMap_Handle_t handle, const char* name); + +/** + * @brief Returns the type of the named buffer + * + * @param[in] handle : Handle to access the IOBufferDataType map + * + * @param[in] name : The name of the buffer + * + * @return The type of the buffer, or UNSPECIFIED if the buffer does not exist + * + */ +SNPE_API +Snpe_IOBufferDataType_t Snpe_IOBufferDataTypeMap_GetBufferDataType(Snpe_IOBufferDataTypeMap_Handle_t handle, const char* name); + +/** + * @brief Returns the type of the first buffer + * + * @param handle : Handle to access the IOBufferDataType map + * + * @return The type of the first buffer, or SNPE_IO_BUFFER_DATATYPE_UNSPECIFIED if the map is empty. + */ +SNPE_API +Snpe_IOBufferDataType_t Snpe_IOBufferDataTypeMap_GetBufferDataTypeOfFirst(Snpe_IOBufferDataTypeMap_Handle_t handle); + +/** + * @brief Returns the size of the buffer type map. + * + * @param[in] handle : Handle to access the IOBufferDataType map + * + * @return The size of the map + * + */ +SNPE_API +size_t Snpe_IOBufferDataTypeMap_Size(Snpe_IOBufferDataTypeMap_Handle_t handle); + +/** + * @brief Checks the existence of the named buffer in the map + * + * @param[in] handle : Handle to access the IOBufferDataType map + * + * @param[in] name : The name of the buffer + * + * @return 1 if the named buffer exists, 0 otherwise. + * + */ +SNPE_API +int Snpe_IOBufferDataTypeMap_Find(Snpe_IOBufferDataTypeMap_Handle_t handle, const char* name); + +/** + * @brief Resets the map + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IOBufferDataTypeMap_Clear(Snpe_IOBufferDataTypeMap_Handle_t handle); + +/** + * @brief Checks whether the map is empty + * + * @return 1 if the map is empty, 0 otherwise. + * + */ +SNPE_API +int Snpe_IOBufferDataTypeMap_Empty(Snpe_IOBufferDataTypeMap_Handle_t handle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_IOBUFFER_DATATYPE_MAP_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.hpp new file mode 100644 index 00000000..c39d3320 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IOBufferDataTypeMap.hpp @@ -0,0 +1,69 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include + +#include "DlEnums.hpp" + + +#include "DlSystem/IOBufferDataTypeMap.h" + +namespace DlSystem { + +class IOBufferDataTypeMap : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_IOBufferDataTypeMap_Delete}; + +public: + + IOBufferDataTypeMap() + : BaseType(Snpe_IOBufferDataTypeMap_Create()) + { } + + void add(const char* name, IOBufferDataType_t bufferDataType){ + Snpe_IOBufferDataTypeMap_Add(handle(), name, static_cast(bufferDataType)); + } + + void remove(const char* name){ + Snpe_IOBufferDataTypeMap_Remove(handle(), name); + } + + IOBufferDataType_t getBufferDataType(const char* name){ + return static_cast(Snpe_IOBufferDataTypeMap_GetBufferDataType(handle(), name)); + } + + IOBufferDataType_t getBufferDataType(){ + return static_cast(Snpe_IOBufferDataTypeMap_GetBufferDataTypeOfFirst(handle())); + } + + size_t size() const{ + return Snpe_IOBufferDataTypeMap_Size(handle()); + } + + bool find(const char* name) const{ + return Snpe_IOBufferDataTypeMap_Find(handle(), name); + } + + void clear(){ + Snpe_IOBufferDataTypeMap_Clear(handle()); + } + + bool empty() const{ + return Snpe_IOBufferDataTypeMap_Empty(handle()); + } +}; + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, IOBufferDataTypeMap) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.h new file mode 100644 index 00000000..913f3bdc --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.h @@ -0,0 +1,118 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef _DL_SYSTEM_ITENSOR_H_ +#define _DL_SYSTEM_ITENSOR_H_ + +#include + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/TensorShape.h" +#include "DlSystem/DlError.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * Represents a tensor which holds n-dimensional data. It is important to + * understand how the tensor data is represented in memory + * relative to the tensor dimensions. Tensors store data in + * memory in row-major order (i.e. the last tensor dimension is + * the fastest varying one). For example, if you have a two + * dimensional tensor with 3 rows and 2 columns (i.e. the tensor + * dimensions are 3,2 as returned in tensor dimension vectors) + * with the following data in terms rows and columns: + * + * | 1 2 |
+ * | 3 4 |
+ * | 5 6 |
+ * + * This data would be stored in memory as 1,2,3,4,5,6. + */ +typedef void* Snpe_ITensor_Handle_t; + + +/** + * Destroys/frees an ITensor + * + * @param[in] userBufferHandle : Handle to access the IUserBuffer + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_ITensor_Delete(Snpe_ITensor_Handle_t iTensorHandle); + +/** + * Returns a tensor iterator pointing to the beginning + * of the data in the tensor. + * + * @param[in] tensorHandle : Handle to access ITensor + * + * @return The tensor data as a void pointer. + */ +SNPE_API +void* Snpe_ITensor_GetData(Snpe_ITensor_Handle_t tensorHandle); + +/** + * @brief Gets the shape of this tensor. + * + * The last element of the vector represents the fastest varying + * dimension and the zeroth element represents the slowest + * varying dimension, etc. + * + * @param[in] tensorHandle : Handle to access ITensor + * + * @return A TensorShape handle holding the tensor dimensions. + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_ITensor_GetShape(Snpe_ITensor_Handle_t tensorHandle); + +/** + * Returns the element size of the data in the tensor + * (discounting strides). This is how big a buffer would + * need to be to hold the tensor data contiguously in + * memory. + * + * @param[in] tensorHandle : Handle to access ITensor + * + * @return The size of the tensor (in elements). + */ +SNPE_API +size_t Snpe_ITensor_GetSize(Snpe_ITensor_Handle_t tensorHandle); + +SNPE_API +int Snpe_ITensor_IsQuantized(Snpe_ITensor_Handle_t tensorHandle); + +SNPE_API +float Snpe_ITensor_GetDelta(Snpe_ITensor_Handle_t tensorHandle); + +SNPE_API +float Snpe_ITensor_GetOffset(Snpe_ITensor_Handle_t tensorHandle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _DL_SYSTEM_ITENSOR_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.hpp new file mode 100644 index 00000000..4785a39d --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensor.hpp @@ -0,0 +1,95 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "TensorShape.hpp" +#include "ITensorItr.hpp" + +#include "DlSystem/ITensor.h" + + +namespace DlSystem { + + +class ITensor : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_ITensor_Delete}; + + template + T* getData(){ + return static_cast(Snpe_ITensor_GetData(handle())); + } + + template + const T* getData() const{ + return static_cast(Snpe_ITensor_GetData(handle())); + } + +public: + using iterator = DlSystem::ITensorItr; + using const_iterator = DlSystem::ITensorItr; + + + iterator begin(){ + return iterator(getData()); + } + + const_iterator begin() const{ + return const_iterator(getData()); + } + + const_iterator cbegin() const{ + return begin(); + } + + iterator end(){ + return begin() + getSize(); + } + + const_iterator end() const{ + return cbegin() + getSize(); + } + + const_iterator cend() const{ + return end(); + } + + TensorShape getShape() const{ + return moveHandle(Snpe_ITensor_GetShape(handle())); + } + + size_t getSize() const{ + return Snpe_ITensor_GetSize(handle()); + } + + // Serialize to std::ostream is no longer supported + void serialize(std::ostream &output) const = delete; + + bool isQuantized() const{ + return Snpe_ITensor_IsQuantized(handle()); + } + + float GetDelta() const{ + return Snpe_ITensor_GetDelta(handle()); + } + + float GetOffset() const{ + return Snpe_ITensor_GetOffset(handle()); + } +}; + + +} //ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, ITensor) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorFactory.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorFactory.hpp new file mode 100644 index 00000000..5ef1e9d3 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorFactory.hpp @@ -0,0 +1,52 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "ITensor.hpp" + +#include + + +#include "SNPE/SNPEUtil.h" + +namespace DlSystem{ +// NOTE: These factories use a different handle type because they are singletons +// Never copy this pattern unless you're also implementing a singleton +class ITensorFactory : public Wrapper{ + friend BaseType; + + using BaseType::BaseType; + static constexpr DeleteFunctionType DeleteFunction{NoOpDeleter}; + +public: + ITensorFactory() + : BaseType(nullptr) + { } + + + std::unique_ptr createTensor(const TensorShape &shape) noexcept{ + return makeUnique(Snpe_Util_CreateITensor(getHandle(shape))); + } + + // Create from std::istream is no longer supported + std::unique_ptr createTensor(std::istream &input) noexcept = delete; + + std::unique_ptr createTensor(const TensorShape &shape, + const unsigned char *data, + size_t dataSize) noexcept{ + auto handle = Snpe_Util_CreateITensorDataSize(getHandle(shape), data, dataSize); + return makeUnique(handle); + } + +}; + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, ITensorFactory) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItr.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItr.hpp new file mode 100644 index 00000000..801aa217 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItr.hpp @@ -0,0 +1,199 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include +#include +#include + +#include "Wrapper.hpp" +#include "ITensorItrImpl.hpp" + +namespace DlSystem{ + +template +class ITensorItr{ +public: + using iterator_category = std::bidirectional_iterator_tag; + using pointer = typename std::conditional::type; + using value_type = float; + using difference_type = std::ptrdiff_t; + using reference = typename std::conditional::type; + + + ITensorItr() = delete; + virtual ~ITensorItr() = default; + + explicit ITensorItr(pointer data) noexcept + : m_Impl{nullptr}, + m_IsTrivial{true}, + m_Data{data}, + m_DataStart{data} + { } + + ITensorItr(std::unique_ptr impl, + bool isTrivial = false, + float* data = nullptr) + : m_Impl(impl->clone()), + m_IsTrivial(isTrivial), + m_Data(data), + m_DataStart(data) + { } + + ITensorItr(const ITensorItr& itr) + : m_Impl(itr.m_Impl ? itr.m_Impl->clone() : nullptr), + m_IsTrivial(itr.m_IsTrivial), + m_Data(itr.m_Data), + m_DataStart(itr.m_DataStart) + { } + + ITensorItr(ITensorItr&& itr) noexcept + : m_Impl(std::move(itr.m_Impl)), + m_IsTrivial(itr.m_IsTrivial), + m_Data(itr.m_Data), + m_DataStart(itr.m_DataStart) + { } + + ITensorItr& operator=(const ITensorItr& other){ + if (this == &other) return *this; + + m_Impl = other.m_Impl ? other.m_Impl->clone() : nullptr; + m_IsTrivial = other.m_IsTrivial; + m_Data = other.m_Data; + m_DataStart = other.m_DataStart; + return *this; + } + ITensorItr& operator=(ITensorItr&& other) noexcept{ + if(this != &other){ + m_Impl = std::move(other.m_Impl); + m_IsTrivial = other.m_IsTrivial; + m_Data = other.m_Data; + m_DataStart = other.m_DataStart; + } + return *this; + } + + inline ITensorItr& operator++(){ + if (m_IsTrivial){ + m_Data++; + } else { + m_Impl->increment(); + } + return *this; + } + inline ITensorItr operator++(int){ + ITensorItr tmp(*this); + operator++(); + return tmp; + } + inline ITensorItr& operator--(){ + if (m_IsTrivial){ + m_Data--; + } else { + m_Impl->decrement(); + } + return *this; + } + inline ITensorItr operator--(int){ + ITensorItr tmp(*this); + operator--(); + return tmp; + } + inline ITensorItr& operator+=(int rhs){ + if (m_IsTrivial){ + m_Data += rhs; + } else { + m_Impl->increment(rhs); + } + return *this; + } + inline friend ITensorItr operator+(ITensorItr lhs, int rhs){ + lhs += rhs; + return lhs; + } + inline ITensorItr& operator-=(int rhs){ + if (m_IsTrivial){ + m_Data -= rhs; + } else { + m_Impl->decrement(rhs); + } + return *this; + } + inline friend ITensorItr operator-(ITensorItr lhs, int rhs){ + lhs -= rhs; + return lhs; + } + + inline size_t operator-(const ITensorItr& rhs){ + if (m_IsTrivial) return (m_Data - m_DataStart) - (rhs.m_Data - rhs.m_DataStart); + return m_Impl->getPosition() - rhs.m_Impl->getPosition(); + } + + inline friend bool operator<(const ITensorItr& lhs, const ITensorItr& rhs){ + if (lhs.m_IsTrivial) return lhs.m_Data < rhs.m_Data; + return lhs.m_Impl->dataPointer() < rhs.m_Impl->dataPointer(); + } + inline friend bool operator>(const ITensorItr& lhs, const ITensorItr& rhs){ + return rhs < lhs; + } + inline friend bool operator<=(const ITensorItr& lhs, const ITensorItr& rhs){ + return !(lhs > rhs); + } + inline friend bool operator>=(const ITensorItr& lhs, const ITensorItr& rhs){ + return !(lhs < rhs); + } + + inline bool operator==(const ITensorItr& rhs) const{ + if (m_IsTrivial) return m_Data == rhs.m_Data; + return m_Impl->dataPointer() == rhs.m_Impl->dataPointer(); + } + inline bool operator!=(const ITensorItr& rhs) const{ + return !operator==(rhs); + } + + inline reference operator[](size_t idx){ + if (m_IsTrivial) return *(m_DataStart + idx); + return m_Impl->getReferenceAt(idx); + } + inline reference operator*(){ + if (m_IsTrivial) return *m_Data; + return m_Impl->getReference(); + } + inline reference operator->(){ + return *(*this); + } + inline float* dataPointer() const{ + if (m_IsTrivial) return m_Data; + return m_Impl->dataPointer(); + } + + +protected: + std::unique_ptr<::DlSystem::ITensorItrImpl> m_Impl; + bool m_IsTrivial = false; + pointer m_Data = nullptr; + pointer m_DataStart = nullptr; +}; + + +inline void fill(ITensorItr first, ITensorItr end, float val){ + std::fill(first, end, val); +} +template +OutItr copy(InItr first, InItr last, OutItr result){ + return std::copy(first, last, result); +} + +} // ns DlSystem + + +// ALIAS_IN_ZDL_NAMESPACE +namespace zdl{ namespace DlSystem{ + template + using ITensorItr = ::DlSystem::ITensorItr; +}} diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItrImpl.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItrImpl.hpp new file mode 100644 index 00000000..6b9a497b --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/ITensorItrImpl.hpp @@ -0,0 +1,32 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once +#include "Wrapper.hpp" + +namespace DlSystem { + +class ITensorItrImpl { +public: + ITensorItrImpl() = default; + virtual ~ITensorItrImpl() = default; + + virtual float getValue() const = 0; + virtual float& getReference() = 0; + virtual float& getReferenceAt(size_t idx) = 0; + virtual float* dataPointer() const = 0; + virtual void increment(int incVal = 1) = 0; + virtual void decrement(int decVal = 1) = 0; + virtual size_t getPosition() = 0; + virtual std::unique_ptr clone() = 0; + +private: + ITensorItrImpl& operator=(const ITensorItrImpl& other) = delete; + ITensorItrImpl(const ITensorItrImpl& other) = delete; +}; + +} // ns DlSystem diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.h new file mode 100644 index 00000000..fc4cc316 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.h @@ -0,0 +1,714 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _IUSER_BUFFER_H +#define _IUSER_BUFFER_H + +#include +#include + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/TensorShape.h" +#include "DlSystem/DlError.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE UserByfferEncoding handle + */ +typedef void* Snpe_UserBufferEncoding_Handle_t; + +/** + * @brief . + * + * An enum class of all supported element types in a IUserBuffer + */ +//enum class Snpe_UserBufferEncoding_ElementType_t +typedef enum +{ + /// Unknown element type. + SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNKNOWN = 0, + + /// Each element is presented by float. + SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT = 1, + + /// Each element is presented by an unsigned int. + SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNSIGNED8BIT = 2, + + /// Each element is presented by float16. + SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT16 = 3, + + /// Each element is presented by an 8-bit quantized value. + SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF8 = 10, + + /// Each element is presented by an 16-bit quantized value. + SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF16 = 11, + + /// Each element is presented by Int32 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT32 = 12, + + /// Each element is presented by UInt32 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT32 = 13, + + /// Each element is presented by Int8 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT8 = 14, + + /// Each element is presented by UInt8 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT8 = 15, + + /// Each element is presented by Int16 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT16 = 16, + + /// Each element is presented by UInt16 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT16 = 17, + + /// Each element is present by Bool8 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_BOOL8 = 18, + + /// Each element is present by Int64 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT64 = 19, + + /// Each element is present by UInt64 + SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT64 = 20 + +}Snpe_UserBufferEncoding_ElementType_t; + + +/** + * @brief Retrieves the element type + * + * @param[in] userBufferEncodingHandle : Handle to access userBufferEncoding + * + * @return Element type + */ +SNPE_API +Snpe_UserBufferEncoding_ElementType_t Snpe_UserBufferEncoding_GetElementType(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access userBufferEncoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncoding_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Destroys/frees a UserBufferEncoding + * + * @param[in] userBufferEncodingHandle : Handle to access UserBufferEncoding + * + * @return indication of success/failures + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncoding_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + + +/** + * @brief . + * + * A base class buffer source type + * + * @note User buffer from CPU support all kinds of runtimes; + * User buffer from GLBUFFER support only GPU runtime. + */ +typedef void* Snpe_UserBufferSource_Handle_t; + +typedef enum +{ + /// Unknown buffer source type. + SNPE_USERBUFFERSOURCE_SOURCETYPE_UNKNOWN = 0, + + /// The network inputs are from CPU buffer. + SNPE_USERBUFFERSOURCE_SOURCETYPE_CPU = 1, + + /// The network inputs are from OpenGL buffer. + SNPE_USERBUFFERSOURCE_SOURCETYPE_GLBUFFER = 2 +}Snpe_UserBufferSource_SourceType_t; + +/** + * @brief Retrieves the source type + * + * @param[in] userBufferSourceHandle : Handle to access userBufferSource + * + * @return Source type + */ +SNPE_API +Snpe_UserBufferSource_SourceType_t Snpe_UserBufferSource_GetSourceType(Snpe_UserBufferSource_Handle_t userBufferSourceHandle); + +/** + * @brief Destroys/frees a UserBufferSource + * + * @param[in] userBufferSourceHandle : Handle to access UserBufferSource + * + * @return indication of success/failures + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferSource_Delete(Snpe_UserBufferSource_Handle_t userBufferSourceHandle); + +/** + * @brief . + * + * An source type where input data is delivered from OpenGL buffer + */ +SNPE_API +Snpe_UserBufferSource_Handle_t Snpe_UserBufferSourceGLBuffer_Create(); + +/** + * @brief Destroys the userBuffer + * + * @param[in] userBufferSourceHandle : Handle to access the UserBuffer + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferSourceGLBuffer_Delete(Snpe_UserBufferSource_Handle_t userBufferSourceHandle); + +// Encoding 8 Bit +/** + * @brief . + * + * An encoding type where each element is represented by an unsigned int. + * + * Userbuffer size assumes uint8 encoding for each element. + * (i.e., a tensor with dimensions (2,3) will be represented by (2 * 3) * 1 = 6 bytes in memory). + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingUnsigned8Bit_Create(); + +/** + * @brief Copy Constructor for UserBufferEncodingUnsigned8Bit + * + * An encoding type where each element is represented by an unsigned int. + * + * Userbuffer size assumes uint8 encoding for each element. + * (i.e., a tensor with dimensions (2,3) will be represented by (2 * 3) * 1 = 6 bytes in memory). + * + * @param[in] otherHandle : a handle to another UserBufferEncodingUnsigned8Bit to copy + * + * @return a handle to the UserBufferEncodingUnsigned8Bit + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingUnsigned8Bit_CreateCopy(Snpe_UserBufferEncoding_Handle_t otherHandle); + +/** + * @brief Destroys the encodingUnsigned8Bit + * + * @param[in] userBufferEncodingHandle : Handle to access the encodingUnsigned8Bit + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingUnsigned8Bit_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingUnsigned8Bit_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + + +// Encoding Float +/** + * @brief . + * + * An encoding type where each element is represented by a float. + * + * Userbuffer size assumes float encoding for each element. + * (i.e., a tensor with dimensions (2,3) will be represented by (2 * 3) * 4 = 24 bytes in memory). + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingFloat_Create(); + +/** + * @brief Copy Constructor for UserBufferEncodingFloat + * + * An encoding type where each element is represented by a float. + * + * Userbuffer size assumes float encoding for each element. + * (i.e., a tensor with dimensions (2,3) will be represented by (2 * 3) * 4 = 24 bytes in memory). + * + * @param[in] otherHandle : a handle to another UserBufferEncodingFloat to copy + * + * @return a handle to the constructed UserBufferEncodingFloat + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingFloat_CreateCopy(Snpe_UserBufferEncoding_Handle_t otherHandle); + +/** + * @brief Destroys the encodingFloat + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingFloat_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingFloat_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +// Encoding FloatN +/** + * @brief . + * + * An encoding type where each element is represented by a float N + * + * Userbuffer size assumes float N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 16 will be represented by (2 * 3) * 2 = 12 bytes in memory). + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingFloatN_Create(uint8_t bWidth); + +/** + * @brief Copy Constructor for UserBufferEncodingFloatN + * + * An encoding type where each element is represented by a float N + * + * Userbuffer size assumes float N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 16 will be represented by (2 * 3) * 2 = 12 bytes in memory). + * + * @param[in] otherHandle : a handle to another UserBufferEncodingFloatN to copy + * + * @return a handle to the constructed UserBufferEncodingFloatN + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingFloatN_CreateCopy(Snpe_UserBufferEncoding_Handle_t otherHandle); + + +/** + * @brief Destroys the encodingFloatN + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingFloatN_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingFloatN_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + + +/** + * @brief Get the Float type corresponding to a given bitwidth + * + * @param width bitwidth of Float type + * + * @return ElementType corresponding to a Float of width bits + */ +SNPE_API +Snpe_UserBufferEncoding_ElementType_t Snpe_UserBufferEncodingFloatN_GetTypeFromWidth(uint8_t width); + +/** + * @brief . + * + * An encoding type where each element is represented by tfN, which is an + * N-bit quantized value, which has an exact representation of 0.0 + * + * Userbuffer size assumes tf N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 16 will be represented by (2 * 3) * 2 = 12 bytes in memory). + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingTfN_Create(uint64_t stepFor0, float stepSize, uint8_t bWidth); + +/** + * @brief Copy Constructor for UserBufferEncodingTfN + * + * An encoding type where each element is represented by tfN, which is an + * N-bit quantized value, which has an exact representation of 0.0 + * + * Userbuffer size assumes tf N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 16 will be represented by (2 * 3) * 2 = 12 bytes in memory). + * @param otherHandle the UserBufferEncodingTfN to copy + * @return a handle to a newly constructed UserBufferEncodingTfN + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingTfN_CreateCopy(Snpe_UserBufferEncoding_Handle_t otherHandle); + +/** + * @brief Destroys the encodingTfN + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingTfN_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingTfN_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Sets the step value that represents 0 + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @param[in] stepExactly0 : The step value that represents 0 + * + */ +SNPE_API +void Snpe_UserBufferEncodingTfN_SetStepExactly0(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle, uint64_t stepExactly0); + +/** + * @brief Sets the float value that each step represents + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @param[in] quantizedStepSize : The float value of each step size + * + */ +SNPE_API +void Snpe_UserBufferEncodingTfN_SetQuantizedStepSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle, float quantizedStepSize); + +/** + * @brief Retrieves the step that represents 0.0 + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Step value + */ +SNPE_API +uint64_t Snpe_UserBufferEncodingTfN_GetStepExactly0(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the step size + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Step size + */ +SNPE_API +float Snpe_UserBufferEncodingTfN_GetQuantizedStepSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * Calculates the minimum floating point value that + * can be represented with this encoding. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Minimum representable floating point value + */ +SNPE_API +float Snpe_UserBufferEncodingTfN_GetMin(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * Calculates the maximum floating point value that + * can be represented with this encoding. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Maximum representable floating point value + */ +SNPE_API +float Snpe_UserBufferEncodingTfN_GetMax(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Get the tfN type corresponding to a given bitwidth + * + * @param width bitwidth of tfN type + * + * @return ElementType corresponding to a tfN of width bits + */ +SNPE_API +Snpe_UserBufferEncoding_ElementType_t Snpe_UserBufferEncodingTfN_GetTypeFromWidth(uint8_t width); + +// Encoding Int N +/** + * @brief . + * + * An encoding type where each element is represented by a Int + * + * Userbuffer size assumes int N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 32 will be represented by (2 * 3) * 4 = 24 bytes in memory). + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingIntN_Create(uint8_t bWidth); + +/** + * @brief Copy Constructor for UserBufferEncodingIntN + * + * An encoding type where each element is represented by a Int + * + * Userbuffer size assumes int N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 32 will be represented by (2 * 3) * 4 = 24 bytes in memory). + * @param otherHandle the UserBufferEncodingIntN to copy + * @return a handle to a newly constructed UserBufferEncodingIntN + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingIntN_CreateCopy(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Destroys the encodingIntN + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingIntN_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingIntN_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Get the int type corresponding to a given bitwidth + * + * @param width bitwidth of int type + * + * @return ElementType corresponding to a int of width bits + */ +SNPE_API +Snpe_UserBufferEncoding_ElementType_t Snpe_UserBufferEncodingIntN_GetTypeFromWidth(uint8_t bWidth); + +// Encoding Uint N +/** + * @brief . + * + * An encoding type where each element is represented by a Uint + * + * Userbuffer size assumes uint N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 32 will be represented by (2 * 3) * 4 = 24 bytes in memory). + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingUintN_Create(uint8_t bWidth); + +/** + * @brief Copy Constructor for UserBufferEncodingUintN + * + * An encoding type where each element is represented by a Uint + * + * Userbuffer size assumes uint N encoding for each element. + * (i.e., a tensor with dimensions (2,3) with a provided bitwidth of 32 will be represented by (2 * 3) * 4 = 24 bytes in memory). + * @param otherHandle the UserBufferEncodingUintN to copy + * @return a handle to a newly constructed UserBufferEncodingUintN + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingUintN_CreateCopy(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Destroys the encodingUintN + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingUintN_Delete(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferEncodingHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingUintN_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Get the uint type corresponding to a given bitwidth + * + * @param width bitwidth of uint type + * + * @return ElementType corresponding to a uint of width bits + */ +SNPE_API +Snpe_UserBufferEncoding_ElementType_t Snpe_UserBufferEncodingUintN_GetTypeFromWidth(uint8_t bWidth); + + +// Encoding Bool +/** + * @brief . + * + * An encoding type where each element is represented by a Bool + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingBool_Create(uint8_t bWidth); + +/** + * @brief Copy Constructor for UserBufferEncodingBool + * + * An encoding type where each element is represented by a bool + * + * @param otherHandle the UserBufferEncodingBool to copy + * @return a handle to a newly constructed UserBufferEncodingBool + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_UserBufferEncodingBool_CreateCopy(Snpe_UserBufferEncoding_Handle_t userBufferEncodingHandle); + +/** + * @brief Destroys the encodingBool + * + * @param[in] userBufferHandle : Handle to access the encoding + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferEncodingBool_Delete(Snpe_UserBufferEncoding_Handle_t userBufferHandle); + +/** + * @brief Retrieves the size of the element, in bytes. + * + * @param[in] userBufferHandle : Handle to access the encoding + * + * @return Size of the element, in bytes. + */ +SNPE_API +size_t Snpe_UserBufferEncodingBool_GetElementSize(Snpe_UserBufferEncoding_Handle_t userBufferHandle); + + + +/** + * A typedef to indicate a SNPE IUserBuffer handle + * UserBuffer contains a pointer and info on how to walk it and interpret its content. + */ +typedef void* Snpe_IUserBuffer_Handle_t; + +/** + * Destroys/frees an IUserBuffer + * + * @param[in] userBufferHandle : Handle to access the IUserBuffer + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_IUserBuffer_Delete(Snpe_IUserBuffer_Handle_t userBufferHandle); + + +/** + * @brief Retrieves the total number of bytes between elements in each dimension if + * the buffer were to be interpreted as a multi-dimensional array. + * + * @param[in] userBufferHandle : Handle to access the user Buffer + * + * @warning Do not modify the TensorShape returned by reference. Treat it as a const reference. + * + * @return A const reference to the number of bytes between elements in each dimension. + * e.g. A tightly packed tensor of floats with dimensions [4, 3, 2] would + * return strides of [24, 8, 4]. + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_IUserBuffer_GetStrides_Ref(Snpe_IUserBuffer_Handle_t userBufferHandle); + +/** + * @brief Retrieves the size of the buffer, in bytes. + * + * @param[in] userBufferHandle : Handle to access the user Buffer + * + * @return Size of the underlying buffer, in bytes. + */ +SNPE_API +size_t Snpe_IUserBuffer_GetSize(Snpe_IUserBuffer_Handle_t userBufferHandle); + +/** + * @brief Retrieves the size of the inference data in the buffer, in bytes. + * + * The inference results from a dynamic-sized model may not be exactly the same size + * as the UserBuffer provided to SNPE. This function can be used to get the amount + * of output inference data, which may be less or greater than the size of the UserBuffer. + * + * If the inference results fit in the UserBuffer, getOutputSize() would be less than + * or equal to getSize(). But if the inference results were more than the capacity of + * the provided UserBuffer, the results would be truncated to fit the UserBuffer. But, + * getOutputSize() would be greater than getSize(), which indicates a bigger buffer + * needs to be provided to SNPE to hold all of the inference results. + * + * @param[in] userBufferHandle : Handle to access the user Buffer + * + * @return Size required for the buffer to hold all inference results, which can be less + * or more than the size of the buffer, in bytes. + */ +SNPE_API +size_t Snpe_IUserBuffer_GetOutputSize(Snpe_IUserBuffer_Handle_t userBufferHandle); + +/** + * @brief Changes the underlying memory that backs the UserBuffer. + * + * This can be used to avoid creating multiple UserBuffer objects + * when the only thing that differs is the memory location. + * + * @param[in] userBufferHandle : Handle to access the user Buffer + * + * @param[in] buffer : Pointer to the memory location + * + * @return Whether the set succeeds. + */ +SNPE_API +int Snpe_IUserBuffer_SetBufferAddress(Snpe_IUserBuffer_Handle_t userBufferHandle, void* buffer); + +/** + * @brief Gets a reference to the data encoding object of + * the underlying buffer + * + * This is necessary when the UserBuffer is re-used, and the encoding + * parameters can change. For example, each input can be quantized with + * different step sizes. + * + * @param[in] userBufferHandle : Handle to access the user Buffer + * + * @return Data encoding meta-data + */ +SNPE_API +Snpe_UserBufferEncoding_Handle_t Snpe_IUserBuffer_GetEncoding_Ref(Snpe_IUserBuffer_Handle_t userBufferHandle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _IUSER_BUFFER_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.hpp new file mode 100644 index 00000000..727c195b --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBuffer.hpp @@ -0,0 +1,390 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include +#include "TensorShape.hpp" + +#include "DlSystem/IUserBuffer.h" + + +namespace DlSystem { + + +class UserBufferEncoding: public Wrapper { + friend BaseType; + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserBufferEncoding_Delete}; +protected: + UserBufferEncoding(HandleType handle) + : BaseType(handle) + { } +public: + + virtual ~UserBufferEncoding() = default; + + UserBufferEncoding(UserBufferEncoding&& other) noexcept + : BaseType(std::move(other)) + { } + + enum class ElementType_t + { + /// Unknown element type. + UNKNOWN = 0, + + /// Each element is presented by 32-bit float. + FLOAT = 1, + + /// Each element is presented by an unsigned int. + UNSIGNED8BIT = 2, + + /// Each element is presented by 16-bit float. + FLOAT16 = 3, + + /// Each element is presented by an 8-bit quantized value. + TF8 = 10, + + /// Each element is presented by an 16-bit quantized value. + TF16 = 11, + + /// Each element is presented by Int32 + INT32 = 12, + + /// Each element is presented by UInt32 + UINT32 = 13, + + /// Each element is presented by Int8 + INT8 = 14, + + /// Each element is presented by UInt8 + UINT8 = 15, + + /// Each element is presented by Int16 + INT16 = 16, + + /// Each element is presented by UInt16 + UINT16 = 17, + + // Each element is presented by Bool8 + BOOL8 = 18, + + // Each element is presented by Int64 + INT64 = 19, + + // Each element is presented by UInt64 + UINT64 = 20 + }; + + ElementType_t getElementType() const noexcept{ + return static_cast(Snpe_UserBufferEncoding_GetElementType(handle())); + } + + size_t getElementSize() const noexcept{ + return Snpe_UserBufferEncoding_GetElementSize(handle()); + } +}; + + +class UserBufferSource: public Wrapper { + friend BaseType; + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserBufferSource_Delete}; + +public: + enum class SourceType_t + { + /// Unknown buffer source type. + UNKNOWN = 0, + + /// The network inputs are from CPU buffer. + CPU = 1, + + /// The network inputs are from OpenGL buffer. + GLBUFFER = 2 + }; +protected: + UserBufferSource(HandleType handle) + : BaseType(handle) + { } +public: + SourceType_t getSourceType() const noexcept{ + return static_cast(Snpe_UserBufferSource_GetSourceType(handle())); + } + +}; + +class UserBufferSourceGLBuffer : public UserBufferSource{ +public: + UserBufferSourceGLBuffer() + : UserBufferSource(Snpe_UserBufferSourceGLBuffer_Create()) + { } +}; + +class UserBufferEncodingUnsigned8Bit : public UserBufferEncoding{ +public: + using UserBufferEncoding::UserBufferEncoding; + UserBufferEncodingUnsigned8Bit() + : UserBufferEncoding(Snpe_UserBufferEncodingUnsigned8Bit_Create()) + { } +}; + +class UserBufferEncodingFloatN : public UserBufferEncoding{ +public: + using UserBufferEncoding::UserBufferEncoding; + + UserBufferEncodingFloatN(uint8_t bWidth=32) + : UserBufferEncoding(Snpe_UserBufferEncodingFloatN_Create(bWidth)) + { } + + UserBufferEncodingFloatN(const UserBufferEncodingFloatN& other) + : UserBufferEncoding(Snpe_UserBufferEncodingFloatN_CreateCopy(other.handle())) + { } + + static ElementType_t getTypeFromWidth(uint8_t width){ + return static_cast(Snpe_UserBufferEncodingFloatN_GetTypeFromWidth(width)); + } +}; + +class UserBufferEncodingFloat : public UserBufferEncoding{ +public: + using UserBufferEncoding::UserBufferEncoding; + UserBufferEncodingFloat() + : UserBufferEncoding(Snpe_UserBufferEncodingFloat_Create()) + { } + UserBufferEncodingFloat(const UserBufferEncodingFloat& other) + : UserBufferEncoding(Snpe_UserBufferEncodingFloat_CreateCopy(other.handle())) + { } + + UserBufferEncodingFloat(UserBufferEncodingFloat&& other) noexcept + : UserBufferEncoding(std::move(other)) + { } +}; + + +class UserBufferEncodingTfN : public UserBufferEncoding{ +public: + + using UserBufferEncoding::UserBufferEncoding; + template::value && std::is_floating_point::value, int>::type = 0> + UserBufferEncodingTfN(T stepFor0, U stepSize, uint8_t bWidth=8) + : UserBufferEncoding(Snpe_UserBufferEncodingTfN_Create(stepFor0, stepSize, bWidth)) + { } + + UserBufferEncodingTfN(const UserBufferEncoding& ubEncoding) + : UserBufferEncoding(Snpe_UserBufferEncodingTfN_CreateCopy(getHandle(ubEncoding))) + { } + UserBufferEncodingTfN(const UserBufferEncodingTfN& ubEncoding) + : UserBufferEncoding(Snpe_UserBufferEncodingTfN_CreateCopy(getHandle(ubEncoding))) + { } + + void setStepExactly0(uint64_t stepExactly0){ + Snpe_UserBufferEncodingTfN_SetStepExactly0(handle(), stepExactly0); + } + + void setQuantizedStepSize(const float quantizedStepSize){ + Snpe_UserBufferEncodingTfN_SetQuantizedStepSize(handle(), quantizedStepSize); + } + + uint64_t getStepExactly0() const{ + return Snpe_UserBufferEncodingTfN_GetStepExactly0(handle()); + } + + float getMin() const{ + return Snpe_UserBufferEncodingTfN_GetMin(handle()); + } + float getMax() const{ + return Snpe_UserBufferEncodingTfN_GetMax(handle()); + } + + float getQuantizedStepSize() const{ + return Snpe_UserBufferEncodingTfN_GetQuantizedStepSize(handle()); + } + + static ElementType_t getTypeFromWidth(uint8_t width){ + return static_cast(Snpe_UserBufferEncodingTfN_GetTypeFromWidth(width)); + } +}; + +class UserBufferEncodingIntN : public UserBufferEncoding{ +public: + + UserBufferEncodingIntN(uint8_t bWidth=32) + : UserBufferEncoding(Snpe_UserBufferEncodingIntN_Create(bWidth)) + { } + + UserBufferEncodingIntN(const UserBufferEncoding& ubEncoding) + : UserBufferEncoding(Snpe_UserBufferEncodingIntN_CreateCopy(getHandle(ubEncoding))) + { } + + static ElementType_t getTypeFromWidth(uint8_t width){ + return static_cast(Snpe_UserBufferEncodingIntN_GetTypeFromWidth(width)); + } +}; + + + +class UserBufferEncodingUintN : public UserBufferEncoding{ +public: + + UserBufferEncodingUintN(uint8_t bWidth=32) + : UserBufferEncoding(Snpe_UserBufferEncodingUintN_Create(bWidth)) + { } + + UserBufferEncodingUintN(const UserBufferEncoding& ubEncoding) + : UserBufferEncoding(Snpe_UserBufferEncodingUintN_CreateCopy(getHandle(ubEncoding))) + { } + + static ElementType_t getTypeFromWidth(uint8_t width){ + return static_cast(Snpe_UserBufferEncodingUintN_GetTypeFromWidth(width)); + } +}; + + +class UserBufferEncodingTf8 : public UserBufferEncodingTfN{ +public: + using UserBufferEncodingTfN::UserBufferEncodingTfN; + UserBufferEncodingTf8() = delete; + + template::value && std::is_floating_point::value, int>::type = 0> + UserBufferEncodingTf8(T stepFor0, U stepSize) + : UserBufferEncodingTfN(stepFor0, stepSize, 8) + { } + + UserBufferEncodingTf8(const UserBufferEncoding& ubEncoding) + : UserBufferEncodingTfN(ubEncoding) + { } + +}; + +class UserBufferEncodingBool : public UserBufferEncoding{ +public: + UserBufferEncodingBool(uint8_t bWidth=8) + : UserBufferEncoding(Snpe_UserBufferEncodingBool_Create(bWidth)) + { } + + UserBufferEncodingBool(const UserBufferEncoding& ubEncoding) + : UserBufferEncoding(Snpe_UserBufferEncodingBool_CreateCopy(getHandle(ubEncoding))) + { } +}; + +class IUserBuffer: public Wrapper { + friend BaseType; + using BaseType::BaseType; + static constexpr DeleteFunctionType DeleteFunction{Snpe_IUserBuffer_Delete}; + +public: + const TensorShape& getStrides() const{ + return *makeReference(Snpe_IUserBuffer_GetStrides_Ref(handle())); + } + + size_t getSize() const{ + return Snpe_IUserBuffer_GetSize(handle()); + } + + size_t getOutputSize() const{ + return Snpe_IUserBuffer_GetOutputSize(handle()); + } + + bool setBufferAddress(void* buffer) noexcept{ + return Snpe_IUserBuffer_SetBufferAddress(handle(), buffer); + } + + const UserBufferEncoding& getEncoding() const noexcept{ + auto h = Snpe_IUserBuffer_GetEncoding_Ref(handle()); + switch(Snpe_UserBufferEncoding_GetElementType(h)){ + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNSIGNED8BIT: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT8: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT16: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT32: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT8: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT16: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT32: + return *makeReference(h); + + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT16: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF8: + return *makeReference(h); + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF16: + return *makeReference(h); + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_BOOL8: + return *makeReference(h); + + default: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNKNOWN: + return *makeReference(h); + } + } + UserBufferEncoding& getEncoding() noexcept{ + auto h = Snpe_IUserBuffer_GetEncoding_Ref(handle()); + switch(Snpe_UserBufferEncoding_GetElementType(h)){ + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNSIGNED8BIT: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT8: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT16: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UINT32: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT8: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT16: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_INT32: + return *makeReference(h); + + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_FLOAT16: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF8: + return *makeReference(h); + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_TF16: + return *makeReference(h); + + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_BOOL8: + return *makeReference(h); + + default: + case SNPE_USERBUFFERENCODING_ELEMENTTYPE_UNKNOWN: + return *makeReference(h); + } + } + +}; + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncoding) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferSource) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferSourceGLBuffer) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingUnsigned8Bit) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingFloatN) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingFloat) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingTfN) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingIntN) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingUintN) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferEncodingTf8) + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, IUserBuffer) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBufferFactory.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBufferFactory.hpp new file mode 100644 index 00000000..b3bbb087 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/IUserBufferFactory.hpp @@ -0,0 +1,68 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include + +#include "Wrapper.hpp" +#include "IUserBuffer.hpp" +#include "TensorShape.hpp" + + +#include "SNPE/SNPEUtil.h" + +namespace DlSystem{ + + +// NOTE: These factories use a different handle type because they are singletons +// Never copy this pattern unless you're also implementing a singleton +class IUserBufferFactory : public Wrapper{ + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{NoOpDeleter}; + +public: + IUserBufferFactory() + : BaseType(nullptr) + { } + + std::unique_ptr createUserBuffer(void *buffer, + size_t bufSize, + const TensorShape &strides, + UserBufferEncoding* userBufferEncoding) noexcept{ + if(!userBufferEncoding) return {}; + auto handle = Snpe_Util_CreateUserBuffer(buffer, + bufSize, + getHandle(strides), + getHandle(userBufferEncoding)); + return makeUnique(handle); + } + + std::unique_ptr createUserBuffer(void *buffer, + size_t bufSize, + const TensorShape &strides, + UserBufferEncoding* userBufferEncoding, + UserBufferSource* userBufferSource) noexcept{ + if(!userBufferEncoding || !userBufferSource) return {}; + auto handle = Snpe_Util_CreateUserBufferFromSource(buffer, + bufSize, + getHandle(strides), + getHandle(*userBufferEncoding), + getHandle(*userBufferSource)); + return makeUnique(handle); + } + +}; + + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, IUserBufferFactory) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.h new file mode 100644 index 00000000..15b2a089 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.h @@ -0,0 +1,329 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_PLATFORMCONFIG_H +#define DL_SYSTEM_PLATFORMCONFIG_H + +#include "DlSystem/DlError.h" +#include "DlSystem/DlEnums.h" +#include "DlSystem/SnpeApiExportDefine.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * @brief . + * + * A structure OpenGL configuration + * + * @note When certain OpenGL context and display are provided to UserGLConfig for using + * GPU buffer as input directly, the user MUST ensure the particular OpenGL + * context and display remain vaild throughout the execution of neural network models. + */ +typedef void* Snpe_UserGLConfig_Handle_t; + +/** + * @brief . + * + * Creates a new userGLConfig + * + */ +SNPE_API +Snpe_UserGLConfig_Handle_t Snpe_UserGLConfig_Create(); + +/** + * @brief Destroys the userGLConfig + * + * @param[in] handle : Handle to access the userGLConfig + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserGLConfig_Delete(Snpe_UserGLConfig_Handle_t handle); + +/** + * @brief Sets the EGL context + * + * @param[in] handle : Handle to access userGLConfig + * + * @param[in] userGLContext : void pointer + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserGLConfig_SetUserGLContext(Snpe_UserGLConfig_Handle_t handle, void* userGLContext); + +/** + * @brief Sets the EGL Display + * + * @param[in] handle : Handle to access userGLConfig + * + * @param[in] userGLDisplay : void pointer + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserGLConfig_SetUserGLDisplay(Snpe_UserGLConfig_Handle_t handle, void* userGLDisplay); + + +/** + * @brief Get EGL context + * + * @param[in] handle : Handle to access userGLConfig + * + * @return userGLContext of type void pointer + * + */ +SNPE_API +void* Snpe_UserGLConfig_GetUserGLContext(Snpe_UserGLConfig_Handle_t handle); + +/** + * @brief Get EGL Display + * + * @param[in] handle : Handle to access userGLConfig + * + * @return userGLDisplay of type void pointer + * + */ +SNPE_API +void* Snpe_UserGLConfig_GetUserGLDisplay(Snpe_UserGLConfig_Handle_t handle); + + +/** + * @brief . + * + * A structure Gpu configuration + */ +typedef void* Snpe_UserGpuConfig_Handle_t; + +/** + * @brief . + * + * Creates a new userGpuConfig + * + */ +SNPE_API +Snpe_UserGpuConfig_Handle_t Snpe_UserGpuConfig_Create(); + +/** + * @brief Destroys the userGpuConfig + * + * @param[in] handle : Handle to access userGLConfig + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserGpuConfig_Delete(Snpe_UserGpuConfig_Handle_t handle); + +/** + * @brief Set the userGpuConfig + * + * @param[in] handle : Handle to access userGpuConfig + * + * @param[in] glHandle : Handle needed to access userGlConfig + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +void Snpe_UserGpuConfig_Set(Snpe_UserGpuConfig_Handle_t handle, Snpe_UserGLConfig_Handle_t glHandle); + +/** + * @brief Get the userGpuConfig + * + * @param[in] handle : Handle to access userGpuConfig + * + * @return Handle needed to access userGlConfig + */ +SNPE_API +Snpe_UserGLConfig_Handle_t Snpe_UserGpuConfig_Get_Ref(Snpe_UserGpuConfig_Handle_t handle); + + + +/** + * A typedef to indicate a SNPE PlatformConfig handle + */ +typedef void* Snpe_PlatformConfig_Handle_t; + + +/** + * @brief . + * + * Creates a new PlatformConfig + * + */ +SNPE_API +Snpe_PlatformConfig_Handle_t Snpe_PlatformConfig_Create(); + + +/** + * @brief Copy-Construct a PlatformConfig from another PlatformConfig + * + * @param[in] otherHandle Handle to the other PlatformConfig + * + * @return Handle to the Copy-Constructed PlatformConfig + */ +SNPE_API +Snpe_PlatformConfig_Handle_t Snpe_PlatformConfig_CreateCopy(Snpe_PlatformConfig_Handle_t otherHandle); + +/** + * @brief Destroys the PlatformConfig + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PlatformConfig_Delete(Snpe_PlatformConfig_Handle_t handle); + + +typedef enum +{ + /// Unknown platform type. + SNPE_PLATFORMCONFIG_PLATFORMTYPE_UNKNOWN = 0, + + /// Snapdragon CPU. + SNPE_PLATFORMCONFIG_PLATFORMTYPE_CPU = 1, + + /// Adreno GPU. + SNPE_PLATFORMCONFIG_PLATFORMTYPE_GPU = 2, + + /// Hexagon DSP. + SNPE_PLATFORMCONFIG_PLATFORMTYPE_DSP = 3 +} Snpe_PlatformConfig_PlatformType_t; + + +/** + * @brief Retrieves the platform type + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @return Platform type + */ +SNPE_API +Snpe_PlatformConfig_PlatformType_t Snpe_PlatformConfig_GetPlatformType(Snpe_PlatformConfig_Handle_t handle); + +/** + * @brief Indicates whther the plaform configuration is valid. + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @return 1 if the platform configuration is valid; 0 otherwise. + */ +SNPE_API +int Snpe_PlatformConfig_IsValid(Snpe_PlatformConfig_Handle_t handle); + +/** + * @brief Retrieves the Gpu configuration + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @return userGpuConfig populated with the Gpu configuration. + * + */ +SNPE_API +Snpe_UserGpuConfig_Handle_t Snpe_PlatformConfig_GetUserGpuConfig(Snpe_PlatformConfig_Handle_t handle); + +/** + * @brief Sets the Gpu configuration + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @param[in] gpuHandle : Gpu Configuration handle + * + * @return 1 if Gpu configuration was successfully set; 0 otherwise. + */ +SNPE_API +int Snpe_PlatformConfig_SetUserGpuConfig(Snpe_PlatformConfig_Handle_t handle, Snpe_UserGpuConfig_Handle_t gpuHandle); + +/** + * @brief Sets the platform options + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @param[in] options : Options as a const char* in the form of "keyword:options" + * + * @return 1 if options are pass validation; otherwise 0. If false, the options are not updated. + */ +SNPE_API +int Snpe_PlatformConfig_SetPlatformOptions(Snpe_PlatformConfig_Handle_t handle, const char* options); + +/** + * @brief Indicates whther the plaform configuration is valid. + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @return 1 if the platform configuration is valid; 0 otherwise. + */ +SNPE_API +int Snpe_PlatformConfig_IsOptionsValid(Snpe_PlatformConfig_Handle_t handle); + +/** + * @brief Gets the platform options + * + * @param[in] handle : Handle needed to access the platformConfig + * + * @return Options as a const char* + */ +SNPE_API +const char* Snpe_PlatformConfig_GetPlatformOptions(Snpe_PlatformConfig_Handle_t handle); + +/** + * @brief Sets the platform options + * + * @note the returned string will be invalidated by subsequent calls to this function + * + * @param[in] handle : Handle needed to access the platformConfig + * @param[in] optionName : Name of platform options" + * @param[in] value : Value of specified optionName + * + * @return If 1, add "optionName:value" to platform options if optionName don't exist, otherwise update the + * value of specified optionName. + * If 0, the platform options will not be changed. + */ +SNPE_API +int Snpe_PlatformConfig_SetPlatformOptionValue(Snpe_PlatformConfig_Handle_t handle, const char* optionName, const char* value); + +/** + * @brief Removes the platform options + * + * @param[in] handle : Handle needed to access the platformConfig + * @param[in] optionName : Name of platform options" + * @param[in] value : Value of specified optionName + * + * @return If 1, removed "optionName:value" to platform options if optionName don't exist, do nothing. + * If 0, the platform options will not be changed. + */ +SNPE_API +int Snpe_PlatformConfig_RemovePlatformOptionValue(Snpe_PlatformConfig_Handle_t handle, const char* optionName, const char* value); + +SNPE_API +void Snpe_PlatformConfig_SetIsUserGLBuffer(int isUserGLBuffer); + +SNPE_API +int Snpe_PlatformConfig_GetIsUserGLBuffer(); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_PLATFORMCONFIG_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.hpp new file mode 100644 index 00000000..5995c51b --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/PlatformConfig.hpp @@ -0,0 +1,265 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include "DlSystem/PlatformConfig.h" + +namespace DlSystem { + +struct UserGLConfig +{ + /// Holds user EGL context. + /// + void* userGLContext = nullptr; + + /// Holds user EGL display. + void* userGLDisplay = nullptr; +}; + +struct UserGpuConfig{ + /// Holds user OpenGL configuration. + /// + UserGLConfig userGLConfig; +}; + +class PlatformConfig : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_PlatformConfig_Delete}; + + class UserGLConfigInternal : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserGLConfig_Delete}; + + public: + UserGLConfigInternal() + : BaseType(Snpe_UserGLConfig_Create()) + { } + UserGLConfigInternal(const UserGLConfig& uglc) + : UserGLConfigInternal() + { + setUserGLContext(uglc.userGLContext); + setUserGLDisplay(uglc.userGLDisplay); + } + void setUserGLContext(void* userGLContext){ + Snpe_UserGLConfig_SetUserGLContext(handle(), userGLContext); + } + void setUserGLDisplay(void* userGLDisplay){ + Snpe_UserGLConfig_SetUserGLDisplay(handle(), userGLDisplay); + } + + void* getUserGLContext(){ + return Snpe_UserGLConfig_GetUserGLContext(handle()); + } + void* getUserGLDisplay(){ + return Snpe_UserGLConfig_GetUserGLDisplay(handle()); + } + }; + + + + class UserGpuConfigInternal : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserGpuConfig_Delete}; + + public: + UserGpuConfigInternal() + : BaseType(Snpe_UserGpuConfig_Create()) + { } + + void set(const UserGLConfig& userGLConfig){ + UserGLConfigInternal uglc(userGLConfig); + Snpe_UserGpuConfig_Set(handle(), getHandle(uglc)); + } + + void get(UserGLConfig& uglc){ + UserGLConfigInternal uglci(moveHandle(Snpe_UserGpuConfig_Get_Ref(handle()), true)); + + uglc.userGLContext = uglci.getUserGLContext(); + uglc.userGLDisplay = uglci.getUserGLDisplay(); + } + + }; +public: + + /** + * @brief . + * + * An enum class of all supported platform types + */ + enum class PlatformType_t + { + /// Unknown platform type. + UNKNOWN = 0, + + /// Snapdragon CPU. + CPU = 1, + + /// Adreno GPU. + GPU = 2, + + /// Hexagon DSP. + DSP = 3 + }; + + /** + * @brief . + * + * A union class user platform configuration information + */ + struct PlatformConfigInfo + { + /// Holds user GPU Configuration. + /// + UserGpuConfig userGpuConfig; + + }; + + ~PlatformConfig() = default; + + PlatformConfig() + : BaseType(Snpe_PlatformConfig_Create()) + { } + + PlatformConfig(const PlatformConfig& other) + : BaseType(Snpe_PlatformConfig_CreateCopy(other.handle())) + { } + + /** + * @brief Retrieves the platform type + * + * @return Platform type + */ + PlatformType_t getPlatformType() const{ + return static_cast(Snpe_PlatformConfig_GetPlatformType(handle())); + }; + + /** + * @brief Indicates whther the plaform configuration is valid. + * + * @return True if the platform configuration is valid; false otherwise. + */ + bool isValid() const{ + return Snpe_PlatformConfig_IsValid(handle()); + }; + + /** + * @brief Retrieves the Gpu configuration + * + * @param[out] userGpuConfig The passed in userGpuConfig populated with the Gpu configuration on return. + * + * @return True if Gpu configuration was retrieved; false otherwise. + */ + bool getUserGpuConfig(UserGpuConfig& userGpuConfig) const{ + auto platformType = static_cast(Snpe_PlatformConfig_GetPlatformType(handle())); + if(platformType != PlatformType_t::GPU) return false; + + UserGpuConfigInternal gpuConf(moveHandle(Snpe_PlatformConfig_GetUserGpuConfig(handle()))); + + gpuConf.get(userGpuConfig.userGLConfig); + return true; + } + + /** + * @brief Sets the Gpu configuration + * + * @param[in] userGpuConfig Gpu Configuration + * + * @return True if Gpu configuration was successfully set; false otherwise. + */ + bool setUserGpuConfig(UserGpuConfig& userGpuConfig){ + UserGpuConfigInternal gpuConf; + gpuConf.set(userGpuConfig.userGLConfig); + return Snpe_PlatformConfig_SetUserGpuConfig(handle(), getHandle(gpuConf)); + } + + /** + * @brief Sets the platform options + * + * @param[in] options Options as a string in the form of "keyword:options" + * + * @return True if options are pass validation; otherwise false. If false, the options are not updated. + */ + bool setPlatformOptions(const std::string& options){ + return Snpe_PlatformConfig_SetPlatformOptions(handle(), options.c_str()); + } + + /** + * @brief Indicates whther the plaform configuration is valid. + * + * @return True if the platform configuration is valid; false otherwise. + */ + bool isOptionsValid() const{ + return Snpe_PlatformConfig_IsOptionsValid(handle()); + } + + /** + * @brief Gets the platform options + * + * @return Options as a string + */ + std::string getPlatformOptions() const { + return Snpe_PlatformConfig_GetPlatformOptions(handle()); + } + + /** + * @brief Sets the platform options + * + * @param[in] optionName Name of platform options" + * @param[in] value Value of specified optionName + * + * @return If true, add "optionName:value" to platform options if optionName don't exist, otherwise update the + * value of specified optionName. + * If false, the platform options will not be changed. + */ + bool setPlatformOptionValue(const std::string& optionName, const std::string& value){ + return Snpe_PlatformConfig_SetPlatformOptionValue(handle(), optionName.c_str(), value.c_str()); + } + + /** + * @brief Removes the platform options + * + * @param[in] optionName Name of platform options" + * @param[in] value Value of specified optionName + * + * @return If true, removed "optionName:value" to platform options if optionName don't exist, do nothing. + * If false, the platform options will not be changed. + */ + bool removePlatformOptionValue(const std::string& optionName, const std::string& value){ + return Snpe_PlatformConfig_RemovePlatformOptionValue(handle(), optionName.c_str(), value.c_str()); + } + + static void SetIsUserGLBuffer(bool isUserGLBuffer){ + Snpe_PlatformConfig_SetIsUserGLBuffer(isUserGLBuffer); + } + static bool GetIsUserGLBuffer(){ + return Snpe_PlatformConfig_GetIsUserGLBuffer(); + } + +}; + + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserGLConfig) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserGpuConfig) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, PlatformConfig) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.h new file mode 100644 index 00000000..2b699a7a --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.h @@ -0,0 +1,203 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_RUNTIME_LIST_H +#define DL_SYSTEM_RUNTIME_LIST_H + +#include + +#include "DlSystem/DlEnums.h" +#include "DlSystem/DlError.h" + +#include "StringList.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE RuntimeList handle + */ +typedef void* Snpe_RuntimeList_Handle_t; + +/** + * @brief . + * + * Creates a new runtime list + * + */ +SNPE_API +Snpe_RuntimeList_Handle_t Snpe_RuntimeList_Create(); + + +/** + * Copy-Constructs a RuntimeList and returns a handle to it + * + * @param runtimeListHandle the other RuntimeList to copy + * + * @return the handle to the created RuntimeList + */ +SNPE_API +Snpe_RuntimeList_Handle_t Snpe_RuntimeList_CreateCopy(Snpe_RuntimeList_Handle_t runtimeListHandle); + +/** + * @brief Destroys the RuntimeList + * + * @param[in] runtimeListHandle : Handle needed to access the runtimeList + * + * @return Error code. Returns SNPE_SUCCESS if destruction successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeList_Delete(Snpe_RuntimeList_Handle_t runtimeListHandle); + +/** + * Copy-assigns the contents of srcHandle into dstHandle + * + * @param src Source RuntimeList handle + * + * @param dst Destination RuntimeList handle + * + * @return SNPE_SUCCESS on successful copy-assignment + */ +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeList_Assign(Snpe_RuntimeList_Handle_t src, Snpe_RuntimeList_Handle_t dst); + +/** + * @brief Returns the Runtime from list at position index + * + * @param[in] runtimeListHandle: Handle needed to access the runtimeList + * + * @param[in] index : position in runtimeList + * + * @return The Runtime from list at position index + */ +SNPE_API +Snpe_Runtime_t Snpe_RuntimeList_GetRuntime(Snpe_RuntimeList_Handle_t runtimeListHandle, int index); + +/** + * @brief Set the Runtime of the list at position index + * + * @param[in] runtimeListHandle : Handle needed to access the runtimeList + * + * @param[in] index : position in runtimeList + * + * @param[in] runtime : The Runtime to assign to position index + * + * @return SNPE_SUCCESS on success + */ +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeList_SetRuntime(Snpe_RuntimeList_Handle_t runtimeListHandle, size_t index, Snpe_Runtime_t runtime); + +/** + * @brief Adds runtime to the end of the runtime list + * order of precedence is former followed by latter entry + * + * @param[in] runtimeListHandle: Handle needed to access the runtimeList + * + * @param[in] runtime to add + * + * @return Error code. Ruturns SNPE_SUCCESS If the runtime added successfully + */ +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeList_Add(Snpe_RuntimeList_Handle_t runtimeListHandle, Snpe_Runtime_t runtime); + +/** + * @brief Removes the runtime from the list + * + * @param[in] runtimeListHandle: Handle needed to access the runtimeList + * + * @param[in] runtime to be removed + * + * @return Error code. Ruturns SNPE_SUCCESS If the runtime removed successfully + */ +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeList_Remove(Snpe_RuntimeList_Handle_t runtimeListHandle, Snpe_Runtime_t runtime) ; + +/** + * @brief Returns the number of runtimes in the list + * + * @param[in] runtimeListHandle: Handle needed to access the runtimeList + * + * @return number of entries in the runtimeList. + */ +SNPE_API +size_t Snpe_RuntimeList_Size(Snpe_RuntimeList_Handle_t runtimeListHandle) ; + +/** + * @brief Returns 1 if the list is empty + * + * @param[in] runtimeListHandle: Handle needed to access the runtimeList + * + * @return 1 if list empty, 0 otherwise. + */ +SNPE_API +int Snpe_RuntimeList_Empty(Snpe_RuntimeList_Handle_t runtimeListHandle) ; + +/** + * @brief . + * + * Removes all runtime from the list + * + * @param[in] runtimeListHandle: Handle needed to access the runtimeList + * + * @return Error code. Returns SNPE_SUCCESS if runtime list is cleared successfully. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeList_Clear(Snpe_RuntimeList_Handle_t runtimeListHandle); + +/** + * @brief Get a StringList of names from the runtime list in order of precedence + * + * @param runtimeListHandle Handle to a RuntimeList + * + * @return Handle to a StringList + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_RuntimeList_GetRuntimeListNames(Snpe_RuntimeList_Handle_t runtimeListHandle); + +/** + * @brief . + * + * @param[in] runtime const char* + * Returns a Runtime enum corresponding to the in param string + * + */ +SNPE_API +Snpe_Runtime_t Snpe_RuntimeList_StringToRuntime(const char* str); + +/** + * @brief . + * + * @param[in] runtime + * Returns a const char* corresponding to the in param runtime enum + * + */ +SNPE_API +const char* Snpe_RuntimeList_RuntimeToString(Snpe_Runtime_t runtime); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_RUNTIME_LIST_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.hpp new file mode 100644 index 00000000..a2abf2b7 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/RuntimeList.hpp @@ -0,0 +1,115 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "StringList.hpp" +#include "DlEnums.hpp" +#include "DlSystem/RuntimeList.h" + + + + + + +namespace DlSystem { + +class RuntimeList : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_RuntimeList_Delete}; + + static Runtime_t GetRuntime(HandleType handle, size_t idx){ + return static_cast(Snpe_RuntimeList_GetRuntime(handle, int(idx))); + } + static Snpe_ErrorCode_t SetRuntime(HandleType handle, size_t idx, Runtime_t runtime){ + return Snpe_RuntimeList_SetRuntime(handle, idx, static_cast(runtime)); + } + +private: + using RuntimeReference = WrapperDetail::MemberIndexedReference; + friend RuntimeReference; + +public: + + RuntimeList() + : BaseType(Snpe_RuntimeList_Create()) + { } + RuntimeList(const RuntimeList& other) + : BaseType(Snpe_RuntimeList_CreateCopy(other.handle())) + { } + RuntimeList(RuntimeList&& other) noexcept + : BaseType(std::move(other)) + { } + + RuntimeList(const Runtime_t& runtime) + : BaseType(Snpe_RuntimeList_Create()) + { + Snpe_RuntimeList_Add(handle(), static_cast(runtime)); + } + + RuntimeList& operator=(const RuntimeList& other){ + if(this != &other){ + Snpe_RuntimeList_Assign(other.handle(), handle()); + } + return *this; + } + + RuntimeList& operator=(RuntimeList&& other) noexcept{ + return moveAssign(std::move(other)); + } + + Runtime_t operator[](size_t idx) const{ + return GetRuntime(handle(), idx); + } + + RuntimeReference operator[](size_t idx) noexcept{ + return {*this, idx}; + } + + bool add(const Runtime_t& runtime){ + return SNPE_SUCCESS == Snpe_RuntimeList_Add(handle(), static_cast(runtime)); + } + + void remove(Runtime_t runtime) noexcept{ + Snpe_RuntimeList_Remove(handle(), static_cast(runtime)); + } + + size_t size() const noexcept{ + return Snpe_RuntimeList_Size(handle()); + } + + bool empty() const noexcept{ + return Snpe_RuntimeList_Empty(handle()); + } + + void clear() noexcept{ + Snpe_RuntimeList_Clear(handle()); + } + + StringList getRuntimeListNames() const{ + return moveHandle(Snpe_RuntimeList_GetRuntimeListNames(handle())); + } + + static Runtime_t stringToRuntime(const char* runtimeStr){ + return static_cast(Snpe_RuntimeList_StringToRuntime(runtimeStr)); + } + static const char* runtimeToString(Runtime_t runtime){ + return Snpe_RuntimeList_RuntimeToString(static_cast(runtime)); + } + +}; + + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, RuntimeList) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/SnpeApiExportDefine.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/SnpeApiExportDefine.h new file mode 100644 index 00000000..62c6718f --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/SnpeApiExportDefine.h @@ -0,0 +1,34 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +// Macro controlling visibility of SNPE API + +#ifdef __cplusplus +extern "C" { +#endif + +#ifndef SNPE_API +#define SNPE_API +#endif + +#ifdef __cplusplus +} // extern "C" +#endif diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/String.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/String.hpp new file mode 100644 index 00000000..85b2ef22 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/String.hpp @@ -0,0 +1,70 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + + +#include + + +#include "Wrapper.hpp" + +namespace DlSystem{ + + +// Just a backwards compatible wrapper for std::string +class String{ +public: + String() = delete; + explicit String(const std::string& str) + : m_String(str) + { } + explicit String(std::string&& str) noexcept + : m_String(std::move(str)) + { } + + explicit String(const char* str) + : m_String(str) + { } + + String(String&& other) noexcept = default; + String(const String& other) = delete; + + + String& operator=(String&& other) noexcept = default; + String& operator=(const String& other) = delete; + + bool operator<(const String& rhs) const noexcept{ return m_String < rhs.m_String; } + bool operator>(const String& rhs) const noexcept{ return m_String > rhs.m_String; } + bool operator<=(const String& rhs) const noexcept{ return m_String <= rhs.m_String; } + bool operator>=(const String& rhs) const noexcept{ return m_String >= rhs.m_String; } + bool operator==(const String& rhs) const noexcept{ return m_String == rhs.m_String; } + bool operator!=(const String& rhs) const noexcept{ return m_String != rhs.m_String; } + + + bool operator<(const std::string& rhs) const noexcept{ return m_String < rhs; } + bool operator>(const std::string& rhs) const noexcept{ return m_String > rhs; } + bool operator<=(const std::string& rhs) const noexcept{ return m_String <= rhs; } + bool operator>=(const std::string& rhs) const noexcept{ return m_String >= rhs; } + bool operator==(const std::string& rhs) const noexcept{ return m_String == rhs; } + bool operator!=(const std::string& rhs) const noexcept{ return m_String != rhs; } + + + const char* c_str() const noexcept{ return m_String.c_str(); } + + explicit operator std::string&() noexcept{ return m_String; } + explicit operator const std::string&() const noexcept{ return m_String; } + +private: + std::string m_String; +}; + + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, String) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.h new file mode 100644 index 00000000..faa793b3 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.h @@ -0,0 +1,154 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_STRING_LIST_H +#define DL_SYSTEM_STRING_LIST_H + +#ifdef __cplusplus +#include +#else +#include +#endif + +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * A typedef to indicate a SNPE StringList handle + */ +typedef void* Snpe_StringList_Handle_t; + +/** + * Constructs a StringList and returns a handle to it + * + * @return the handle to the created StringList + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_StringList_Create(); + +/** + * Constructs a StringList and returns a handle to it + * + * @param[in] size : size of list + * + * @return the handle to the created StringList + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_StringList_CreateSize(size_t size); + +/** + * Constructs a StringList and returns a handle to it + * + * @param[in] other : StringList handle to be copied from + * + * @return the handle to the created StringList + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_StringList_CreateCopy(Snpe_StringList_Handle_t other); + +/** + * Destroys/frees a StringList + * + * @param[in] stringListHandle : Handle to access the stringList + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_StringList_Delete(Snpe_StringList_Handle_t stringListHandle); + + +/** + * Append a string to the list. + * + * @param[in] stringListHandle : Handle to access the stringList + * @param[in] str Null-terminated ASCII string to append to the list. + * + * @return SNPE_SUCCESS if Append operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_StringList_Append(Snpe_StringList_Handle_t stringListHandle, const char* string); + +/** + * Returns the string at the indicated position, + * or an empty string if the positions is greater than the size + * of the list. + * + * @param[in] stringListHandle : Handle to access the stringList + * @param[in] idx Position in the list of the desired string + * + * @return the string at the indicated position + */ +SNPE_API +const char* Snpe_StringList_At(Snpe_StringList_Handle_t stringListHandle, size_t idx); + +/** + * Pointer to the first string in the list. + * Can be used to iterate through the list. + * + * @param[in] stringListHandle : Handle to access the stringList + * + * @return Pointer to the first string in the list. + */ +SNPE_API +const char** Snpe_StringList_Begin(Snpe_StringList_Handle_t stringListHandle); + +/** + * Pointer to one after the last string in the list. + * Can be used to iterate through the list. + * + * @param[in] stringListHandle : Handle to access the stringList + * + * @return Pointer to one after the last string in the list + */ +SNPE_API +const char** Snpe_StringList_End(Snpe_StringList_Handle_t stringListHandle); + +/** + * Return the number of valid string pointers held by this list. + * + * @param[in] stringListHandle : Handle to access the stringList + * + * @return The size of the StringList + */ +SNPE_API +size_t Snpe_StringList_Size(Snpe_StringList_Handle_t stringListHandle); + +/** + * Copy-assigns the contents of src into dst + * + * @param src Source StringList handle + * @param dst Destination StringList handle + * + * @return SNPE_SUCCESS on successful copy-assignment + */ +SNPE_API +Snpe_ErrorCode_t Snpe_StringList_Assign(Snpe_StringList_Handle_t src, Snpe_StringList_Handle_t dst); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_STRING_LIST_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.hpp new file mode 100644 index 00000000..2fd84bf1 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/StringList.hpp @@ -0,0 +1,73 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "DlSystem/DlError.hpp" + +#include "DlSystem/StringList.h" + + +namespace DlSystem { + +class StringList : public Wrapper{ + friend BaseType; + using BaseType::BaseType; + static constexpr DeleteFunctionType DeleteFunction = Snpe_StringList_Delete; + +public: + StringList() + : BaseType(Snpe_StringList_Create()) + { } + explicit StringList(size_t length) + : BaseType(Snpe_StringList_CreateSize(length)) + { } + StringList(const StringList& other) + : BaseType(Snpe_StringList_CreateCopy(other.handle())) + { } + StringList(StringList&& other) noexcept + : BaseType(std::move(other)) + { } + + + StringList& operator=(const StringList& other){ + if(this != &other){ + Snpe_StringList_Assign(other.handle(), handle()); + } + return *this; + } + StringList& operator=(StringList&& other) noexcept{ + return moveAssign(std::move(other)); + } + + + DlSystem::ErrorCode append(const char* str){ + return static_cast(Snpe_StringList_Append(handle(), str)); + } + + const char* at(size_t idx) const noexcept{ + return Snpe_StringList_At(handle(), idx); + } + + const char** begin() const noexcept{ + return Snpe_StringList_Begin(handle()); + } + const char** end() const noexcept{ + return Snpe_StringList_End(handle()); + } + + size_t size() const noexcept{ + return Snpe_StringList_Size(handle()); + } + +}; + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, StringList) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.h new file mode 100644 index 00000000..aa367eda --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.h @@ -0,0 +1,154 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_TENSORMAP_H +#define DL_SYSTEM_TENSORMAP_H + +#include "DlSystem/ITensor.h" +#include "DlSystem/StringList.h" +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * A typedef to indicate a SNPE Tensor Map handle + */ +typedef void* Snpe_TensorMap_Handle_t; + + +/** + * Constructs a TensorMap and returns a handle to it + * + * @return the handle to the created TensorMap + */ +SNPE_API +Snpe_TensorMap_Handle_t Snpe_TensorMap_Create(); + + +/** + * Copy-Constructs a TensorMap and returns a handle to it + * + * @param tensorMapHandle the other TensorMap to copy + * + * @return the handle to the created TensorMap + */ +SNPE_API +Snpe_TensorMap_Handle_t Snpe_TensorMap_CreateCopy(Snpe_TensorMap_Handle_t tensorMapHandle); + +/** + * Copy-assigns the contents of srcHandle into dstHandle + * + * @param src Source TensorMap handle + * + * @param dst Destination TensorMap handle + * + * @return SNPE_SUCCESS on successful copy-assignment + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorMap_Assign(Snpe_TensorMap_Handle_t srcHandle, Snpe_TensorMap_Handle_t dstHandle); + + +/** + * Destroys/frees Tensor Map + * + * @param[in] handle : handle to tensorMap + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorMap_Delete(Snpe_TensorMap_Handle_t handle); + +/** + * @brief Adds a name and the corresponding tensor pointer + * to the map + * + * @param[in] handle : Handle to tensorMap + * @param[in] name : The name of the tensor + * @param[in] tensorHandle : Handle to access ITensor + * + * @note If a tensor with the same name already exists, the + * tensor is replaced with the existing tensor. + */ +SNPE_API +void Snpe_TensorMap_Add(Snpe_TensorMap_Handle_t handle, const char *name, Snpe_ITensor_Handle_t tensorHandle); + +/** + * @brief Removes a mapping of tensor and its name by its name + * + * @param[in] handle : Handle to tensorMap + * @param[in] name : The name of tensor to be removed + * + * @note If no tensor with the specified name is found, nothing + * is done. + */ +SNPE_API +void Snpe_TensorMap_Remove(Snpe_TensorMap_Handle_t handle, const char *name); + +/** + * @brief Returns the number of tensors in the map + * + * @param[in] handle : Handle to tensorMap + * + * @return Number of tensors in the map + */ +SNPE_API +size_t Snpe_TensorMap_Size(Snpe_TensorMap_Handle_t handle); + +/** + * @brief . + * + * @param[in] handle : Handle to tensorMap + * Removes all tensors from the map + */ +SNPE_API +void Snpe_TensorMap_Clear(Snpe_TensorMap_Handle_t handle); + +/** + * @brief Returns the tensor given its name. + * + * @param[in] handle : Handle to tensorMap + * @param[in] name : The name of the tensor to get. + * + * @return nullptr if no tensor with the specified name is + * found; otherwise, a valid pointer to the tensor. + */ +SNPE_API +Snpe_ITensor_Handle_t Snpe_TensorMap_GetTensor_Ref(Snpe_TensorMap_Handle_t handle, const char *name); + +/** + * @brief . + * + * @param[in] handle : Handle to tensorMap + * + * @return A StringList of the names of all tensors + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_TensorMap_GetTensorNames(Snpe_TensorMap_Handle_t handle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_TENSOR_MAP_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.hpp new file mode 100644 index 00000000..20a6c21f --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorMap.hpp @@ -0,0 +1,81 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "DlSystem/ITensor.hpp" +#include "DlSystem/StringList.hpp" +#include "DlSystem/DlError.hpp" + +#include "DlSystem/TensorMap.h" + +namespace DlSystem { + +class TensorMap : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_TensorMap_Delete}; +public: + + TensorMap() + : BaseType(Snpe_TensorMap_Create()) + { } + + TensorMap(const TensorMap& other) + : BaseType(Snpe_TensorMap_CreateCopy(other.handle())) + { } + + TensorMap(TensorMap&& other) noexcept + : BaseType(std::move(other)) + { } + + TensorMap& operator=(const TensorMap& other){ + if(this != &other){ + Snpe_TensorMap_Assign(other.handle(), handle()); + } + return *this; + } + TensorMap& operator=(TensorMap&& other) noexcept{ + return moveAssign(std::move(other)); + } + + DlSystem::ErrorCode add(const char* name, ITensor* tensor){ + if(!tensor) return DlSystem::ErrorCode::SNPE_CAPI_BAD_ARGUMENT; + Snpe_TensorMap_Add(handle(), name, getHandle(*tensor)); + return DlSystem::ErrorCode::NONE; + } + + void remove(const char* name) noexcept{ + Snpe_TensorMap_Remove(handle(), name); + } + + size_t size() const noexcept{ + return Snpe_TensorMap_Size(handle()); + } + + void clear() noexcept{ + Snpe_TensorMap_Clear(handle()); + } + + + ITensor* getTensor(const char* name) const noexcept{ + return makeReference(Snpe_TensorMap_GetTensor_Ref(handle(), name)); + } + + StringList getTensorNames() const{ + return moveHandle(Snpe_TensorMap_GetTensorNames(handle())); + } + +}; + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, TensorMap) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.h new file mode 100644 index 00000000..1fde628c --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.h @@ -0,0 +1,174 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_TENSOR_SHAPE_H +#define DL_SYSTEM_TENSOR_SHAPE_H + +#include + +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE TensorShape handle + */ +typedef void* Snpe_TensorShape_Handle_t; + + +/** + * @brief . + * + * Creates a new shape with a list of dims specified in array + * + * @param[in] dims The dimensions are specified in which the last + * element of the vector represents the fastest varying + * dimension and the zeroth element represents the slowest + * varying, etc. + * + * @param[in] size Size of the array. + * + * @return the handle to the created TensorShape + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_TensorShape_CreateDimsSize(const size_t *dims, size_t size); + +/** + * Constructs a TensorShape and returns a handle to it + * + * @return the handle to the created TensorShape + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_TensorShape_Create(); + +/** + * @brief . + * + * copy constructor. + * @param[in] other object to copy. + * + * @return the handle to the created TensorShape. + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_TensorShape_CreateCopy(Snpe_TensorShape_Handle_t other); + +/** + * Destroys/frees Tensor Shape + * + * @param[in] handle : handle to tensorShape + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShape_Delete(Snpe_TensorShape_Handle_t tensorShapeHandle); + +/** + * Copy-assigns the contents of srcHandle into dstHandle + * + * @param srcHandle Source TensorShape handle + * @param dstHandle Destination TensorShape handle + * + * @return SNPE_SUCCESS on successful copy-assignment + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShape_Assign(Snpe_TensorShape_Handle_t srcHandle, Snpe_TensorShape_Handle_t dstHandle); + +/** + * @brief . + * + * Concatenates additional dimensions specified in + * the array to the existing dimensions. + * + * @param[in] handle : handle to tensorShape + * @param[in] dims The dimensions are specified in which the last + * element of the vector represents the fastest varying + * dimension and the zeroth element represents the slowest + * varying, etc. + * + * @param[in] size Size of the array. + * + */ +SNPE_API +void Snpe_TensorShape_Concatenate(Snpe_TensorShape_Handle_t tensorShape, const size_t *dims, size_t size); + +/** + * @brief . + * + * @param[in] handle : handle to tensorShape + * + * Retrieves the rank i.e. number of dimensions. + * + * @return The rank + */ +SNPE_API +size_t Snpe_TensorShape_Rank(Snpe_TensorShape_Handle_t tensorShape); + +/** + * @brief . + * + * @param[in] handle : handle to tensorShape + * + * @param[in] index : Position in the dimension array. + * + * @return The dimension value in tensor shape + */ +SNPE_API +size_t Snpe_TensorShape_At(Snpe_TensorShape_Handle_t tensorShapeHandle, size_t index); + +/** + * @brief Set a value in a TensorShape at the provided index + * + * @param[in] handle : handle to tensorShape + * + * @param[in] index : Position in the dimension array. + * + * @param[in] value : Dimension value to set + * + * @return SNPE_SUCCESS on success + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShape_Set(Snpe_TensorShape_Handle_t tensorShapeHandle, size_t index, size_t value); + +/** + * @brief . + * + * Retrieves a pointer to the first dimension of shape + * + * @param[in] handle : handle to tensorShape + * + * @return nullptr if no dimension exists; otherwise, points to + * the first dimension. + * + */ +SNPE_API +const size_t* Snpe_TensorShape_GetDimensions(Snpe_TensorShape_Handle_t tensorShape); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_TENSOR_SHAPE_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.hpp new file mode 100644 index 00000000..776637c7 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShape.hpp @@ -0,0 +1,104 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include +#include +#include + +#include "Wrapper.hpp" + +#include "DlSystem/TensorShape.h" + +namespace DlSystem { + + +using Dimension = size_t; + + + +class TensorShape : public Wrapper { + friend BaseType; + using BaseType::BaseType; + +protected: + static constexpr DeleteFunctionType DeleteFunction{Snpe_TensorShape_Delete}; + +private: + using DimensionReference = WrapperDetail::MemberIndexedReference; + friend DimensionReference; + +public: + + TensorShape() + : BaseType(Snpe_TensorShape_Create()) + { } + + TensorShape(const TensorShape& other) + : BaseType(Snpe_TensorShape_CreateCopy(other.handle())) + { } + + TensorShape(TensorShape&& other) noexcept + : BaseType(std::move(other)) + { } + + TensorShape(std::initializer_list dims) + : BaseType(Snpe_TensorShape_CreateDimsSize(dims.begin(), dims.size())) + { } + + TensorShape& operator=(const TensorShape& other) noexcept{ + if(this != &other){ + Snpe_TensorShape_Assign(other.handle(), handle()); + } + return *this; + } + + TensorShape& operator=(TensorShape&& other) noexcept{ + return moveAssign(std::move(other)); + } + + TensorShape(const size_t *dims, size_t size) + : BaseType(Snpe_TensorShape_CreateDimsSize(dims, size)) + { } + + TensorShape(const std::vector& dims) + : TensorShape(dims.data(), dims.size()) + { } + + + void concatenate(const size_t *dims, size_t size){ + Snpe_TensorShape_Concatenate(handle(), dims, size); + } + + void concatenate(const size_t &dim){ + return concatenate(&dim, 1); + } + + size_t operator[](size_t idx) const{ + return Snpe_TensorShape_At(handle(), idx); + } + + DimensionReference operator[](size_t idx){ + return {*this, idx}; + } + + size_t rank() const{ + return Snpe_TensorShape_Rank(handle()); + } + + const size_t* getDimensions() const{ + return Snpe_TensorShape_GetDimensions(handle()); + } + + +}; + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, Dimension) +ALIAS_IN_ZDL_NAMESPACE(DlSystem, TensorShape) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.h new file mode 100644 index 00000000..520fa5ab --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.h @@ -0,0 +1,163 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + + +/** + * @file + */ + +#ifndef _SNPE_TENSOR_SHAPE_MAP_H_ +#define _SNPE_TENSOR_SHAPE_MAP_H_ + + +#ifdef __cplusplus +#include +#else +#include +#endif + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" + +#include "DlSystem/TensorShape.h" +#include "DlSystem/StringList.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE TensorShapeMap handle + */ +typedef void* Snpe_TensorShapeMap_Handle_t; + +/** + * Constructs a TensorShapeMap and returns a handle to it + * + * @return the handle to the created TensorShapeMap + */ +SNPE_API +Snpe_TensorShapeMap_Handle_t Snpe_TensorShapeMap_Create(); + +/** + * @brief . + * + * copy constructor. + * + * @param[in] tsmHandle : Handle to the other object to copy. + * @return the handle to the created TensorShapeMap + */ +SNPE_API +Snpe_TensorShapeMap_Handle_t Snpe_TensorShapeMap_CreateCopy(Snpe_TensorShapeMap_Handle_t tsmHandle); + +/** + * Destroys/frees Tensor Shape Map + * + * @param[in] tsmhandle : handle to access Tensor Shape Map + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShapeMap_Delete(Snpe_TensorShapeMap_Handle_t tsmHandle); + +/** + * @brief . + * + * assignment operator. Copy-assigns from srcHandle to dstHandle + * @param[in] srcHandle : handle to source Tensor Shape Map object + * @param[out] dstHandle : handle to destination Tensor Shape Map object + * + * @return Returns SNPE_SUCCESS if Assignment successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShapeMap_Assign(Snpe_TensorShapeMap_Handle_t srcHandle, Snpe_TensorShapeMap_Handle_t dstHandle); + +/** + * @brief Adds a name and the corresponding tensor pointer + * to the map + * + * @param[in] tsmhandle : handle to access Tensor Shape Map + * @param[in] name The name of the tensor + * @param[in] tsHandle : Handle to access Tensor Shape + * + * @return Returns SNPE_SUCCESS if Add operation successful + * @note If a tensor with the same name already exists, no new + * tensor is added. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShapeMap_Add(Snpe_TensorShapeMap_Handle_t tsmHandle, const char* name, Snpe_TensorShape_Handle_t tsHandle); + +/** + * @brief Removes a mapping of tensor and its name by its name + * + * @param[in] tsmhandle : handle to access Tensor Shape Map + * @param[in] name The name of tensor to be removed + * @return Returns SNPE_SUCCESS if Remove operation successful + * + * @note If no tensor with the specified name is found, nothing + * is done. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShapeMap_Remove(Snpe_TensorShapeMap_Handle_t tsmHandle, const char* name); + +/** + * @brief Returns the number of tensors in the map + * @param[in] tsmhandle : handle to access Tensor Shape Map + * @return Returns number entries in TensorShapeMap + */ +SNPE_API +size_t Snpe_TensorShapeMap_Size(Snpe_TensorShapeMap_Handle_t tsmHandle); + +/** + * @brief . + * + * Removes all tensors from the map + * @param[in] tsmhandle : handle to access Tensor Shape Map + * @return Returns SNPE_SUCCESS if Clear operation successful + */ +SNPE_API +Snpe_ErrorCode_t Snpe_TensorShapeMap_Clear(Snpe_TensorShapeMap_Handle_t tsmHandle); + +/** + * @brief Returns the tensor given its name. + * + * @param[in] tsmhandle : handle to access Tensor Shape Map + * @param[in] name The name of the tensor to get. + * + * @return nullptr if no tensor with the specified name is + * found; otherwise, a valid Tensor Shape Handle. + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_TensorShapeMap_GetTensorShape(Snpe_TensorShapeMap_Handle_t tsmHandle, const char* name); + +/** + * @brief . + * + * @param[in] tsmHandle : handle to access Tensor Shape Map + * @return A stringList Handle to access names of all tensor shapes + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_TensorShapeMap_GetTensorShapeNames(Snpe_TensorShapeMap_Handle_t tsmHandle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_TENSOR_SHAPE_MAP_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.hpp new file mode 100644 index 00000000..8b79a6e2 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/TensorShapeMap.hpp @@ -0,0 +1,77 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include "DlSystem/StringList.hpp" +#include "DlSystem/TensorShape.hpp" +#include "DlSystem/DlError.hpp" + +#include "DlSystem/TensorShapeMap.h" + +namespace DlSystem { + +class TensorShapeMap : public Wrapper { + friend BaseType; + using BaseType::BaseType; + static constexpr DeleteFunctionType DeleteFunction{Snpe_TensorShapeMap_Delete}; + +public: + TensorShapeMap() + : BaseType(Snpe_TensorShapeMap_Create()) + { } + TensorShapeMap(const TensorShapeMap& other) + : BaseType(Snpe_TensorShapeMap_CreateCopy(other.handle())) + { } + TensorShapeMap(TensorShapeMap&& other) noexcept + : BaseType(std::move(other)) + { } + + TensorShapeMap& operator=(const TensorShapeMap& other){ + if(this != &other){ + Snpe_TensorShapeMap_Assign(other.handle(), handle()); + } + return *this; + } + TensorShapeMap& operator=(TensorShapeMap&& other) noexcept{ + return moveAssign(std::move(other)); + } + + DlSystem::ErrorCode add(const char *name, const TensorShape& tensorShape){ + return static_cast( + Snpe_TensorShapeMap_Add(handle(), name, getHandle(tensorShape)) + ); + } + + DlSystem::ErrorCode remove(const char* name) noexcept{ + return static_cast(Snpe_TensorShapeMap_Remove(handle(), name)); + } + + size_t size() const noexcept{ + return Snpe_TensorShapeMap_Size(handle()); + } + + DlSystem::ErrorCode clear() noexcept{ + return static_cast(Snpe_TensorShapeMap_Clear(handle())); + } + + TensorShape getTensorShape(const char* name) const noexcept{ + return moveHandle(Snpe_TensorShapeMap_GetTensorShape(handle(), name)); + } + + StringList getTensorShapeNames() const{ + return moveHandle(Snpe_TensorShapeMap_GetTensorShapeNames(handle())); + } + +}; + +} // ns DlSystem + + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, TensorShapeMap) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.h new file mode 100644 index 00000000..2da1c792 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.h @@ -0,0 +1,151 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_USER_BUFFER_MAP_H +#define DL_SYSTEM_USER_BUFFER_MAP_H + +#include "DlSystem/StringList.h" +#include "DlSystem/IUserBuffer.h" +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE UserBufferMap handle + */ +typedef void* Snpe_UserBufferMap_Handle_t; + +/** + * @brief . + * + * Creates a new empty UserBuffer map + */ +SNPE_API +Snpe_UserBufferMap_Handle_t Snpe_UserBufferMap_Create(); + +/** + * copy constructor. + * @param[in] other : Handle to the other userBufferMap to be copied from. + */ +SNPE_API +Snpe_UserBufferMap_Handle_t Snpe_UserBufferMap_CreateCopy(Snpe_UserBufferMap_Handle_t other); + + +/** + * @brief Adds a name and the corresponding UserBuffer pointer + * to the map + * + * @param[in] handle : Handle to access UserBufferMap + * @param[in] name : The name of the UserBuffer + * @param[in] bufferHandle : Handle to access UserBuffer + * + * @note If a UserBuffer with the same name already exists, the new + * UserBuffer pointer would be updated. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferMap_Add(Snpe_UserBufferMap_Handle_t handle, const char *name, Snpe_IUserBuffer_Handle_t bufferHandle); + +/** + * @brief Removes a mapping of one UserBuffer and its name by its name + * + * @param[in] handle : Handle to access UserBufferMap + * + * @param[in] name : The name of UserBuffer to be removed + * + * @note If no UserBuffer with the specified name is found, nothing + * is done. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferMap_Remove(Snpe_UserBufferMap_Handle_t handle, const char *name); + +/** + * @brief Returns the number of UserBuffers in the map + * @param[in] handle : Handle to access UserBufferMap + */ +SNPE_API +size_t Snpe_UserBufferMap_Size(Snpe_UserBufferMap_Handle_t handle); + +/** + * @brief . + * + * @param[in] handle : Handle to access UserBufferMap + * Removes all UserBuffers from the map + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferMap_Clear(Snpe_UserBufferMap_Handle_t handle); + +/** + * @brief Returns the UserBuffer given its name. + * + * @param[in] handle : Handle to access UserBufferMap + * + * @param[in] name : The name of the UserBuffer to get. + * + * @return nullptr if no UserBuffer with the specified name is + * found; otherwise, a valid pointer to the UserBuffer. + */ +SNPE_API +Snpe_IUserBuffer_Handle_t Snpe_UserBufferMap_GetUserBuffer_Ref(Snpe_UserBufferMap_Handle_t handle , const char *name); + +/** + * @brief . + * + * Returns the names of all UserBuffers + * + * @param[in] handle : Handle to access UserBufferMap + * + * @return A list of UserBuffer names. + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_UserBufferMap_GetUserBufferNames(Snpe_UserBufferMap_Handle_t handle); + +/** + * Copy-assigns the contents of srcHandle into dstHandle + * + * @param src Source UserBufferMap handle + * @param dst Destination UserBufferMap handle + * + * @return SNPE_SUCCESS on successful copy-assignment + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferMap_Assign(Snpe_UserBufferMap_Handle_t srcHandle, Snpe_UserBufferMap_Handle_t dstHandle); + +/** + * Destroys/frees UserBuffer Map + * + * @param[in] handle : Handle to access UserBuffer Map + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferMap_Delete(Snpe_UserBufferMap_Handle_t handle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_USER_BUFFER_MAP_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.hpp new file mode 100644 index 00000000..acf3207c --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserBufferMap.hpp @@ -0,0 +1,80 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include + +#include "Wrapper.hpp" +#include "DlSystem/DlError.hpp" +#include "DlSystem/StringList.hpp" +#include "DlSystem/IUserBuffer.hpp" + +#include "DlSystem/UserBufferMap.h" + +namespace DlSystem { + +class UserBufferMap : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserBufferMap_Delete}; + +public: + UserBufferMap() + : BaseType(Snpe_UserBufferMap_Create()) + { } + + UserBufferMap(const UserBufferMap& other) + : BaseType(Snpe_UserBufferMap_CreateCopy(other.handle())) + { } + UserBufferMap(UserBufferMap&& other) noexcept + : BaseType(std::move(other)) + { } + + UserBufferMap& operator=(const UserBufferMap& other){ + if(this != &other){ + Snpe_UserBufferMap_Assign(other.handle(), handle()); + } + return *this; + } + UserBufferMap& operator=(UserBufferMap&& other) noexcept{ + return moveAssign(std::move(other)); + } + + DlSystem::ErrorCode add(const char* name, IUserBuffer* buffer){ + if(!buffer) return ErrorCode::SNPE_CAPI_BAD_ARGUMENT; + return static_cast(Snpe_UserBufferMap_Add(handle(), name, getHandle(*buffer))); + } + + DlSystem::ErrorCode remove(const char* name) noexcept{ + return static_cast(Snpe_UserBufferMap_Remove(handle(), name)); + } + + size_t size() const noexcept{ + return Snpe_UserBufferMap_Size(handle()); + } + + DlSystem::ErrorCode clear() noexcept{ + return static_cast(Snpe_UserBufferMap_Clear(handle())); + } + + IUserBuffer* getUserBuffer(const char* name) const noexcept{ + return makeReference(Snpe_UserBufferMap_GetUserBuffer_Ref(handle(), name)); + } + + StringList getUserBufferNames() const{ + return moveHandle(Snpe_UserBufferMap_GetUserBufferNames(handle())); + } + +}; + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserBufferMap) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.h new file mode 100644 index 00000000..c927d33e --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.h @@ -0,0 +1,156 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef DL_SYSTEM_USER_MEMORY_MAP_H +#define DL_SYSTEM_USER_MEMORY_MAP_H + +#include "DlSystem/StringList.h" +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE User Memory handle + */ +typedef void* Snpe_UserMemoryMap_Handle_t; + +/** + * @brief . + * + * Creates a new empty UserMemory map + */ +SNPE_API +Snpe_UserMemoryMap_Handle_t Snpe_UserMemoryMap_Create(); + +/** + * copy constructor. + * @param[in] other : Handle to the other object to copy. + */ +SNPE_API +Snpe_UserMemoryMap_Handle_t Snpe_UserMemoryMap_Copy(Snpe_UserMemoryMap_Handle_t other); + +/** + * Copy-assigns the contents of srcHandle into dstHandle + * + * @param[in] srcHandle Source UserMemoryMap handle + * + * @param[out] dstHandle Destination UserMemoryMap handle + * + * @return SNPE_SUCCESS on successful copy-assignment + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserMemoryMap_Assign(Snpe_UserMemoryMap_Handle_t srcHandle, Snpe_UserMemoryMap_Handle_t dstHandle); + +/** + * Destroys/frees UserMemory Map + * + * @param[in] handle : Handle to access UserMemory Map + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserMemoryMap_Delete(Snpe_UserMemoryMap_Handle_t handle); + +/** + * @brief Adds a name and the corresponding buffer address + * to the map + * + * @param[in] handle : Handle to access UserMemory Map + * @param[in] name : The name of the UserMemory + * @param[in] address : The pointer to the Buffer Memory + * + * @note If a UserBuffer with the same name already exists, the new + * address would be updated. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserMemoryMap_Add(Snpe_UserMemoryMap_Handle_t handle, const char *name, void *address); + +/** + * @brief Removes a mapping of one Buffer address and its name by its name + * + * @param[in] handle : Handle to access UserMemory Map + * @param[in] name : The name of Memory address to be removed + * + * @note If no UserBuffer with the specified name is found, nothing + * is done. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserMemoryMap_Remove(Snpe_UserMemoryMap_Handle_t handle, const char *name); + +/** + * @brief Returns the number of User Memory addresses in the map + * @param[in] handle : Handle to access UserMemory Map + */ +SNPE_API +size_t Snpe_UserMemoryMap_Size(Snpe_UserMemoryMap_Handle_t handle); + +/** + * @brief . + * + * Removes all User Memory from the map + * @param[in] handle : Handle to access UserMemory Map + */ +SNPE_API +Snpe_ErrorCode_t Snpe_UserMemoryMap_Clear(Snpe_UserMemoryMap_Handle_t handle); + +/** + * @brief . + * Returns the names of all User Memory + * + * @param[in] handle : Handle to access UserMemory Map + * + * @return Returns a handle to the stringList. + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_UserMemoryMap_GetUserBufferNames(Snpe_UserMemoryMap_Handle_t handle); + +/** + * @brief Returns the no of UserMemory addresses mapped to the buffer + * + * @param[in] handle : Handle to access UserMemory Map + * @param[in] name : The name of the UserMemory + * + */ +SNPE_API +size_t Snpe_UserMemoryMap_GetUserMemoryAddressCount(Snpe_UserMemoryMap_Handle_t handle, const char *name); + +/** + * @brief Returns address at a specified index corresponding to a UserMemory buffer name + * + * @param[in] handle : Handle to access UserMemory Map + * @param[in] name : The name of the buffer + * @param[in] index : The index in the list of addresses + * + */ +SNPE_API +void* Snpe_UserMemoryMap_GetUserMemoryAddressAtIndex(Snpe_UserMemoryMap_Handle_t handle, const char *name, uint32_t index); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // DL_SYSTEM_USER_MEMORY_MAP_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.hpp new file mode 100644 index 00000000..36e9cd37 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/DlSystem/UserMemoryMap.hpp @@ -0,0 +1,76 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "DlSystem/DlError.hpp" +#include "DlSystem/StringList.hpp" + +#include "DlSystem/UserMemoryMap.h" + +namespace DlSystem { + +class UserMemoryMap : public Wrapper { + friend BaseType; +// Use this to get free move Ctor and move assignment operator, provided this class does not specify +// as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserMemoryMap_Delete}; +public: + UserMemoryMap() + : BaseType(Snpe_UserMemoryMap_Create()) + { } + UserMemoryMap(const UserMemoryMap& other) + : BaseType(Snpe_UserMemoryMap_Copy(other.handle())) + { } + UserMemoryMap(UserMemoryMap&& other) noexcept + : BaseType(std::move(other)) + { } + + UserMemoryMap& operator=(const UserMemoryMap& other){ + if(this != &other){ + Snpe_UserMemoryMap_Assign(handle(), other.handle()); + } + return *this; + } + + DlSystem::ErrorCode add(const char* name, void* address) noexcept{ + return static_cast(Snpe_UserMemoryMap_Add(handle(), name, address)); + } + + DlSystem::ErrorCode remove(const char* name){ + return static_cast(Snpe_UserMemoryMap_Remove(handle(), name)); + } + + size_t size() const noexcept{ + return Snpe_UserMemoryMap_Size(handle()); + } + + DlSystem::ErrorCode clear() noexcept{ + return static_cast(Snpe_UserMemoryMap_Clear(handle())); + } + + StringList getUserBufferNames() const{ + return moveHandle(Snpe_UserMemoryMap_GetUserBufferNames(handle())); + } + + size_t getUserMemoryAddressCount(const char* name) const noexcept{ + return Snpe_UserMemoryMap_GetUserMemoryAddressCount(handle(), name); + } + + void* getUserMemoryAddressAtIndex(const char* name, uint32_t index) const noexcept{ + return Snpe_UserMemoryMap_GetUserMemoryAddressAtIndex(handle(), name, index); + } + +}; + + +} // ns DlSystem + +ALIAS_IN_ZDL_NAMESPACE(DlSystem, UserMemoryMap) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.h new file mode 100644 index 00000000..282ee547 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.h @@ -0,0 +1,107 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _PLATFORM_VALIDATOR_H_ +#define _PLATFORM_VALIDATOR_H_ + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" +#include "DlSystem/DlEnums.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * A typedef to indicate a SNPE PlatformValidator handle + */ +typedef void* Snpe_PlatformValidator_Handle_t; + +/** + * @brief . + * + * Creates a new Platform Validator + * + */ +SNPE_API +Snpe_PlatformValidator_Handle_t Snpe_PlatformValidator_Create(); + + +/** + * Destroys/frees Platform Validator + * + * @param[in] handle : Handle to access Platform Validator + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PlatformValidator_Delete(Snpe_PlatformValidator_Handle_t handle); + +/** + * @brief Sets the runtime processor for compatibility check + * + * @return Void + */ +SNPE_API +void Snpe_PlatformValidator_SetRuntime(Snpe_PlatformValidator_Handle_t handle, + Snpe_Runtime_t runtime, + bool unsignedPD=true); + +/** + * @brief Checks if the Runtime prerequisites for SNPE are available. + * + * @return 1 if the Runtime prerequisites are available, else 0. + */ +SNPE_API +int Snpe_PlatformValidator_IsRuntimeAvailable(Snpe_PlatformValidator_Handle_t handle, + bool unsignedPD=true); + +/** + * @brief Returns the core version for the Runtime selected. + * + * @return char* which contains the actual core version value + */ +SNPE_API +const char* Snpe_PlatformValidator_GetCoreVersion(Snpe_PlatformValidator_Handle_t handle); + +/** + * @brief Returns the library version for the Runtime selected. + * + * @return char* which contains the actual lib version value + */ +SNPE_API +const char* Snpe_PlatformValidator_GetLibVersion(Snpe_PlatformValidator_Handle_t handle); + +/** + * @brief Runs a small program on the runtime and Checks if SNPE is supported for Runtime. + * + * @return If 1, the device is ready for SNPE execution, else return 0. + */ +SNPE_API +int Snpe_PlatformValidator_RuntimeCheck(Snpe_PlatformValidator_Handle_t handle, + bool unsignedPD=true); + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _PLATFORM_VALIDATOR_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.hpp new file mode 100644 index 00000000..de52635c --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/PlatformValidator/PlatformValidator.hpp @@ -0,0 +1,57 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include + +#include "Wrapper.hpp" + +#include "DlSystem/DlEnums.hpp" + + +#include "PlatformValidator/PlatformValidator.h" + + +namespace SNPE { + +class PlatformValidator : public Wrapper { + friend BaseType; + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_PlatformValidator_Delete}; + +public: + PlatformValidator() + : BaseType(Snpe_PlatformValidator_Create()) + { } + + void setRuntime(DlSystem::Runtime_t runtime, bool unsignedPD=true){ + Snpe_PlatformValidator_SetRuntime(handle(), static_cast(runtime), unsignedPD); + } + + bool isRuntimeAvailable(bool unsignedPD=true){ + return Snpe_PlatformValidator_IsRuntimeAvailable(handle(), unsignedPD); + } + + std::string getCoreVersion(){ + return Snpe_PlatformValidator_GetCoreVersion(handle()); + } + + std::string getLibVersion(){ + return Snpe_PlatformValidator_GetLibVersion(handle()); + } + + bool runtimeCheck(bool unsignedPD=true){ + return Snpe_PlatformValidator_RuntimeCheck(handle(), unsignedPD); + } + +}; + +} // ns SNPE + +ALIAS_IN_ZDL_NAMESPACE(SNPE, PlatformValidator) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.h new file mode 100644 index 00000000..8a2bb7d2 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.h @@ -0,0 +1,85 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef _SNPE_APPLICATION_BUFFER_MAP_H_ +#define _SNPE_APPLICATION_BUFFER_MAP_H_ + + +#ifdef __cplusplus +#include +#else +#include +#endif + + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" +#include "DlSystem/StringList.h" + + +#ifdef __cplusplus +extern "C" { +#endif + + +typedef void* Snpe_ApplicationBufferMap_Handle_t; + +SNPE_API +Snpe_ApplicationBufferMap_Handle_t Snpe_ApplicationBufferMap_Create(); + +SNPE_API +Snpe_ErrorCode_t Snpe_ApplicationBufferMap_Delete(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle); + + +SNPE_API +Snpe_ErrorCode_t Snpe_ApplicationBufferMap_Add(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle, + const char* name, + const uint8_t* buff, + size_t size); + +SNPE_API +Snpe_ErrorCode_t Snpe_ApplicationBufferMap_AddFloat(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle, + const char* name, + const float* buff, + size_t size); + +SNPE_API +Snpe_ErrorCode_t Snpe_ApplicationBufferMap_Remove(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle, + const char* name); + +SNPE_API +size_t Snpe_ApplicationBufferMap_Size(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_ApplicationBufferMap_Clear(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle); + + +SNPE_API +Snpe_StringList_Handle_t Snpe_ApplicationBufferMap_GetUserBufferNames(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle); + + +SNPE_API +Snpe_ErrorCode_t Snpe_ApplicationBufferMap_GetUserBuffer(Snpe_ApplicationBufferMap_Handle_t applicationBufferMapHandle, + const char* name, + size_t* size, + const uint8_t** data); + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_APPLICATION_BUFFER_MAP_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.hpp new file mode 100644 index 00000000..6ad745bb --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/ApplicationBufferMap.hpp @@ -0,0 +1,90 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include +#include +#include +#include + +#include "Wrapper.hpp" +#include "DlSystem/StringList.hpp" + +#include "SNPE/ApplicationBufferMap.h" + +namespace PSNPE { + +class ApplicationBufferMap : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_ApplicationBufferMap_Delete}; +public: + ApplicationBufferMap() + : BaseType(Snpe_ApplicationBufferMap_Create()){} + + explicit ApplicationBufferMap(const std::unordered_map> &buffer) + : ApplicationBufferMap(){ + for(const auto &kv: buffer){ + add(kv.first.c_str(), kv.second); + } + } + + void add(const char *name, const std::vector &buff){ + Snpe_ApplicationBufferMap_Add(handle(), name, buff.data(), buff.size()); + } + + void add(const char *name, const std::vector &buff){ + Snpe_ApplicationBufferMap_Add(handle(), name, reinterpret_cast(buff.data()), buff.size()*sizeof(float)); + } + + void remove(const char *name) noexcept{ + Snpe_ApplicationBufferMap_Remove(handle(), name); + } + + size_t size() const noexcept{ + return Snpe_ApplicationBufferMap_Size(handle()); + } + + void clear() noexcept{ + Snpe_ApplicationBufferMap_Clear(handle()); + } + + std::vector getUserBuffer(const char *name) const{ + size_t size{}; + const uint8_t *data{}; + Snpe_ApplicationBufferMap_GetUserBuffer(handle(), name, &size, &data); + + return std::vector(data, data + size); + } + + std::vector operator[](const char *name) const{ + return getUserBuffer(name); + } + + DlSystem::StringList getUserBufferNames() const{ + return moveHandle(Snpe_ApplicationBufferMap_GetUserBufferNames(handle())); + } + + std::unordered_map> getUserBuffer() const{ + std::unordered_map> toret; + for(auto name: getUserBufferNames()){ + toret.emplace(name, getUserBuffer(name)); + } + + return toret; + } + +}; + +} // ns PSNPE + + +ALIAS_IN_ZDL_NAMESPACE(PSNPE, ApplicationBufferMap) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.h new file mode 100644 index 00000000..2358d535 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.h @@ -0,0 +1,898 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022,2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _SNPE_PSNPE_H_ +#define _SNPE_PSNPE_H_ + + +#ifdef __cplusplus +#include +#else +#include +#endif + +#include "DlContainer/DlContainer.h" +#include "SNPE/ApplicationBufferMap.h" +#include "SNPE/RuntimeConfigList.h" +#include "SNPE/UserBufferList.h" +#include "DlSystem/TensorShape.h" +#include "DlSystem/IBufferAttributes.h" + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" + +#include "DlSystem/UserMemoryMap.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * A typedef to indicate the callback PSNPE handle of Async Output mode + */ +typedef void* Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t; + +//SNPE_API +//Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t Snpe_PSNPE_OutputAsyncCallbackParam_Create(size_t index, +// int status, +// const char* errorMsg); +// +//SNPE_API +//Snpe_ErrorCode_t Snpe_PSNPE_OutputAsyncCallbackParam_Delete(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t oacpHandle); + +// NOTE: we don't need _{Create,Delete} functions because the user does not create or delete these handles +// They're passed in to the callback functions they created + +/** + * @brief Get the data index of an output async PSNPE object + * + * @param[in] oacpHandle Handle to access the PSNPE object of output async mode + * + * @return The data idx for output async mode + */ +SNPE_API +size_t Snpe_PSNPE_OutputAsyncCallbackParam_GetDataIdx(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t oacpHandle); + +/** + * @brief Execute an output async PSNPE object + * + * @param[in] oacpHandle Handle to access the PSNPE object of output async mode + * + * @return True if executed successfully with outputAsync mode + */ +SNPE_API +int Snpe_PSNPE_OutputAsyncCallbackParam_GetExecuteStatus(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t oacpHandle); + +/** + * @brief Get the error message during the execution of PSNPE output async mode + * + * @param[in] oacpHandle Handle to access the PSNPE object of output async mode + * + * @return Error message + */ +SNPE_API +const char* Snpe_PSNPE_OutputAsyncCallbackParam_GetErrorMsg(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t oacpHandle); + +/** + * @brief Get the ID of an output async PSNPE object + * + * @param[in] oacpHandle Handle to access the PSNPE object of output async mode + * + * @return The id of an PSNPE object for output async mode + */ +SNPE_API +size_t Snpe_PSNPE_OutputAsyncCallbackParam_GetID(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t oacpHandle); + + + +/** + * A typedef to indicate the output callback of PSNPE handle of input-output async mode + */ +typedef void* Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t; + +/** + * @brief Get the data index of an input-output async PSNPE object + * + * @param[in] oacpHandle Handle to access the PSNPE object of input-output async mode + * + * @return The data index for input-output async mode + */ +SNPE_API +size_t Snpe_PSNPE_InputOutputAsyncCallbackParam_GetDataIdx(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle); + +/** + * @brief Execute an input-output async PSNPE object + * + * @param[in] oacpHandle Handle to access the PSNPE object of input-output async mode + * + * @return True if executed successfully with input-output async mode + */ +SNPE_API +int Snpe_PSNPE_InputOutputAsyncCallbackParam_GetExecuteStatus(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle); + +/** + * @brief Get the error message during the execution of PSNPE input-output async mode + * + * @param[in] oacpHandle Handle to access the PSNPE object of input-output async mode + * + * @return error message + */ +SNPE_API +const char* Snpe_PSNPE_InputOutputAsyncCallbackParam_GetErrorMsg(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle); + +/** + * @brief Get the names of output buffers to the network + * + * @param[in] ioacpHandle Handle to access the PSNPE object of input-output async mode + * + * @return Handle of output buffer name list + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_PSNPE_InputOutputAsyncCallbackParam_GetUserBufferNames(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle); + +/** + * @brief Get the output buffer map of PSNPE object for input-output async mode + * + * @param[in] ioacpHandle Handle to access the PSNPE object of input-output async mode + * + * @return The reference handle of output ApplicationBufferMap + */ +SNPE_API +Snpe_ApplicationBufferMap_Handle_t Snpe_PSNPE_InputOutputAsyncCallbackParam_GetOutputMap_Ref(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle); + +/** + * @brief Get the id of the output callback for input-output async mode + * + * @param[in] oacpHandle Handle to access the PSNPE object of input-output async mode + * + * @return The id for output callback for input-output async mode + */ +SNPE_API +size_t Snpe_PSNPE_InputOutputAsyncCallbackParam_GetID(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle); + +/** + * A typedef to indicate the input callback of PSNPE handle of input-output async mode + */ +typedef void* Snpe_PSNPE_InputOutputInputAsyncCallbackParam_Handle_t; + +/** + * @brief Get the input list for input callback of input-output async mode + * + * @param[in] ioacpHandle Handle to access the object of input callback of input-output async mode + * + * @return List the inputs + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_PSNPE_InputOutputInputAsyncCallbackParam_GetInputs(Snpe_PSNPE_InputOutputInputAsyncCallbackParam_Handle_t ioiacpHandle); + +/** + * @brief Get the input names for input callback of input-output async mode + * + * @param[in] ioacpHandle Handle to access the object of input callback of input-output async mode + * + * @return List the names of input + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_PSNPE_InputOutputInputAsyncCallbackParam_GetInputNames(Snpe_PSNPE_InputOutputInputAsyncCallbackParam_Handle_t ioiacpHandle); + +/** + * @brief Get the id of the input callback for input-output async mode + * + * @param[in] oacpHandle Handle to access the object of input-output async mode + * + * @return The id of input callback for input-output async mode + */ +SNPE_API +size_t Snpe_PSNPE_InputOutputInputAsyncCallbackParam_GetID(Snpe_PSNPE_InputOutputInputAsyncCallbackParam_Handle_t ioiacpHandle); + +/** + * @brief A struct to indicate userbuffer data type in output callback of input-output async mode + */ +typedef struct{ + /// data for the one output + const uint8_t* data; + /// the data size of this output + size_t size; +} Snpe_UserBufferData_t; + +/** + * @brief Get the output data of the output callback for input-output async mode + * + * @param[in] oacpHandle Handle to access the object of output callback of input-output async mode + * + * @param[in] name The output name of output callback of input-output async mode + * + * @return The output data of output callback for input-output async mode + */ +SNPE_API +Snpe_UserBufferData_t Snpe_PSNPE_InputOutputAsyncCallbackParam_GetUserBuffer(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t ioacpHandle, + const char* name); +/** + * A typedef to indicate build configuration + */ +typedef void* Snpe_BuildConfig_Handle_t; + +/** + * A typedef to indicate a PSNPE object + */ +typedef void* Snpe_PSNPE_Handle_t; + +/** + * A typedef to indicate if PSNPE object is built in serial or parallel, default = 0 + */ +typedef enum SNPE_API { + SNPE_PSNPE_BUILDMODE_SERIAL = 0, + SNPE_PSNPE_BUILDMODE_PARALLEL = 1 +} Snpe_PSNPE_BuildMode_t; + +/** + * A typedef to indicate if PSNPE objects are executed in sync mode or output async mode or input-output async mode, default = 0 + */ +typedef enum SNPE_API { + SNPE_PSNPE_INPUTOUTPUTTRANSMISSIONMODE_SYNC = 0, + SNPE_PSNPE_INPUTOUTPUTTRANSMISSIONMODE_OUTPUTASYNC = 1, + SNPE_PSNPE_INPUTOUTPUTTRANSMISSIONMODE_INPUTOUTPUTASYNC = 2 +} Snpe_PSNPE_InputOutputTransmissionMode_t; + +// BuildConfig +/** + * @brief Create the object of snpe build config + * + * @return the SNPE build handle + */ +SNPE_API +Snpe_BuildConfig_Handle_t Snpe_BuildConfig_Create(); + +/** + * @brief Release the object of snpe build config + * + * @param[in] buildConfigHandle Handle to access the object of snpe buid config + * + * @return The error of build config result + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_Delete(Snpe_BuildConfig_Handle_t buildConfigHandle); + +/** + * @brief Get the mode of build snpe object, serial or parallel + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The value of Snpe_PSNPE_BuildMode_t + */ +SNPE_API +Snpe_PSNPE_BuildMode_t Snpe_BuildConfig_GetBuildMode(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set the mode of build snpe object, serial or parallel + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] buildMode A typedef of Snpe_PSNPE_BuildMode_t + * + * @return The result of setting mode + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetBuildMode(Snpe_BuildConfig_Handle_t bcHandle, Snpe_PSNPE_BuildMode_t buildMode); + +/** + * @brief Set the dlc model + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] dlcHandle A handle of snpe DLC container + * + * @return The result of setting dlc model + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetContainer(Snpe_BuildConfig_Handle_t bcHandle, Snpe_DlContainer_Handle_t dlcHandle); + +/** + * @brief Get dlc container in snpe build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The reference handle of DLC container + */ +SNPE_API +Snpe_DlContainer_Handle_t Snpe_BuildConfig_GetContainer_Ref(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set output buffer names in snpe build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] slHandle A handle of the output layer name list + * + * @return The result of setting output names + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetOutputBufferNames(Snpe_BuildConfig_Handle_t bcHandle, Snpe_StringList_Handle_t slHandle); + +/** + * @brief Get output buffer names in snpe build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The reference handle of output buffer name list. + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_BuildConfig_GetOutputBufferNames_Ref(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set output buffer names in snpe build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] slHandle List of tensor names to output. An empty list will result in producing output for the final output tensor of the model. The list will be copied + * + * @return The result of setting output tensors + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetOutputTensors(Snpe_BuildConfig_Handle_t bcHandle, Snpe_StringList_Handle_t slHandle); + +/** + * @brief Get output tensors in snpe build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The reference handle of output tensor list + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_BuildConfig_GetOutputTensors_Ref(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set runtime config list for snpe buildConfig + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] rclHandle Handle to access the object of runtime config list + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetRuntimeConfigList(Snpe_BuildConfig_Handle_t bcHandle, Snpe_RuntimeConfigList_Handle_t rclHandle); + +/** + * @brief Get runtime config list for snpe buildConfig + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The reference handle of runtime config list + */ +SNPE_API +Snpe_RuntimeConfigList_Handle_t Snpe_BuildConfig_GetRuntimeConfigList_Ref(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Get input thread number of input data for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The number of input thread + */ +SNPE_API +size_t Snpe_BuildConfig_GetInputThreadNumbers(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set input thread number of input data for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] threadNumbers The number of input thread for input-output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetInputThreadNumbers(Snpe_BuildConfig_Handle_t bcHandle, size_t threadNumbers); + +/** + * @brief Get output thread number of output data for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The number of output thread + */ +SNPE_API +size_t Snpe_BuildConfig_GetOutputThreadNumbers(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set output thread number of output data for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] threadNumbers The number of output thread for input-output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetOutputThreadNumbers(Snpe_BuildConfig_Handle_t bcHandle, size_t threadNumbers); + +/** + * @brief Set output callback for output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] callbackFunc The ouutput callback function for output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetOutputCallback(Snpe_BuildConfig_Handle_t bcHandle, + void (*callbackFunc)(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t)); +/** + * @brief Set the id of output callback function for output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] id The id of output callback function + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetOutputCallbackID(Snpe_BuildConfig_Handle_t bcHandle, size_t id); + +/** + * @brief Set the inside output callback handle to NULL for output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_ClearOutputCallback(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set output callback for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] callbackFunc The output callback function for input-output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetInputOutputCallback(Snpe_BuildConfig_Handle_t bcHandle, + void (*callbackFunc)(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t)); + +/** + * @brief Set the id of output callback function for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] id The id of output callback function for input-output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetInputOutputCallbackID(Snpe_BuildConfig_Handle_t bcHandle, size_t id); + +/** + * @brief Set the inside output callback handle to NULL for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_ClearInputOutputCallback(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set input callback for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] callbackFunc The input callback function for input-output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetInputOutputInputCallback(Snpe_BuildConfig_Handle_t bcHandle, + Snpe_ApplicationBufferMap_Handle_t (*callbackFunc)( + Snpe_PSNPE_InputOutputInputAsyncCallbackParam_Handle_t + ) + ); + +/** + * @brief Set the id of input callback function for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] id The id of input callback function for input-output async mode + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetInputOutputInputCallbackID(Snpe_BuildConfig_Handle_t bcHandle, size_t id); + +/** + * @brief Set the inside input callback handle to NULL for input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_ClearInputOutputInputCallback(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set the input and output transmission mode including sync mode, output async mode and input-output async mode, defult is sync mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] iotMode The typedef of Snpe_PSNPE_InputOutputTransmissionMode_t + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetInputOutputTransmissionMode(Snpe_BuildConfig_Handle_t bcHandle, + Snpe_PSNPE_InputOutputTransmissionMode_t iotMode); + +/** + * @brief Get the input and output transmission mode including sync mode, output async mode and input-output async mode + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The typedef of Snpe_PSNPE_InputOutputTransmissionMode_t + */ +SNPE_API +Snpe_PSNPE_InputOutputTransmissionMode_t Snpe_BuildConfig_GetInputOutputTransmissionMode(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set the profiling level for PSNPE build config, default is SNPE_PROFILING_LEVEL_OFF + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] profilingLevel The typedef of Snpe_ProfilingLevel_t + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetProfilingLevel(Snpe_BuildConfig_Handle_t bcHandle, Snpe_ProfilingLevel_t profilingLevel); + +/** + * @brief Get the profiling level for PSNPE build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The typedef of Snpe_ProfilingLevel_t + */ +SNPE_API +Snpe_ProfilingLevel_t Snpe_BuildConfig_GetProfilingLevel(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief To be deprecated, set the encode value when you want to divide one image into 2 or 4 parts to run, default is 0 which means the input don't need dividing. + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] encode0 The uint64 value of encode0 + * + * @param[in] encode1 The uint64 value of encode1 + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetEncode(Snpe_BuildConfig_Handle_t bcHandle, uint64_t encode0, uint64_t encode1); + +/** + * @brief To be deprecated, set the encode0 value for snpe build config which is a special feature used in SM8250 + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] encode0 The uint64 value of encode0 + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetEncode0(Snpe_BuildConfig_Handle_t bcHandle, uint64_t encode0); + +/** + * @brief To be deprecated, set the encode1 value for snpe build config which is a special feature used in SM8250 + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] encode1 The uint64 value of encode1 + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetEncode1(Snpe_BuildConfig_Handle_t bcHandle, uint64_t encode1); + +/** + * @brief To be deprecated, get the encode0 and encode1 value for snpe build config which is a special feature used in SM8250 + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The uint64 value of encode + */ +SNPE_API +uint64_t* Snpe_BuildConfig_GetEncode(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief To be deprecated, get the encode0 value for snpe build config which is a special feature used in SM8250 + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The uint64 value of encode0 + */ +SNPE_API +uint64_t Snpe_BuildConfig_GetEncode0(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief To be deprecated, get the encode1 value for snpe build config which is a special feature used in SM8250 + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The uint64 value of encode1 + */ +SNPE_API +uint64_t Snpe_BuildConfig_GetEncode1(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set true or false for enabling init cache for snpe build config, enabling init cache = 1 + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] enableInitCache True for enabing init cache + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetEnableInitCache(Snpe_BuildConfig_Handle_t bcHandle, int enableInitCache); + +/** + * @brief Get the satus of enabling init cache for snpe build config, enabling init cache = 1. + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] enableInitCache True for enabing init cache + * + * @return 1 or 0 for enabling init cache + */ +SNPE_API +int Snpe_BuildConfig_GetEnableInitCache(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Handle needed to access the platformConfig. + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] platformOptions Options as a const char* + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetPlatformOptions(Snpe_BuildConfig_Handle_t bcHandle, const char* platformOptions); + +/** + * @brief Get the optional platform features for snpe build config + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return Options as a const char* + */ +SNPE_API +const char* Snpe_BuildConfig_GetPlatformOptions(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Set the path directory of output diag log you want to save + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @param[in] diaglogOutputDir The string directory + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_BuildConfig_SetDiaglogOutputDir(Snpe_BuildConfig_Handle_t bcHandle, const char* diaglogOutputDir); + +/** + * @brief Get the path of output diag log + * + * @param[in] bcHandle Handle to access the object of snpe buid config + * + * @return The string directory + */ +SNPE_API +const char* Snpe_BuildConfig_GetDiaglogOutputDir(Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Create the handle of PSNPE object + * + * @return The handle of PSNPE object + */ +SNPE_API +Snpe_PSNPE_Handle_t Snpe_PSNPE_Create(); + +/** + * @brief Release the handle of PSNPE object + * + * @param[in] psnpeHandle Handle to access the PSNPE object + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_Delete(Snpe_PSNPE_Handle_t psnpeHandle); + +/** + * @brief Build the instance of PSNPE object accorading of snpe build config + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_Build(Snpe_PSNPE_Handle_t psnpeHandle, Snpe_BuildConfig_Handle_t bcHandle); + +/** + * @brief Execute PSNPE object for sync mode. + * + * @param[in] psnpeHandle Handle to access the PSNPE object + * + * @param[in] inputBufferListHandle Handle to access the input user buffer list + * + * @param[in] outputBufferListHandle Handle to access the output user buffer list + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_Execute(Snpe_PSNPE_Handle_t psnpeHandle, + Snpe_UserBufferList_Handle_t inputBufferListHandle, + Snpe_UserBufferList_Handle_t outputBufferListHandle); + +/** + * @brief Execute PSNPE object for input-output async mode + * + * @param[in] psnpeHandle Handle to access the PSNPE object + * + * @param[in] inputMapHandle Handle to access the input buffer map + * + * @param[in] dataIndex The index of input data + * + * @param[in] isTF8buff If the input buffer is TF8 + * + * @param[in] isTF8Outputbuff If the output buffer is TF8 + * + * @return The result error message + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_ExecuteInputOutputAsync(Snpe_PSNPE_Handle_t psnpeHandle, + Snpe_StringList_Handle_t inputMapHandle, + size_t dataIndex, + int isTF8buff, + int isTF8Outputbuff); + +/** + * @brief Get the input tensor names for PSNPE object. + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @return The string list of input tensor names + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_PSNPE_GetInputTensorNames(Snpe_PSNPE_Handle_t psnpeHandle); + +/** + * @brief Get the output tensor names for PSNPE object + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @return The string list of output tensor names + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_PSNPE_GetOutputTensorNames(Snpe_PSNPE_Handle_t psnpeHandle); + +/** + * @brief Get the input dimension shape for PSNPE object + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @return The tensor shape of input dimension + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_PSNPE_GetInputDimensions(Snpe_PSNPE_Handle_t psnpeHandle); + +/** + * @brief Get the input dimension shape for the specific input name for PSNPE object + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @param[in] name The name of input data + * + * @return The tensor shape of a specific input name + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_PSNPE_GetInputDimensions_Name(Snpe_PSNPE_Handle_t psnpeHandle, const char* name); + +/** + * @brief Get the number of elements in each dimension for input and output buffer + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @param[in] name The name of input and output buffer + * + * @return Dimension size + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_PSNPE_GetBufferAttributesDims(Snpe_PSNPE_Handle_t psnpeHandle, const char* name); + +/* To be deprecated, please use new api Snpe_PSNPE_RegisterUserMemoryMappedBuffers */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_RegisterIonBuffers(Snpe_PSNPE_Handle_t psnpeHandle, Snpe_UserMemoryMap_Handle_t ionBufferMapHandle); + +/* To be deprecated, please use new api Snpe_PSNPE_DeregisterUserMemoryMappedBuffers */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_DeregisterIonBuffers(Snpe_PSNPE_Handle_t psnpeHandle, Snpe_StringList_Handle_t ionBufferNames); + +/** + * @brief Register Client Memory-Mapped Buffers (Example ION buffers in Android) + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] bufferMapHandle A UserMemoryMap of virtual addresses + * + * @note UserBuffer type passed for registration must match the data type of the tensor in the dlc + * For regular UserBuffers SNPE performs an online data conversion (quantization or + * dequantization etc). This is not possible for memory mapped buffers hence can lead to + * issues during execution or accuracy degradation + * + * @return SNPE_SUCCESS upon successful memory mapped buffer registration + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_RegisterUserMemoryMappedBuffers(Snpe_PSNPE_Handle_t psnpeHandle, Snpe_UserMemoryMap_Handle_t bufferMapHandle); + +/** + * @brief Deregister Client Memory-Mapped Buffers (Example ION buffers in Android) + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] bufferNamesHandle A StringList of memory mapped buffer names + * + * @return SNPE_SUCCESS upon successful memory mapped buffer deregistration + */ +SNPE_API +Snpe_ErrorCode_t Snpe_PSNPE_DeregisterUserMemoryMappedBuffers(Snpe_PSNPE_Handle_t psnpeHandle, Snpe_StringList_Handle_t bufferNamesHandle); + +/** + * @brief Get the error message during the failed execution + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @return The error message + */ +SNPE_API +const char* Snpe_PSNPE_GetLastErrorString(Snpe_PSNPE_Handle_t psnpeHandle); + +/** + * @brief Get the handle of IBufferAttributes + * + * @param[in] bcHandle Handle to access the PSNPE object + * + * @param[in] name The name of attribute buffer + * + * @return Handle to access IBufferAttributes + */ +SNPE_API +Snpe_IBufferAttributes_Handle_t Snpe_PSNPE_GetInputOutputBufferAttributes(Snpe_PSNPE_Handle_t psnpeHandle, const char *name); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_PSNPE_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.hpp new file mode 100644 index 00000000..bd3af1ac --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/PSNPE.hpp @@ -0,0 +1,537 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include +#include +#include +#include +#include + + +#include "Wrapper.hpp" + + +#include "DlSystem/DlEnums.hpp" +#include "DlSystem/DlVersion.hpp" +#include "DlSystem/StringList.hpp" +#include "DlSystem/DlOptional.hpp" +#include "DlSystem/IBufferAttributes.hpp" +#include "DlSystem/UserMemoryMap.hpp" + +#include "SNPE/UserBufferList.hpp" +#include "SNPE/ApplicationBufferMap.hpp" +#include "SNPE/RuntimeConfigList.hpp" +#include "DlContainer/IDlContainer.hpp" + +#include "SNPE/RuntimeConfigList.hpp" + + +#include "SNPE/PSNPE.h" + +namespace PSNPE{ + +enum BuildMode { + SERIAL = 0, + PARALLEL = 1 +}; +/** + * @brief Input and output transmission mode + */ +enum InputOutputTransmissionMode { + sync = 0, + outputAsync = 1, + inputOutputAsync = 2 +}; + + +struct OutputAsyncCallbackParam : public Wrapper { +private: + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{NoOpDeleter}; + + + template + using DataIndexReference = WrapperDetail::GenericConstMemberReference + ; + + + template + using ExecuteStatusReference = WrapperDetail::GenericConstMemberReference + >; + + + static std::string ErrMsgGetter(Snpe_DlVersion_Handle_t handle){ + return Snpe_PSNPE_OutputAsyncCallbackParam_GetErrorMsg(handle); + } + template + using ErrorMsgReference = WrapperDetail::GenericConstMemberReference + ; + + template + using CallbackIDReference = WrapperDetail::GenericConstMemberReference + ; + + + + +public: + OutputAsyncCallbackParam() = delete; + OutputAsyncCallbackParam(OutputAsyncCallbackParam&& other) noexcept + : BaseType(std::move(other)) + { } + + DataIndexReference dataIndex{*this}; + ExecuteStatusReference executeStatus{*this}; + ErrorMsgReference errorMsg{*this}; + + CallbackIDReference callbackID{*this}; +}; + + + +struct InputOutputInputAsyncCallbackParam : public Wrapper { +private: + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{NoOpDeleter}; + + + static std::vector GetInputs(HandleType handle){ + DlSystem::StringList inputs(moveHandle(Snpe_PSNPE_InputOutputInputAsyncCallbackParam_GetInputs(handle))); + + return std::vector(inputs.begin(), inputs.end()); + } + + template + using InputsReference = WrapperDetail::GenericConstMemberReference + ; + + + static DlSystem::StringList GetInputNames(HandleType handle){ + return moveHandle(Snpe_PSNPE_InputOutputInputAsyncCallbackParam_GetInputNames(handle)); + } + template + using InputNamesReference = WrapperDetail::GenericConstMemberReference + ; + + template + using CallbackIDReference = WrapperDetail::GenericConstMemberReference + ; + + +public: + InputOutputInputAsyncCallbackParam() = delete; + InputOutputInputAsyncCallbackParam(InputOutputInputAsyncCallbackParam&& other) noexcept + : BaseType(std::move(other)) + { } + + InputsReference> inputs{*this}; + InputNamesReference inputNames{*this}; + CallbackIDReference callbackID{*this}; + +}; + + + + + +struct InputOutputAsyncCallbackParam : public Wrapper { +private: + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{NoOpDeleter}; + + template + using DataIndexReference = WrapperDetail::GenericConstMemberReference + ; + + static bool GetExecuteStatus(HandleType handle){ + return Snpe_PSNPE_InputOutputAsyncCallbackParam_GetExecuteStatus(handle); + } + template + using ExecuteStatusReference = WrapperDetail::GenericConstMemberReference + ; + + static std::string ErrMsgGetter(Snpe_DlVersion_Handle_t handle){ + return Snpe_PSNPE_OutputAsyncCallbackParam_GetErrorMsg(handle); + } + template + using ErrorMsgReference = WrapperDetail::GenericConstMemberReference + ; + + + + // This should work + static ApplicationBufferMap GetOutputMap(HandleType handle){ + return moveHandle(Snpe_PSNPE_InputOutputAsyncCallbackParam_GetOutputMap_Ref(handle), true); + } + + template + using OutputMapReference = WrapperDetail::GenericConstMemberReference + ; + + template + using CallbackIDReference = WrapperDetail::GenericConstMemberReference + ; + +public: + + InputOutputAsyncCallbackParam(InputOutputAsyncCallbackParam&& other) noexcept + : BaseType(std::move(other)) + { } + + DataIndexReference dataIndex{*this}; + OutputMapReference outputMap{*this}; /// OOOH, this will be super tricky to not have a copy every time + ExecuteStatusReference executeStatus{*this}; + ErrorMsgReference errorMsg{*this}; + CallbackIDReference callbackID{*this}; +}; + +/** + * @brief This callback is called when the output data is ready, only use for Output Async mode + */ +using OutputAsyncCallbackFunc = std::function; +/** + * @brief This callback is called when the output data is ready, only use for Output-Input Async mode + */ +using InputOutputAsyncCallbackFunc = std::function; +/** + * @brief This callback is called when the input data is ready,only use for Output-Input Async mode + */ +using InputOutputAsyncInputCallback = std::function(InputOutputInputAsyncCallbackParam)>; + + +struct BuildConfig final { + BuildMode buildMode = BuildMode::SERIAL; ///< Specify build in serial mode or parallel mode + zdl::DlContainer::IDlContainer* container;///< The opened container ptr + zdl::DlSystem::StringList outputBufferNames;///< Specify the output layer name + zdl::DlSystem::StringList outputTensors;///< Specify the output layer name + RuntimeConfigList runtimeConfigList;///< The runtime config list for PSNPE, @see RuntimeConfig + size_t inputThreadNumbers = 1;///< Specify the number of threads used in the execution phase to process input data, only used in inputOutputAsync mode + size_t outputThreadNumbers = 1;///< Specify the number of threads used in the execution phase to process output data, only used in inputOutputAsync and outputAsync mode + OutputAsyncCallbackFunc outputCallback;///< The callback to deal with output data ,only used in outputAsync mode + InputOutputAsyncCallbackFunc inputOutputCallback;///< The callback to deal with output data ,only used in inputOutputAsync mode + InputOutputAsyncInputCallback inputOutputInputCallback;///< The callback to deal with input data ,only used in inputOutputAsync mode + InputOutputTransmissionMode inputOutputTransmissionMode = InputOutputTransmissionMode::sync;///< Specify execution mode + zdl::DlSystem::ProfilingLevel_t profilingLevel = zdl::DlSystem::ProfilingLevel_t::OFF;///< Specify profiling level for Diaglog + uint64_t encode[2] = {0, 0}; + bool enableInitCache = false; + std::string platformOptions; + std::string diaglogOutputDir = "./diaglogs/"; ///< Specify a diaglog output directory to save the generated Diaglog files. + + size_t callbackID{}; +}; + + + + + +class PSNPE : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_PSNPE_Delete}; +// struct BuildConfigInternal : public Wrapper{ +// +// }; +public: + PSNPE() + : BaseType(Snpe_PSNPE_Create()) + { } + +private: + + template + static std::unordered_map& getCallbackMap(){ + static std::unordered_map toret; + return toret; + } + template + static std::mutex& getCallbackMapMutex(){ + static std::mutex mtx; + return mtx; + } + + static void outputCallbackTrampoline(Snpe_PSNPE_OutputAsyncCallbackParam_Handle_t paramHandle){ + OutputAsyncCallbackParam param(moveHandle(paramHandle)); + std::function callback; + { + std::lock_guard lk(getCallbackMapMutex()); + callback = getCallbackMap()[param.callbackID]; + } + callback(std::move(param)); + } + static void inputOutputCallbackTrampoline(Snpe_PSNPE_InputOutputAsyncCallbackParam_Handle_t paramHandle){ + InputOutputAsyncCallbackParam param(moveHandle(paramHandle)); + std::function callback; + { + std::lock_guard lk(getCallbackMapMutex()); + callback = getCallbackMap()[param.callbackID]; + } + callback(std::move(param)); + } + + static Snpe_ApplicationBufferMap_Handle_t inputOutputInputCallbackTrampoline( + Snpe_PSNPE_InputOutputInputAsyncCallbackParam_Handle_t paramHandle + ){ + InputOutputInputAsyncCallbackParam param(moveHandle(paramHandle)); + + std::function(InputOutputInputAsyncCallbackParam)> callback; + { + std::lock_guard lk(getCallbackMapMutex()); + callback = getCallbackMap()[param.callbackID]; + } + auto abm = callback(std::move(param)); + return WrapperDetail::HandleReleaser::release(*abm); + } + + template + class CallbackIdManager{ + public: + ~CallbackIdManager(){ + clear(); + } + std::pair registerCallback(WrapperCallbackType func){ + size_t id = get(); + + std::lock_guard lk(getCallbackMapMutex()); + getCallbackMap()[id] = std::move(func); + return {id, CapiCallback}; + } + private: + size_t m_CallbackId{}; + + void clear(){ + if(m_CallbackId){ + std::lock_guard lk(getCallbackMapMutex()); + getCallbackMap().erase(m_CallbackId); + } + } + + size_t get(){ + static std::atomic id{0}; + clear(); + m_CallbackId = ++id; + return m_CallbackId; + } + + }; + CallbackIdManager outputCallbackIdManager; + + CallbackIdManager inputOutputCallbackIdManager; + + CallbackIdManager inputOutputInputCallbackIdManager; + + +public: + + + + bool build(BuildConfig& buildConfig) noexcept{ + // Copy the BuildConfig across the CAPI boundary + + Snpe_BuildConfig_Handle_t bcHandle = Snpe_BuildConfig_Create(); + + Snpe_BuildConfig_SetBuildMode(bcHandle, static_cast(buildConfig.buildMode)); + Snpe_BuildConfig_SetContainer(bcHandle, getHandle(buildConfig.container)); + Snpe_BuildConfig_SetOutputBufferNames(bcHandle, getHandle(buildConfig.outputBufferNames)); + Snpe_BuildConfig_SetOutputTensors(bcHandle, getHandle(buildConfig.outputTensors)); + Snpe_BuildConfig_SetRuntimeConfigList(bcHandle, getHandle(buildConfig.runtimeConfigList)); + + Snpe_BuildConfig_SetInputThreadNumbers(bcHandle, buildConfig.inputThreadNumbers); + Snpe_BuildConfig_SetOutputThreadNumbers(bcHandle, buildConfig.outputThreadNumbers); + + + if(buildConfig.outputCallback){ + auto id_callback = outputCallbackIdManager.registerCallback(buildConfig.outputCallback); + Snpe_BuildConfig_SetOutputCallbackID(bcHandle, id_callback.first); + Snpe_BuildConfig_SetOutputCallback(bcHandle, id_callback.second); + } + + if(buildConfig.inputOutputCallback){ + auto id_callback = inputOutputCallbackIdManager.registerCallback(buildConfig.inputOutputCallback); + Snpe_BuildConfig_SetInputOutputCallbackID(bcHandle, id_callback.first); + Snpe_BuildConfig_SetInputOutputCallback(bcHandle, id_callback.second); + } + + if(buildConfig.inputOutputInputCallback){ + auto id_callback = inputOutputInputCallbackIdManager.registerCallback(buildConfig.inputOutputInputCallback); + Snpe_BuildConfig_SetInputOutputInputCallbackID(bcHandle, id_callback.first); + Snpe_BuildConfig_SetInputOutputInputCallback(bcHandle, id_callback.second); + } + + + Snpe_BuildConfig_SetInputOutputTransmissionMode(bcHandle, + static_cast(buildConfig.inputOutputTransmissionMode)); + + Snpe_BuildConfig_SetProfilingLevel(bcHandle, static_cast(buildConfig.profilingLevel)); + Snpe_BuildConfig_SetEncode(bcHandle, buildConfig.encode[0], buildConfig.encode[1]); + Snpe_BuildConfig_SetEnableInitCache(bcHandle, buildConfig.enableInitCache); + Snpe_BuildConfig_SetPlatformOptions(bcHandle, buildConfig.platformOptions.c_str()); + Snpe_BuildConfig_SetDiaglogOutputDir(bcHandle, buildConfig.diaglogOutputDir.c_str()); + + + auto status = Snpe_PSNPE_Build(handle(), bcHandle); + Snpe_BuildConfig_Delete(bcHandle); + + + return status == SNPE_SUCCESS; + } + + /** + * @brief Execute snpe instances in Async Output mode and Sync mode + * + * @param[in] inputBufferList A list of user buffers that contains the input data + * + * @param[in,out] outputBufferList A list of user buffers that will hold the output data + * + */ + bool execute(UserBufferList& inputBufferList, UserBufferList& outputBufferList) noexcept{ + return SNPE_SUCCESS == Snpe_PSNPE_Execute(handle(), getHandle(inputBufferList), getHandle(outputBufferList)); + } + + /** + * @brief Execute snpe instances in Async Input/Output mode + * + * @param[in]inputMap A map of input buffers that contains input data. The names of buffers + * need to be matched with names retrived through getInputTensorNames() + * + * @param dataIndex Index of the input data + * + * @param isTF8buff Whether prefer to using 8 bit quantized element for inference + * + * @return True if executed successfully; flase, otherwise. + */ + bool executeInputOutputAsync(const DlSystem::StringList& inputMap, size_t dataIndex, bool isTF8buff, bool isTF8Outputbuff) noexcept{ + return SNPE_SUCCESS == Snpe_PSNPE_ExecuteInputOutputAsync(handle(), getHandle(inputMap), dataIndex, isTF8buff, isTF8Outputbuff); + } + bool executeInputOutputAsync(const std::vector& inputMap, size_t dataIndex, bool isTF8buff, bool isTF8Outputbuff) noexcept{ + DlSystem::StringList sl(inputMap.size()); + for(auto&& e : inputMap) sl.append(e.c_str()); + return executeInputOutputAsync(sl, dataIndex, isTF8buff, isTF8Outputbuff); + } + + bool executeInputOutputAsync(const DlSystem::StringList& inputMap, size_t dataIndex, bool isTF8buff) noexcept{ + return executeInputOutputAsync(inputMap, dataIndex, isTF8buff, isTF8buff); + } + bool executeInputOutputAsync(const std::vector& inputMap, size_t dataIndex, bool isTF8buff) noexcept{ + return executeInputOutputAsync(inputMap, dataIndex, isTF8buff, isTF8buff); + } + + + + /** + * @brief Returns the input layer names of the network. + * + * @return StringList which contains the input layer names + */ + const DlSystem::StringList getInputTensorNames() const noexcept{ + return moveHandle(Snpe_PSNPE_GetInputTensorNames(handle())); + } + + /** + * @brief Returns the output layer names of the network. + * + * @return StringList which contains the output layer names + */ + const DlSystem::StringList getOutputTensorNames() const noexcept{ + return moveHandle(Snpe_PSNPE_GetOutputTensorNames(handle())); + } + + /** + * @brief Returns the input tensor dimensions of the network. + * + * @return TensorShape which contains the dimensions. + */ + const DlSystem::TensorShape getInputDimensions() const noexcept{ + return moveHandle(Snpe_PSNPE_GetInputDimensions(handle())); + } + + const zdl::DlSystem::TensorShape getInputDimensions(const char *name) const noexcept{ + return moveHandle(Snpe_PSNPE_GetInputDimensions_Name(handle(), name)); + } + + /** + * @brief Returns attributes of buffers. + * + * @see zdl::SNPE + * + * @return BufferAttributes of input/output tensor named. + */ + zdl::DlSystem::TensorShape getBufferAttributesDims(const char *name) const noexcept{ + return moveHandle(Snpe_PSNPE_GetBufferAttributesDims(handle(), name)); + } + + DlSystem::Optional getInputOutputBufferAttributes(const char *name) const noexcept{ + return { + new DlSystem::IBufferAttributes(moveHandle(Snpe_PSNPE_GetInputOutputBufferAttributes(handle(), name))), + DlSystem::Optional::LIFECYCLE::POINTER_OWNED + }; + } + /* To be deprecated, please use new api registerMemoryMappedBuffers */ + bool registerIonBuffers(const DlSystem::UserMemoryMap& ionBufferMap) const noexcept{ + return SNPE_SUCCESS == Snpe_PSNPE_RegisterIonBuffers(handle(), getHandle(ionBufferMap)); + } + /* To be deprecated, please use new api deregisterMemoryMappedBuffers */ + bool deregisterIonBuffers(const DlSystem::StringList& ionBufferNames) const noexcept{ + return SNPE_SUCCESS == Snpe_PSNPE_DeregisterIonBuffers(handle(), getHandle(ionBufferNames)); + } + + bool registerMemoryMappedBuffers(const DlSystem::UserMemoryMap& memoryMappedBufferMap) noexcept{ + return SNPE_SUCCESS == Snpe_PSNPE_RegisterUserMemoryMappedBuffers(handle(), getHandle(memoryMappedBufferMap)); + } + + bool deregisterMemoryMappedBuffers(const DlSystem::StringList& bufferNames) noexcept{ + return SNPE_SUCCESS == Snpe_PSNPE_DeregisterUserMemoryMappedBuffers(handle(), getHandle(bufferNames)); + } + + const char* getLastErrorString(){ + return Snpe_PSNPE_GetLastErrorString(handle()); + } + +private: + PSNPE(const PSNPE&) = delete; + PSNPE& operator=(const PSNPE&) = delete; + +}; + +} // ns PSNPE + + + +ALIAS_IN_ZDL_NAMESPACE(PSNPE, BuildMode) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, InputOutputTransmissionMode) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, OutputAsyncCallbackParam) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, InputOutputAsyncCallbackParam) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, InputOutputInputAsyncCallbackParam) + +ALIAS_IN_ZDL_NAMESPACE(PSNPE, OutputAsyncCallbackFunc) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, InputOutputAsyncCallbackFunc) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, InputOutputAsyncInputCallback) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, BuildConfig) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, PSNPE) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.h new file mode 100644 index 00000000..59295d59 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.h @@ -0,0 +1,118 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef _SNPE_RUNTIME_CONFIG_LIST_H_ +#define _SNPE_RUNTIME_CONFIG_LIST_H_ + + +#ifdef __cplusplus +#include +#else +#include +#endif + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" + +#include "DlSystem/DlEnums.h" +#include "DlSystem/RuntimeList.h" +#include "DlSystem/TensorShapeMap.h" + + +#ifdef __cplusplus +extern "C" { +#endif + +typedef void* Snpe_RuntimeConfig_Handle_t; + +SNPE_API +Snpe_RuntimeConfig_Handle_t Snpe_RuntimeConfig_Create(); + +SNPE_API +Snpe_RuntimeConfig_Handle_t Snpe_RuntimeConfig_CreateCopy(Snpe_RuntimeConfig_Handle_t rcHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfig_Delete(Snpe_RuntimeConfig_Handle_t rcHandle); + + +SNPE_API +Snpe_Runtime_t Snpe_RuntimeConfig_GetRuntime(Snpe_RuntimeConfig_Handle_t rcHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfig_SetRuntime(Snpe_RuntimeConfig_Handle_t rcHandle, Snpe_Runtime_t runtime); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfig_SetRuntimeList(Snpe_RuntimeConfig_Handle_t rcHandle, Snpe_RuntimeList_Handle_t rlHandle); + +SNPE_API +Snpe_RuntimeList_Handle_t Snpe_RuntimeConfig_GetRuntimeList_Ref(Snpe_RuntimeConfig_Handle_t rcHandle); + +SNPE_API +Snpe_PerformanceProfile_t Snpe_RuntimeConfig_GetPerformanceProfile(Snpe_RuntimeConfig_Handle_t rcHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfig_SetPerformanceProfile(Snpe_RuntimeConfig_Handle_t rcHandle, Snpe_PerformanceProfile_t perfProfile); + +SNPE_API +int Snpe_RuntimeConfig_GetEnableCPUFallback(Snpe_RuntimeConfig_Handle_t rcHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfig_SetEnableCPUFallback(Snpe_RuntimeConfig_Handle_t rcHandle, int enableCpuFallback); + + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfig_SetInputDimensionsMap(Snpe_RuntimeConfig_Handle_t rcHandle, Snpe_TensorShapeMap_Handle_t tsmHandle); + +SNPE_API +Snpe_TensorShapeMap_Handle_t Snpe_RuntimeConfig_GetInputDimensionsMap_Ref(Snpe_RuntimeConfig_Handle_t rcHandle); + + + +typedef void* Snpe_RuntimeConfigList_Handle_t; + +SNPE_API +Snpe_RuntimeConfigList_Handle_t Snpe_RuntimeConfigList_Create(); + +SNPE_API +Snpe_RuntimeConfigList_Handle_t Snpe_RuntimeConfigList_CreateSize(size_t size); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfigList_Delete(Snpe_RuntimeConfigList_Handle_t rclHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfigList_PushBack(Snpe_RuntimeConfigList_Handle_t rclHandle, Snpe_RuntimeConfig_Handle_t rcHandle); + +SNPE_API +Snpe_RuntimeConfig_Handle_t Snpe_RuntimeConfigList_At_Ref(Snpe_RuntimeConfigList_Handle_t rclHandle, size_t idx); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfigList_Assign(Snpe_RuntimeConfigList_Handle_t rclSrcHandle, Snpe_RuntimeConfigList_Handle_t rclDstHandle); + +SNPE_API +size_t Snpe_RuntimeConfigList_Size(Snpe_RuntimeConfigList_Handle_t rclHandle); + +SNPE_API +size_t Snpe_RuntimeConfigList_Capacity(Snpe_RuntimeConfigList_Handle_t rclHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_RuntimeConfigList_Clear(Snpe_RuntimeConfigList_Handle_t rclHandle); + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_RUNTIME_CONFIG_LIST_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.hpp new file mode 100644 index 00000000..faf052c5 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/RuntimeConfigList.hpp @@ -0,0 +1,153 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + + +#include "DlSystem/DlEnums.hpp" +#include "DlSystem/RuntimeList.hpp" +#include "DlSystem/TensorShapeMap.hpp" + + +#include "SNPE/RuntimeConfigList.h" + +namespace PSNPE { + + + +struct RuntimeConfig : public Wrapper { +private: + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_RuntimeConfig_Delete}; + + template + using RuntimeReference = WrapperDetail::GenericMemberReference + , + CastingSetter >; + + + template + using RuntimeListReference = WrapperMemberReference< + RuntimeListType, + Snpe_RuntimeList_Handle_t, + Snpe_RuntimeConfig_GetRuntimeList_Ref, + Snpe_RuntimeConfig_SetRuntimeList + >; + + template + using InputDimensionsMapReference = WrapperMemberReference< + InputDimensionsMapType, + Snpe_TensorShapeMap_Handle_t, + Snpe_RuntimeConfig_GetInputDimensionsMap_Ref, + Snpe_RuntimeConfig_SetInputDimensionsMap + >; + + template + using PerfProfileReference = WrapperDetail::GenericMemberReference + , + CastingSetter >; + + template + using EnableCPUFallbackReference = WrapperDetail::GenericMemberReference + , + CastingSetter >; + +public: + RuntimeConfig() + : BaseType(Snpe_RuntimeConfig_Create()) + { } + RuntimeConfig(const RuntimeConfig& other) + : BaseType(Snpe_RuntimeConfig_CreateCopy(other.handle())) + { } + + RuntimeConfig(RuntimeConfig&& other) noexcept + : BaseType(std::move(other)) + { } + + RuntimeConfig& operator=(RuntimeConfig&& other) noexcept{ + return moveAssign(std::move(other)); + } + + + RuntimeReference runtime{*this, DlSystem::Runtime_t::CPU_FLOAT32}; + RuntimeListReference runtimeList{*this}; + PerfProfileReference perfProfile{*this, DlSystem::PerformanceProfile_t::HIGH_PERFORMANCE}; + InputDimensionsMapReference inputDimensionsMap{*this}; + EnableCPUFallbackReference enableCPUFallback{*this, false}; + +}; + + +class RuntimeConfigList : public Wrapper { +private: + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_RuntimeConfigList_Delete}; + +public: + RuntimeConfigList() + : BaseType(Snpe_RuntimeConfigList_Create()) + { } + RuntimeConfigList(size_t size) + : BaseType(Snpe_RuntimeConfigList_CreateSize(size)) + { } + + RuntimeConfigList(RuntimeConfigList&& other) noexcept + : BaseType(std::move(other)) + { } + + RuntimeConfigList& operator=(RuntimeConfigList&& other) noexcept{ + return moveAssign(std::move(other)); + } + RuntimeConfigList& operator=(const RuntimeConfigList& other){ + Snpe_RuntimeConfigList_Assign(other.handle(), handle()); + return *this; + } + + + + void push_back(const RuntimeConfig& runtimeConfig){ + Snpe_RuntimeConfigList_PushBack(handle(), getHandle(runtimeConfig)); + } + + RuntimeConfig& operator[](size_t index){ + return *makeReference(Snpe_RuntimeConfigList_At_Ref(handle(), index)); + } + const RuntimeConfig& operator[](size_t index) const{ + return *makeReference(Snpe_RuntimeConfigList_At_Ref(handle(), index)); + } + + size_t size() const noexcept{ + return Snpe_RuntimeConfigList_Size(handle()); + } + size_t capacity() const noexcept{ + return Snpe_RuntimeConfigList_Capacity(handle()); + } + + void clear() noexcept{ + Snpe_RuntimeConfigList_Clear(handle()); + } + +}; + +} // ns PSNPE + + +ALIAS_IN_ZDL_NAMESPACE(PSNPE, RuntimeConfig) +ALIAS_IN_ZDL_NAMESPACE(PSNPE, RuntimeConfigList) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.h new file mode 100644 index 00000000..eb05473a --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.h @@ -0,0 +1,336 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================= +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= + +/** + * @file + */ + +#ifndef _SNPE_SNPE_H_ +#define _SNPE_SNPE_H_ + + +#include "DlSystem/IBufferAttributes.h" +#include "DlSystem/ITensor.h" +#include "DlSystem/TensorShape.h" +#include "DlSystem/TensorMap.h" +#include "DlSystem/StringList.h" +#include "DlSystem/IUserBuffer.h" +#include "DlSystem/UserBufferMap.h" +#include "DlSystem/UserMemoryMap.h" +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" + +#include "DiagLog/IDiagLog.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * A typedef to indicate a SNPE handle + */ +typedef void* Snpe_SNPE_Handle_t; + +/** + * Destroys/frees a SNPE object + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_Delete(Snpe_SNPE_Handle_t snpeHandle); + +/** + * @brief Gets the names of input tensors to the network + * + * To support multiple input scenarios, where multiple tensors are + * passed through execute() in a TensorMap, each tensor needs to + * be uniquely named. The names of tensors can be retrieved + * through this function. + * + * In the case of a single input, one name will be returned. + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @return A StringList of input tensor names. + * + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_SNPE_GetInputTensorNames(Snpe_SNPE_Handle_t snpeHandle); + +/** + * @brief Gets the names of output tensors to the network + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @return List of output tensor names. + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_SNPE_GetOutputTensorNames(Snpe_SNPE_Handle_t snpeHandle); + +/** + * @brief Gets the names of output tensor from the input layer name + * + * @param[in] snpeHandle Handle to access the SNPE object + * @param[in] name Layer name + * + * @return Output tensor names. + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_SNPE_GetOutputTensorNamesByLayerName(Snpe_SNPE_Handle_t snpeHandle, const char* name); + + +/** + * @brief Processes the input data and returns the output + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] inputHandle A map of tensors that contains the input data for + * each input. The names of tensors needs to be + * matched with names retrieved through + * getInputTensorNames() + * + * @param[in,out] outputHandle An empty map of tensors that will contain the output + * data of potentially multiple layers (the key + * in the map is the layer name) upon return + * + * @note output TensorMap has to be empty. To forward propagate + * and get results in user-supplied tensors, use + * Snpe_SNPE_ExecuteUserBuffers(). + * + * @return SNPE_SUCCESS upon successful execution + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_ExecuteITensors(Snpe_SNPE_Handle_t snpeHandle, Snpe_TensorMap_Handle_t inputHandle, Snpe_TensorMap_Handle_t outputHandle); + +/** + * @brief Processes the input data and returns the output + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] inputHandle A single tensor contains the input data. + * + * @param[in,out] outputHandle An empty map of tensors that will contain the output + * data of potentially multiple layers (the key + * in the map is the layer name) upon return + * + * @note output TensorMap has to be empty. To forward propagate + * and get results in user-supplied tensors, use + * Snpe_SNPE_ExecuteUserBuffers. + * + * @return SNPE_SUCCESS upon successful execution + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_ExecuteITensor(Snpe_SNPE_Handle_t snpeHandle, Snpe_ITensor_Handle_t inputHandle, Snpe_TensorMap_Handle_t outputHandle); + +/** + * @brief Processes the input data and returns the output, using + * user-supplied buffers + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] inputHandle A map of UserBuffers that contains the input data for + * each input. The names of UserBuffers needs to be + * matched with names retrieved through + * getInputTensorNames() + * + * @param[in,out] outputHandle A map of UserBuffers that will hold the output + * data of potentially multiple layers (the key + * in the map is the UserBuffer name) + * + * @note input and output UserBuffer maps must be fully pre-populated. with + * dimensions matching what the network expects. + * For example, if there are 5 output UserBuffers they all have to be + * present in map. + * + * Caller must guarantee that for the duration of execute(), the buffer + * stored in UserBuffer would remain valid. For more detail on buffer + * ownership and lifetime requirements, please refer to zdl::DlSystem::UserBuffer + * documentation. + * + * @return SNPE_SUCCESS upon successful execution + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_ExecuteUserBuffers(Snpe_SNPE_Handle_t snpeHandle, Snpe_UserBufferMap_Handle_t inputHandle, Snpe_UserBufferMap_Handle_t outputHandle); + + +/** + * @brief Register Client ION Buffers + * + * @note To be deprecated, please use new api Snpe_SNPE_RegisterUserMemoryMappedBuffers + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] ionBufferMapHandle A UserMemoryMap of virtual addresses + * + * @return SNPE_SUCCESS upon successful ION Buffer registration + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_RegisterIonBuffers(Snpe_SNPE_Handle_t snpeHandle, Snpe_UserMemoryMap_Handle_t ionBufferMapHandle); + +/** + * @brief Deregister Client ION Buffers + * + * @note To be deprecated, please use new api Snpe_SNPE_DeregisterUserMemoryMappedBuffers + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] ionBufferNamesHandle A StringList of ION Buffer names + * + * @return SNPE_SUCCESS upon successful ION Buffer deregistration + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_DeregisterIonBuffers(Snpe_SNPE_Handle_t snpeHandle, Snpe_StringList_Handle_t ionBufferNamesHandle); + +/** + * @brief Register Client Memory-Mapped Buffers (Example ION buffers in Android) + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] bufferMapHandle A UserMemoryMap of virtual addresses + * + * @note UserBuffer type passed for registration must match the data type of the tensor in the dlc + * For regular UserBuffers SNPE performs an online data conversion (quantization or + * dequantization etc). This is not possible for memory mapped buffers hence can lead to + * issues during execution or accuracy degradation + * + * @return SNPE_SUCCESS upon successful memory mapped buffer registration + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_RegisterUserMemoryMappedBuffers(Snpe_SNPE_Handle_t snpeHandle, Snpe_UserMemoryMap_Handle_t bufferMapHandle); + +/** + * @brief Deregister Client Memory-Mapped Buffers (Example ION buffers in Android) + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] bufferNamesHandle A StringList of memory mapped buffer names + * + * @return SNPE_SUCCESS upon successful memory mapped buffer deregistration + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPE_DeregisterUserMemoryMappedBuffers(Snpe_SNPE_Handle_t snpeHandle, Snpe_StringList_Handle_t bufferNamesHandle); + +/** + * @brief Returns the version string embedded at model conversion + * time. + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @return Model version string, which is a free-form string + * supplied at the time of the conversion + * + */ +SNPE_API +const char* Snpe_SNPE_GetModelVersion(Snpe_SNPE_Handle_t snpeHandle); + +/** + * @brief Returns the dimensions of the input data to the model in the + * form of TensorShape. The dimensions in TensorShape corresponds to + * what the tensor dimensions would need to be for an input tensor to + * the model. + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] name input name. + * + * @note Note that this function only makes sense for networks + * that have a fixed input size. For networks in which the + * input size varies with each call of Execute(), this + * function should not be used. + * + * @return a TensorShape that maintains dimensions, + * matching the tensor dimensions for input to the model, + * where the last entry is the fastest varying dimension, etc. + * + * @see Snpe_ITensor_Handle_t + * @see Snpe_TensorShape_Handle_t + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_SNPE_GetInputDimensions(Snpe_SNPE_Handle_t snpeHandle, const char* name); + +/** + * @brief Returns the dimensions of the first input's data to the model in the + * form of TensorShape. The dimensions in TensorShape corresponds to + * what the tensor dimensions would need to be for an input tensor to + * the model. + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @note Note that this function only makes sense for networks + * that have a fixed input size. For networks in which the + * input size varies with each call of Execute(), this + * function should not be used. + * + * @return a TensorShape that maintains dimensions, + * matching the tensor dimensions for first input to the model, + * where the last entry is the fastest varying dimension, etc. + * + * @see Snpe_ITensor_Handle_t + * @see Snpe_TensorShape_Handle_t + */ +SNPE_API +Snpe_TensorShape_Handle_t Snpe_SNPE_GetInputDimensionsOfFirstTensor(Snpe_SNPE_Handle_t snpeHandle); + +/** + * @brief Gets the output layer(s) for the network. + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @note The output layers returned by this function may be + * different than those specified when the network was created + * via the @ref CAPI_SNPEBuilder "SNPEBuilder". For example, if the + * network was created in debug mode with no explicit output + * layers specified, this will contain all layers. + * + * + * @return A StringList of output layer names. + */ +SNPE_API +Snpe_StringList_Handle_t Snpe_SNPE_GetOutputLayerNames(Snpe_SNPE_Handle_t snpeHandle); + +/** + * @brief Returns attributes of buffers used to feed input tensors and receive result from output tensors. + * + * @param[in] snpeHandle Handle to access the SNPE object + * + * @param[in] name Tensor name. + * + * @return BufferAttributes of input/output tensor named + */ +SNPE_API +Snpe_IBufferAttributes_Handle_t Snpe_SNPE_GetInputOutputBufferAttributes(Snpe_SNPE_Handle_t snpeHandle, const char *name); + +/** + * @brief . + * + * Get the diagnostic logging interface + * + * @param[in] snpeHandle Handle to access the SNPE object + * + */ +SNPE_API +Snpe_IDiagLog_Handle_t Snpe_SNPE_GetDiagLogInterface_Ref(Snpe_SNPE_Handle_t snpeHandle); + + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.hpp new file mode 100644 index 00000000..d4ad18df --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPE.hpp @@ -0,0 +1,125 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "DlSystem/StringList.hpp" +#include "DlSystem/TensorMap.hpp" +#include "DlSystem/UserBufferMap.hpp" +#include "DlSystem/UserMemoryMap.hpp" +#include "DlSystem/IBufferAttributes.hpp" +#include "DiagLog/IDiagLog.hpp" + +#include "DlSystem/DlOptional.hpp" + + +#include "SNPE/SNPE.h" + +namespace SNPE{ + +class SNPE : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_SNPE_Delete}; + + template + static DlSystem::Optional makeOptional(H handle){ + return DlSystem::Optional(T(moveHandle(handle))); + } +public: + + DlSystem::Optional getInputTensorNames() const noexcept{ + return makeOptional(Snpe_SNPE_GetInputTensorNames(handle())); + } + + DlSystem::Optional getOutputTensorNames() const noexcept{ + return makeOptional(Snpe_SNPE_GetOutputTensorNames(handle())); + } + + DlSystem::StringList getOutputTensorNamesByLayerName(const char *name) const noexcept{ + return moveHandle(Snpe_SNPE_GetOutputTensorNamesByLayerName(handle(), name)); + } + + bool execute(const DlSystem::TensorMap& input, DlSystem::TensorMap& output) noexcept{ + return SNPE_SUCCESS == Snpe_SNPE_ExecuteITensors(handle(), getHandle(input), getHandle(output)); + } + + + bool execute(const DlSystem::ITensor* input, DlSystem::TensorMap& output) noexcept{ + if(!input) return false; + return SNPE_SUCCESS == Snpe_SNPE_ExecuteITensor(handle(), getHandle(*input), getHandle(output)); + } + + bool execute(const DlSystem::UserBufferMap& input, const DlSystem::UserBufferMap& output) noexcept{ + return SNPE_SUCCESS == Snpe_SNPE_ExecuteUserBuffers(handle(), getHandle(input), getHandle(output)); + } + + + /* To be deprecated, please use new api registerMemoryMappedBuffers */ + bool registerIonBuffers(const DlSystem::UserMemoryMap& ionBufferMap) noexcept{ + return SNPE_SUCCESS == Snpe_SNPE_RegisterUserMemoryMappedBuffers(handle(), getHandle(ionBufferMap)); + } + + /* To be deprecated, please use new api deregisterMemoryMappedBuffers */ + bool deregisterIonBuffers(const DlSystem::StringList& ionBufferNames) noexcept{ + return SNPE_SUCCESS == Snpe_SNPE_DeregisterUserMemoryMappedBuffers(handle(), getHandle(ionBufferNames)); + } + + bool registerMemoryMappedBuffers(const DlSystem::UserMemoryMap& memoryMappedBufferMap) noexcept{ + return SNPE_SUCCESS == Snpe_SNPE_RegisterUserMemoryMappedBuffers(handle(), getHandle(memoryMappedBufferMap)); + } + + bool deregisterMemoryMappedBuffers(const DlSystem::StringList& bufferNames) noexcept{ + return SNPE_SUCCESS == Snpe_SNPE_DeregisterUserMemoryMappedBuffers(handle(), getHandle(bufferNames)); + } + + std::string getModelVersion() const{ + auto str = Snpe_SNPE_GetModelVersion(handle()); + return str ? str : ""; + } + + DlSystem::Optional getInputDimensions() const noexcept{ + return makeOptional(Snpe_SNPE_GetInputDimensionsOfFirstTensor(handle())); + } + + DlSystem::Optional getInputDimensions(const char* name) const noexcept{ + return makeOptional(Snpe_SNPE_GetInputDimensions(handle(), name)); + } + + DlSystem::Optional getOutputLayerNames() const noexcept{ + return makeOptional(Snpe_SNPE_GetOutputLayerNames(handle())); + } + + + DlSystem::Optional getInputOutputBufferAttributes(const char* name) const noexcept{ + return DlSystem::Optional( + new DlSystem::IBufferAttributes(moveHandle(Snpe_SNPE_GetInputOutputBufferAttributes(handle(), name))), + DlSystem::Optional::LIFECYCLE::POINTER_OWNED + ); + } + + DlSystem::Optional getDiagLogInterface() noexcept{ + auto diagLogHandle = Snpe_SNPE_GetDiagLogInterface_Ref(handle()); + if(!diagLogHandle) return {}; + // Bind lifespan of this reference to this object + auto toret = makeReference(diagLogHandle); + return {toret, DlSystem::Optional::LIFECYCLE::POINTER_NOT_OWNED}; + } + +private: + SNPE(const SNPE&) = delete; + SNPE& operator=(const SNPE&) = delete; + +}; + +} // ns SNPE + +ALIAS_IN_ZDL_NAMESPACE(SNPE, SNPE) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.h new file mode 100644 index 00000000..6adcebad --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.h @@ -0,0 +1,334 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _SNPE_BUILDER_H_ +#define _SNPE_BUILDER_H_ + +#include "SNPE/SNPE.h" +#include "DlSystem/DlEnums.h" +#include "DlSystem/DlError.h" +#include "DlSystem/IOBufferDataTypeMap.h" +#include "DlSystem/TensorShapeMap.h" +#include "DlSystem/RuntimeList.h" +#include "DlSystem/PlatformConfig.h" +#include "DlContainer/DlContainer.h" + +#ifdef __cplusplus +extern "C" { +#endif + + + +/** + * A typedef to indicate a SNPEBuilder handle + */ +typedef void* Snpe_SNPEBuilder_Handle_t; + +/** + * The builder class for creating SNPE objects. + * Not meant to be extended. + */ + + +/** + * @brief Constructor of NeuralNetwork Builder ith a supplied model. + * + * @param[in] containerHandle A DlContainer holding the model. + * + * @return A new instance of a SNPEBuilder object + * that can be used to configure and build + * an instance of SNPE. + * + */ +SNPE_API +Snpe_SNPEBuilder_Handle_t Snpe_SNPEBuilder_Create(Snpe_DlContainer_Handle_t containerHandle); + +/** + * Destroys/frees a SNPEBuilder object + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @return SNPE_SUCCESS if Delete operation successful. + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_Delete(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle); + +/** + * @brief Requests a performance profile. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] performanceProfile The target performance profile. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetPerformanceProfile(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_PerformanceProfile_t performanceProfile); + +/** + * @brief Sets the profiling level. Default profiling level for + * SNPEBuilder is off. Off and basic only applies to DSP runtime. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] profilingLevel The target profiling level. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetProfilingLevel(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_ProfilingLevel_t profilingLevel); + +/** + * @brief Sets a preference for execution priority. + * + * This allows the caller to give coarse hint to SNPE runtime + * about the priority of the network. SNPE runtime is free to use + * this information to co-ordinate between different workloads + * that may or may not extend beyond SNPE. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] priority The target performance profile. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetExecutionPriorityHint(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_ExecutionPriorityHint_t priority); + +/** + * @brief Sets the layers that will generate output. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] outputLayerNames List of layer names to + * output. An empty list will + * result in only the final + * layer of the model being + * the output layer. The list + * will be copied. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetOutputLayers(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_StringList_Handle_t outputLayerNames); + +/** + * @brief Sets the output tensor names. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] outputTensorNames List of tensor names to + * output. An empty list will + * result in producing output for the final + * output tensor of the model. + * The list will be copied. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetOutputTensors(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_StringList_Handle_t outputTensorNames); + +/** + * @brief Sets whether this neural network will perform inference with + * input from user-supplied buffers, and write output to user-supplied + * buffers. Default behaviour is to use tensors created by + * ITensorFactory. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] bufferMode Boolean whether to use user-supplied buffer or not. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetUseUserSuppliedBuffers(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, int bufferMode); + +/** + * @brief Sets the debug mode of the runtime. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] debugMode This enables debug mode for the runtime. It + * does two things. For an empty + * outputLayerNames list, all layers will be + * output. It might also disable some internal + * runtime optimizations (e.g., some networks + * might be optimized by combining layers, + * etc.). + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetDebugMode(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, int debugMode); + + + +/** + * @brief Sets network's input dimensions to enable resizing of + * the spatial dimensions of each layer for fully convolutional networks, + * and the batch dimension for all networks. + * + * @param[in] tensorShapeMapHandle : Handle to the map of input names and their new dimensions. + * The new dimensions overwrite the input dimensions + * embedded in the model and then resize each layer + * of the model. If the model contains + * layers whose dimensions cannot be resized e.g FullyConnected, + * exception will be thrown when SNPE instance is actually built. + * In general the batch dimension is always resizable. + * After resizing of layers' dimensions in model based + * on new input dimensions, the new model is revalidated + * against all runtime constraints, whose failures may + * result in cpu fallback situation. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetInputDimensions(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_TensorShapeMap_Handle_t inputDimensionsMapHandle); + +/** + * @brief Sets the mode of init caching functionality. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] mode Boolean. This flag enables/disables the functionality of init caching. + * When init caching functionality is enabled, a set of init caches + * will be created during network building/initialization process + * and will be added to DLC container. If such DLC container is saved + * by the user, in subsequent network building/initialization processes + * these init caches will be loaded from the DLC so as to reduce initialization time. + * In disable mode, no init caches will be added to DLC container. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetInitCacheMode(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, int cacheMode); + +/** + * @brief Returns an instance of SNPE based on the current parameters. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @return A new instance of a @ref CAPI_SNPE "SNPE" object that can be used + * to execute models or null if any errors occur. + */ +SNPE_API +Snpe_SNPE_Handle_t Snpe_SNPEBuilder_Build(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle); + +/** + * @brief Sets the platform configuration. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] platformConfig The platform configuration. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetPlatformConfig(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_PlatformConfig_Handle_t platformConfigHandle); + +/** + * @brief Sets network's runtime order of precedence. Example: + * CPU_FLOAT32, GPU_FLOAT16, AIP_FIXED8_TF + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] runtimeListHandle The list of runtime in order of precedence + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetRuntimeProcessorOrder(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_RuntimeList_Handle_t runtimeListHandle); + +/** + * @brief Sets the unconsumed tensors as output + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] setOutput Boolean. This enables unconsumed tensors (i.e) + * outputs which are not inputs to any + * layer (basically dead ends) to be marked + * for output + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetUnconsumedTensorsAsOutputs(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, int setOutput); + +/** + * @brief Execution terminated when exceeding time limit. + * Only valid for dsp runtime currently. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] timeout Time limit value in microseconds + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetTimeOut(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, uint64_t timeoutMicroSec); + + +/** + * @brief Sets the datatype of the buffer. + * Only valid for dsp runtime currently. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] dataTypeMapHandle Map of the buffer names and the datatype that needs to be set. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetBufferDataType(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, Snpe_IOBufferDataTypeMap_Handle_t dataTypeMapHandle); + +/** + * @brief Sets up the entire initialization callflow to + * happen on the user's thread + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] singleThreadedInit Flag indicating user's intent to perform initialization callflow + * on caller's thread. + * When set to 1, initialization will happen on the user's thread + * When set to 0, initialization will happen on a new thread. This is the default + * behavior (analogous to not calling this API) +*/ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetSingleThreadedInit(Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, int singleThreadedInit); + +/** + * @brief Sets the fixed point execution mode for CPU runtime. + * If a floating point DLC is executed with this option set, the program will be terminated with an exception. + * If a quantized DLC is executed without this option set, the execution will be in floating point mode in CPU. + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] cpuFxpMode Boolean If set to true, enables the fixed point mode. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetCpuFixedPointMode( + Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, bool cpuFxpMode); + +/** + * @brief Sets model name for logging + * + * @param[in] snpeBuilderHandle Handle to access the SNPEBuilder object + * + * @param[in] modelName String Model name for logging. + * + */ +SNPE_API +Snpe_ErrorCode_t Snpe_SNPEBuilder_SetModelName( + Snpe_SNPEBuilder_Handle_t snpeBuilderHandle, const char *modelName); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_BUILDER_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.hpp new file mode 100644 index 00000000..37995f4e --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEBuilder.hpp @@ -0,0 +1,136 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include + + +#include "Wrapper.hpp" +#include "SNPE.hpp" +#include "DlSystem/RuntimeList.hpp" +#include "DlContainer/IDlContainer.hpp" +#include "DlSystem/PlatformConfig.hpp" +#include "DlSystem/TensorShapeMap.hpp" + +#include "DlSystem/DlEnums.hpp" + +#include "DlSystem/IOBufferDataTypeMap.hpp" + +#include "SNPE/SNPEBuilder.h" + + +namespace SNPE { + +class SNPEBuilder : public Wrapper { + friend BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_SNPEBuilder_Delete}; +public: + + explicit SNPEBuilder(DlContainer::IDlContainer *container) + : BaseType(Snpe_SNPEBuilder_Create(getHandle(container))) + { } + + + SNPEBuilder& setPerformanceProfile(DlSystem::PerformanceProfile_t performanceProfile){ + Snpe_SNPEBuilder_SetPerformanceProfile(handle(), static_cast(performanceProfile)); + return *this; + } + + SNPEBuilder& setProfilingLevel(DlSystem::ProfilingLevel_t profilingLevel){ + Snpe_SNPEBuilder_SetProfilingLevel(handle(), static_cast(profilingLevel)); + return *this; + } + + SNPEBuilder& setExecutionPriorityHint(DlSystem::ExecutionPriorityHint_t priority){ + Snpe_SNPEBuilder_SetExecutionPriorityHint(handle(), static_cast(priority)); + return *this; + } + + SNPEBuilder& setOutputLayers(const DlSystem::StringList& outputLayerNames){ + Snpe_SNPEBuilder_SetOutputLayers(handle(), getHandle(outputLayerNames)); + return *this; + } + + SNPEBuilder& setOutputTensors(const DlSystem::StringList& outputTensorNames){ + Snpe_SNPEBuilder_SetOutputTensors(handle(), getHandle(outputTensorNames)); + return *this; + } + + SNPEBuilder& setUseUserSuppliedBuffers(int bufferMode){ + Snpe_SNPEBuilder_SetUseUserSuppliedBuffers(handle(), bufferMode); + return *this; + } + + SNPEBuilder& setDebugMode(int debugMode){ + Snpe_SNPEBuilder_SetDebugMode(handle(), debugMode); + return *this; + } + + SNPEBuilder& setInputDimensions(const DlSystem::TensorShapeMap& inputDimensionsMap){ + Snpe_SNPEBuilder_SetInputDimensions(handle(), getHandle(inputDimensionsMap)); + return *this; + } + + SNPEBuilder& setInitCacheMode(int cacheMode){ + Snpe_SNPEBuilder_SetInitCacheMode(handle(), cacheMode); + return *this; + } + + SNPEBuilder& setPlatformConfig(const DlSystem::PlatformConfig& platformConfigHandle){ + Snpe_SNPEBuilder_SetPlatformConfig(handle(), getHandle(platformConfigHandle)); + return *this; + } + + SNPEBuilder& setRuntimeProcessorOrder(const DlSystem::RuntimeList& runtimeList){ + Snpe_SNPEBuilder_SetRuntimeProcessorOrder(handle(), getHandle(runtimeList)); + return *this; + } + + SNPEBuilder& setUnconsumedTensorsAsOutputs(int setOutput){ + Snpe_SNPEBuilder_SetUnconsumedTensorsAsOutputs(handle(), setOutput); + return *this; + } + + SNPEBuilder& setTimeOut(uint64_t timeoutMicroSec){ + Snpe_SNPEBuilder_SetTimeOut(handle(), timeoutMicroSec); + return *this; + } + + + SNPEBuilder& setBufferDataType(const DlSystem::IOBufferDataTypeMap& dataTypeMap){ + Snpe_SNPEBuilder_SetBufferDataType(handle(), getHandle(dataTypeMap)); + return *this; + } + + SNPEBuilder& setSingleThreadedInit(int singleThreadedInit){ + Snpe_SNPEBuilder_SetSingleThreadedInit(handle(), singleThreadedInit); + return *this; + } + + SNPEBuilder& setCpuFixedPointMode(bool cpuFxpMode){ + Snpe_SNPEBuilder_SetCpuFixedPointMode(handle(), cpuFxpMode); + return *this; + } + + SNPEBuilder& setModelName(DlSystem::String modelName){ + Snpe_SNPEBuilder_SetModelName(handle(), modelName.c_str()); + return *this; + } + + std::unique_ptr build() noexcept{ + auto h = Snpe_SNPEBuilder_Build(handle()); + return h ? makeUnique(h) : nullptr; + } + +}; + +} // ns SNPE + + +ALIAS_IN_ZDL_NAMESPACE(SNPE, SNPEBuilder) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEFactory.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEFactory.hpp new file mode 100644 index 00000000..6c2486ee --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEFactory.hpp @@ -0,0 +1,88 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" + +#include "DlSystem/DlEnums.hpp" +#include "DlSystem/DlVersion.hpp" +#include "DlSystem/ITensorFactory.hpp" +#include "DlSystem/IUserBufferFactory.hpp" + + +#include "SNPE/SNPEUtil.h" +#include "DlSystem/DlEnums.h" + +namespace SNPE { + + +class SNPEFactory { +public: + + + static bool isRuntimeAvailable(DlSystem::Runtime_t runtime){ + return Snpe_Util_IsRuntimeAvailable(static_cast(runtime)); + } + + static bool isRuntimeAvailable(DlSystem::Runtime_t runtime, DlSystem::RuntimeCheckOption_t option){ + return Snpe_Util_IsRuntimeAvailableCheckOption(static_cast(runtime), + static_cast(option)); + } + + static DlSystem::ITensorFactory& getTensorFactory(){ + static DlSystem::ITensorFactory iTensorFactory; + return iTensorFactory; + } + + static DlSystem::IUserBufferFactory& getUserBufferFactory(){ + static DlSystem::IUserBufferFactory iUserBufferFactory; + return iUserBufferFactory; + } + + static DlSystem::Version_t getLibraryVersion(){ + return WrapperDetail::moveHandle(Snpe_Util_GetLibraryVersion()); + } + + static bool setSNPEStorageLocation(const char* storagePath){ + return SNPE_SUCCESS == Snpe_Util_SetSNPEStorageLocation(storagePath); + } + + static bool addOpPackage(const std::string& regLibraryPath){ + return SNPE_SUCCESS == Snpe_Util_AddOpPackage(regLibraryPath.c_str()); + } + + static bool isGLCLInteropSupported(){ + return Snpe_Util_IsGLCLInteropSupported(); + } + + static const char* getLastError(){ + return Snpe_Util_GetLastError(); + } + + static bool initializeLogging(const DlSystem::LogLevel_t& level){ + return Snpe_Util_InitializeLogging(static_cast(level)); + } + + static bool initializeLogging(const DlSystem::LogLevel_t& level, const std::string& logPath){ + return Snpe_Util_InitializeLoggingPath(static_cast(level), logPath.c_str()); + } + + static bool setLogLevel(const DlSystem::LogLevel_t& level){ + return Snpe_Util_SetLogLevel(static_cast(level)); + } + + static bool terminateLogging(){ + return Snpe_Util_TerminateLogging(); + } +}; + + +} // ns SNPE + + +ALIAS_IN_ZDL_NAMESPACE(SNPE, SNPEFactory) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEUtil.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEUtil.h new file mode 100644 index 00000000..a3e1d1e1 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/SNPEUtil.h @@ -0,0 +1,354 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022-2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +/** + * @file + */ + +#ifndef _SNPE_UTIL_H_ +#define _SNPE_UTIL_H_ + +#include "SNPE/SNPE.h" +#include "DlSystem/DlEnums.h" +#include "DlSystem/DlError.h" +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/IUserBuffer.h" +#include "DlSystem/ITensor.h" +#include "DlSystem/TensorShape.h" +#include "DlSystem/DlVersion.h" + +#ifdef __cplusplus +extern "C" { +#endif + + +/** + * @brief Creates a UserBuffer + * + * @param[in] buffer Pointer to the buffer that the caller supplies + * + * @param[in] bufSize Buffer size, in bytes + * + * @param[in] stridesHandle Total number of bytes between elements in each dimension. + * E.g. A tightly packed tensor of floats with dimensions [4, 3, 2] would have strides of [24, 8, 4]. + * + * @param[in] userBufferEncodingHandle Handle to a UserBufferEncoding object + * + * @note Caller has to ensure that memory pointed to by buffer stays accessible + * for the lifetime of the object created + * + * @return Handle to the created UserBuffer + */ +SNPE_API +Snpe_IUserBuffer_Handle_t Snpe_Util_CreateUserBuffer(void *buffer, + size_t bufSize, + Snpe_TensorShape_Handle_t stridesHandle, + Snpe_IUserBuffer_Handle_t userBufferEncodingHandle); + +/** + * @brief Creates a UserBuffer with a provided UserBufferSource + * + * @param[in] buffer Pointer to the buffer that the caller supplies + * + * @param[in] bufSize Buffer size, in bytes + * + * @param[in] stridesHandle Total number of bytes between elements in each dimension. + * E.g. A tightly packed tensor of floats with dimensions [4, 3, 2] would have strides of [24, 8, 4]. + * + * @param[in] userBufferEncodingHandle Handle to a UserBufferEncoding object + * + * @param[in] userBufferSourceHandle Handle to a UserBufferSource object + * + * @return Handle to the created UserBuffer + */ +SNPE_API +Snpe_IUserBuffer_Handle_t Snpe_Util_CreateUserBufferFromSource(void *buffer, + size_t bufSize, + Snpe_TensorShape_Handle_t stridesHandle, + Snpe_IUserBuffer_Handle_t userBufferEncodingHandle, + Snpe_UserBufferSource_Handle_t userBufferSourceHandle); + +/** + * @brief Creates a UserBuffer + * + * @param[in] buffer Pointer to the buffer that the caller supplies + * + * @param[in] bufSize Buffer size, in bytes + * + * @param[in] stridesHandle Total number of bytes between elements in each dimension. + * E.g. A tightly packed tensor of floats with dimensions [4, 3, 2] would have strides of [24, 8, 4]. + * + * @param[in] userBufferEncodingHandle Reference to an UserBufferEncoding object + * + * @param[in] userBufferSourceHandle Reference to an UserBufferSource object + * + * @note Caller has to ensure that memory pointed to by buffer stays accessible + * for the lifetime of the object created + * + * @return the created UserBuffer + * + */ +SNPE_API +Snpe_IUserBuffer_Handle_t Snpe_Util_CreateUserGlBuffer(void *buffer, + size_t bufSize, + Snpe_TensorShape_Handle_t stridesHandle, + Snpe_IUserBuffer_Handle_t userBufferEncodingHandle, + Snpe_IUserBuffer_Handle_t userBufferSourceHandle); + +/** + * Creates a new ITensor with uninitialized data. + * + * ITensor buffer size assumes float32 encoding for each element. + * (i.e., a tensor with dimensions (2,3) will be represented by (2 * 3) * 4 = 24 bytes in memory) + * + * The strides for the tensor will match the tensor dimensions + * (i.e., the tensor data is contiguous in memory). + * + * @param[in] shapeHandle The dimensions for the tensor in which the last + * element of the vector represents the fastest varying + * dimension and the zeroth element represents the slowest + * varying, etc. + * + * @return The created tensor + */ +SNPE_API +Snpe_ITensor_Handle_t Snpe_Util_CreateITensor(Snpe_TensorShape_Handle_t shapeHandle); + +/** + * Create a new ITensor with specific data. + * (i.e. the tensor data is contiguous in memory). This tensor is + * primarily used to create a tensor where tensor size can't be + * computed directly from dimension. One such example is + * NV21-formatted image, or any YUV formatted image + * + * @param[in] shapeHandle The dimensions for the tensor in which the last + * element of the vector represents the fastest varying + * dimension and the zeroth element represents the slowest + * varying, etc. + * + * @param[in] data The actual data with which the Tensor object is filled. + * + * @param[in] dataSize The size of data + * + * @return A handle to the created tensor + */ +SNPE_API +Snpe_ITensor_Handle_t Snpe_Util_CreateITensorDataSize(Snpe_TensorShape_Handle_t shapeHandle, const uint8_t* data, size_t dataSize); + +/** + * Create a new ITensor with specific data. + * (i.e. the tensor data is contiguous in memory). This tensor is + * primarily used to create a tensor where tensor size can't be + * computed directly from dimension. One such example is + * NV21-formatted image, or any YUV formatted image + * + * @param[in] shapeHandle The dimensions for the tensor in which the last + * element of the vector represents the fastest varying + * dimension and the zeroth element represents the slowest + * varying, etc. + * + * @param[in] data The actual data with which the Tensor object is filled. + * + * @param[in] dataSize The size of data + * + * @return the created tensor + */ +SNPE_API +Snpe_ITensor_Handle_t Snpe_Util_CreateITensor_NV21(Snpe_TensorShape_Handle_t shapeHandle, unsigned char *data, size_t dataSize); + +/** + * Indicates whether the supplied runtime is available on the + * current platform. + * + * @param[in] runtime The target runtime to check. + * + * @return Boolean: Non-zero if the supplied runtime is available; 0 otherwise + * + */ +SNPE_API +int Snpe_Util_IsRuntimeAvailable(Snpe_Runtime_t runtime); + +/** + * Indicates whether the supplied runtime is available on the + * current platform. + * + * @param[in] runtime The target runtime to check. + * + * @param[in] runtimeCheckOption Extent to perform runtime available check. + * + * @return Boolean: Non-zero if the supplied runtime is available; 0 otherwise + * + */ +SNPE_API +int Snpe_Util_IsRuntimeAvailableCheckOption(Snpe_Runtime_t runtime, Snpe_RuntimeCheckOption_t runtimeCheckOption); + + +/** + * Gets the version of the SNPE library. + * + * @return Version of the SNPE library. + * + */ +SNPE_API +Snpe_DlVersion_Handle_t Snpe_Util_GetLibraryVersion(); + +/** + * Set the SNPE storage location for all SNPE instances in this + * process. Note that this may only be called once, and if so + * must be called before creating any SNPE instances. + * + * @param[in] storagePath Absolute path to a directory which SNPE may + * use for caching and other storage purposes. + * + * @return Boolean: Non-zero if the supplied path was succesfully set as + * the SNPE storage location, 0 otherwise. + * + */ +SNPE_API +int Snpe_Util_SetSNPEStorageLocation(const char* storagePath); + +/** + * @brief Register a user-defined op package with SNPE. + * + * @param[in] regLibraryPath Path to the registration library + * that allows clients to register a set of operations that are + * part of the package, and share op info with SNPE + * + * @return Boolean: Non-zero if successful, 0 otherwise. + */ +SNPE_API +int Snpe_Util_AddOpPackage(const char* regLibraryPath ); + +/** + * Indicates whether the OpenGL and OpenCL interoperability is supported + * on GPU platform. + * + * @return Boolean: Non-zero if the OpenGL and OpenCl interop is supported; 0 otherwise + * + */ +SNPE_API +int Snpe_Util_IsGLCLInteropSupported(); + +/** + * @return A string description of the last error + */ +SNPE_API +const char* Snpe_Util_GetLastError(); + +/** + * Initializes logging with the specified log level. + * initializeLogging with level, is used on Android platforms + * and after successful initialization, SNPE + * logs are printed in android logcat logs. + * + * It is recommended to initializeLogging before creating any + * SNPE instances, in order to capture information related to + * core initialization. If this is called again after first + * time initialization, subsequent calls are ignored. + * Also, Logging can be re-initialized after a call to + * terminateLogging API by calling initializeLogging again. + * + * A typical usage of Logging life cycle can be + * initializeLogging() + * any other SNPE API like isRuntimeAvailable() + * * setLogLevel() - optional - can be called anytime + * between initializeLogging & terminateLogging + * SNPE instance creation, inference, destroy + * terminateLogging(). + * + * Please note, enabling logging can have performance impact. + * + * @param[in] level Log level (LOG_INFO, LOG_WARN, etc.). + * + * @return Boolean: non-zero if successful, 0 otherwise. + */ +SNPE_API +int Snpe_Util_InitializeLogging(Snpe_LogLevel_t level); + +/** + * Initializes logging with the specified log level and log path. + * initializeLogging with level & log path, is used on non Android + * platforms and after successful initialization, SNPE + * logs are printed in std output & into log files created in the + * log path. + * + * It is recommended to initializeLogging before creating any + * SNPE instances, in order to capture information related to + * core initialization. If this is called again after first + * time initialization, subsequent calls are ignored. + * Also, Logging can be re-initialized after a call to + * terminateLogging API by calling initializeLogging again. + * + * A typical usage of Logging life cycle can be + * initializeLogging() + * any other SNPE API like isRuntimeAvailable() + * * setLogLevel() - optional - can be called anytime + * between initializeLogging & terminateLogging + * SNPE instance creation, inference, destroy + * terminateLogging() + * + * Please note, enabling logging can have performance impact + * + * @param[in] level Log level (LOG_INFO, LOG_WARN, etc.). + * + * @param[in] logPath of directory to store logs. + * If path is empty, the default path is "./Log". + * For android, the log path is ignored. + * + * @return Boolean: non-zero if successful, 0 otherwise. + */ +SNPE_API +int Snpe_Util_InitializeLoggingPath(Snpe_LogLevel_t level, const char* logPath); + +/** + * Updates the current logging level with the specified level. + * setLogLevel is optional, called anytime after initializeLogging + * and before terminateLogging, to update the log level set. + * Log levels can be updated multiple times by calling setLogLevel + * A call to setLogLevel() is ignored if it is made before + * initializeLogging() or after terminateLogging() + * + * @param[in] level Log level (LOG_INFO, LOG_WARN, etc.). + * + * @return Boolean: non-zero if successful, 0 otherwise. + */ +SNPE_API +int Snpe_Util_SetLogLevel(Snpe_LogLevel_t level); + +/** + * Terminates logging. + * + * It is recommended to terminateLogging after initializeLogging + * in order to disable logging information. + * If this is called before initialization or after first time termination, + * calls are ignored. + * + * @warning Snpe_Util_TerminateLogging() must not be called while another thread is executing. + * In a multi-threaded use case, the individual threads must have a cooperative life cycle + * management strategy for the logger. + * + * @return Boolean: non-zero if successful, 0 otherwise. + */ +SNPE_API +int Snpe_Util_TerminateLogging(); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_UTIL_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.h new file mode 100644 index 00000000..e6a42ddb --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.h @@ -0,0 +1,77 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2022,2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef _SNPE_USER_BUFFER_LIST_H_ +#define _SNPE_USER_BUFFER_LIST_H_ + + +#ifdef __cplusplus +#include +#else +#include +#endif + +#include "DlSystem/SnpeApiExportDefine.h" +#include "DlSystem/DlError.h" + +#include "DlSystem/UserBufferMap.h" + +#ifdef __cplusplus +extern "C" { +#endif + +typedef void* Snpe_UserBufferList_Handle_t; + +SNPE_API +Snpe_UserBufferList_Handle_t Snpe_UserBufferList_Create(); + +SNPE_API +Snpe_UserBufferList_Handle_t Snpe_UserBufferList_CreateCopy(Snpe_UserBufferList_Handle_t userBufferListHandle); + +SNPE_API +Snpe_UserBufferList_Handle_t Snpe_UserBufferList_CreateSize(size_t size); + +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferList_Delete(Snpe_UserBufferList_Handle_t userBufferListHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferList_PushBack(Snpe_UserBufferList_Handle_t userBufferListHandle, + Snpe_UserBufferMap_Handle_t userBufferMapHandle); + +SNPE_API +Snpe_UserBufferMap_Handle_t Snpe_UserBufferList_At_Ref(Snpe_UserBufferList_Handle_t userBufferListHandle, + size_t idx); + +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferList_Assign(Snpe_UserBufferList_Handle_t srcUserBufferListHandle, + Snpe_UserBufferList_Handle_t dstUserBufferListHandle); + +SNPE_API +size_t Snpe_UserBufferList_Size(Snpe_UserBufferList_Handle_t userBufferListHandle); + +SNPE_API +size_t Snpe_UserBufferList_Capacity(Snpe_UserBufferList_Handle_t userBufferListHandle); + +SNPE_API +Snpe_ErrorCode_t Snpe_UserBufferList_Clear(Snpe_UserBufferList_Handle_t userBufferListHandle); + + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // _SNPE_USER_BUFFER_LIST_H_ diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.hpp new file mode 100644 index 00000000..fec82dbc --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SNPE/UserBufferList.hpp @@ -0,0 +1,76 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#include "Wrapper.hpp" +#include "DlSystem/UserBufferMap.hpp" + +#include "SNPE/UserBufferList.h" + + +namespace PSNPE { + +class UserBufferList : public Wrapper { + friend BaseType; + // Use this to get free move Ctor and move assignment operator, provided this class does not specify + // as copy assignment operator or copy Ctor + using BaseType::BaseType; + + static constexpr DeleteFunctionType DeleteFunction{Snpe_UserBufferList_Delete}; + +public: + UserBufferList() + : BaseType(Snpe_UserBufferList_Create()) + { } + explicit UserBufferList(size_t size) + : BaseType(Snpe_UserBufferList_CreateSize(size)) + { } + + UserBufferList(const UserBufferList& other) + : BaseType(Snpe_UserBufferList_CreateCopy(other.handle())) + { } + UserBufferList(UserBufferList&& other) noexcept + : BaseType(std::move(other)) + { } + + UserBufferList& operator=(const UserBufferList& other){ + if(this != &other){ + Snpe_UserBufferList_Assign(other.handle(), handle()); + } + return *this; + } + UserBufferList& operator=(UserBufferList&& other){ + return moveAssign(std::move(other)); + } + + + void push_back(const DlSystem::UserBufferMap& userBufferMap){ + Snpe_UserBufferList_PushBack(handle(), getHandle(userBufferMap)); + } + + DlSystem::UserBufferMap& operator[](size_t idx){ + return *makeReference(Snpe_UserBufferList_At_Ref(handle(), idx)); + } + + size_t size() const noexcept{ + return Snpe_UserBufferList_Size(handle()); + } + + size_t capacity() const noexcept{ + return Snpe_UserBufferList_Capacity(handle()); + } + + void clear() noexcept{ + Snpe_UserBufferList_Clear(handle()); + } +}; + + +} // ns PSNPE + +ALIAS_IN_ZDL_NAMESPACE(PSNPE, UserBufferList) diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoBase.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoBase.h new file mode 100644 index 00000000..f7af604a --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoBase.h @@ -0,0 +1,546 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2019-2022 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef SNPE_UDO_BASE_H +#define SNPE_UDO_BASE_H + +#include + +// Provide values to use for API version. +#define API_VERSION_MAJOR 1 +#define API_VERSION_MINOR 6 +#define API_VERSION_TEENY 0 + +/** @addtogroup c_plus_plus_apis C++ +@{ */ + +// Defines a bitmask of enum values. +typedef uint32_t SnpeUdo_Bitmask_t; +typedef SnpeUdo_Bitmask_t Udo_Bitmask_t; + +// A string of characters, rather than an array of bytes. +// Assumed to be UTF-8. +typedef char* SnpeUdo_String_t; +typedef SnpeUdo_String_t Udo_String_t; + +// The maximum allowable length of a SnpeUdo_String_t in bytes, +// including null terminator. SNPE will truncate strings longer +// than this. +#define SNPE_UDO_MAX_STRING_SIZE 1024 + +/** + * An enum which holds the various error types. + * The error types are divided to classes : + * 0 - 99 : generic errors + * 100 - 200 : errors related to configuration + * + */ +typedef enum +{ + /// No Error + SNPE_UDO_NO_ERROR = 0, UDO_NO_ERROR = 0, + /// Unsupported value for core type + SNPE_UDO_WRONG_CORE = 1, UDO_WRONG_CORE = 1, + /// Invalid attribute/argument passed into UDO API + SNPE_UDO_INVALID_ARGUMENT = 2, UDO_INVALID_ARGUMENT = 2, + /// Unsupported feature error + SNPE_UDO_UNSUPPORTED_FEATURE = 3, UDO_UNSUPPORTED_FEATURE = 3, + /// Error relating to memory allocation + SNPE_UDO_MEM_ALLOC_ERROR = 4, UDO_MEM_ALLOC_ERROR = 4, + /* Configuration Specific errors */ + /// No op with given attributes available in library + SNPE_UDO_WRONG_OPERATION = 100, UDO_WRONG_OPERATION = 100, + /// Unsupported value for core type in UDO configuration + SNPE_UDO_WRONG_CORE_TYPE = 101, UDO_WRONG_CORE_TYPE = 101, + /// Wrong number of params in UDO definition + SNPE_UDO_WRONG_NUM_OF_PARAMS = 102, UDO_WRONG_NUM_OF_PARAMS = 102, + /// Wrong number of dimensions for tensor(s) in UDO definition + SNPE_UDO_WRONG_NUM_OF_DIMENSIONS = 103, UDO_WRONG_NUM_OF_DIMENSIONS = 103, + /// Wrong number of input tensors in UDO definition + SNPE_UDO_WRONG_NUM_OF_INPUTS = 104, UDO_WRONG_NUM_OF_INPUTS = 104, + /// Wrong number of output tensors in UDO definition + SNPE_UDO_WRONG_NUM_OF_OUTPUTS = 105, UDO_WRONG_NUM_OF_OUTPUTS = 105, + SNPE_UDO_PROGRAM_CACHE_NOT_FOUND = 106, UDO_PROGRAM_CACHE_NOT_FOUND = 106, + SNPE_UDO_UNKNOWN_ERROR = 0xFFFFFFFF, UDO_UNKNOWN_ERROR = 0xFFFFFFFF +} SnpeUdo_ErrorType_t; + +typedef SnpeUdo_ErrorType_t Udo_ErrorType_t; + +/** + * An enum which holds the various data types. + * Designed to be used as single values or combined into a bitfield parameter + * (0x1, 0x2, 0x4, etc) + * \n FIXED_XX types are targeted for data in tensors. + * \n UINT / INT types are targeted for scalar params + */ +typedef enum +{ + /// data type: 16-bit floating point + SNPE_UDO_DATATYPE_FLOAT_16 = 0x01, UDO_DATATYPE_FLOAT_16 = 0x01, + /// data type: 32-bit floating point + SNPE_UDO_DATATYPE_FLOAT_32 = 0x02, UDO_DATATYPE_FLOAT_32 = 0x02, + /// data type: 4-bit fixed point + SNPE_UDO_DATATYPE_FIXED_4 = 0x04, UDO_DATATYPE_FIXED_4 = 0x04, + /// data type: 8-bit fixed point + SNPE_UDO_DATATYPE_FIXED_8 = 0x08, UDO_DATATYPE_FIXED_8 = 0x08, + /// data type: 16-bit fixed point + SNPE_UDO_DATATYPE_FIXED_16 = 0x10, UDO_DATATYPE_FIXED_16 = 0x10, + /// data type: 32-bit fixed point + SNPE_UDO_DATATYPE_FIXED_32 = 0x20, UDO_DATATYPE_FIXED_32 = 0x20, + /// data type: 8-bit unsigned integer + SNPE_UDO_DATATYPE_UINT_8 = 0x100, UDO_DATATYPE_UINT_8 = 0x100, + /// data type: 16-bit unsigned integer + SNPE_UDO_DATATYPE_UINT_16 = 0x200, UDO_DATATYPE_UINT_16 = 0x200, + /// data type: 32-bit unsigned integer + SNPE_UDO_DATATYPE_UINT_32 = 0x400, UDO_DATATYPE_UINT_32 = 0x400, + /// data type: 8-bit signed integer + SNPE_UDO_DATATYPE_INT_8 = 0x1000, UDO_DATATYPE_INT_8 = 0x1000, + /// data type: 16-bit signed integer + SNPE_UDO_DATATYPE_INT_16 = 0x2000, UDO_DATATYPE_INT_16 = 0x2000, + /// data type: 32-bit signed integer + SNPE_UDO_DATATYPE_INT_32 = 0x4000, UDO_DATATYPE_INT_32 = 0x4000, + SNPE_UDO_DATATYPE_LAST = 0xFFFFFFFF, UDO_DATATYPE_LAST = 0xFFFFFFFF +} SnpeUdo_DataType_t; + +typedef SnpeUdo_DataType_t Udo_DataType_t; + +/** + * An enum which holds the various layouts. + * Designed to be used as single values or combined into a bitfield parameter + * (0x1, 0x2, 0x4, etc) + */ +typedef enum +{ + /// data layout (4D): NHWC (batch-height-width-channel) + SNPE_UDO_LAYOUT_NHWC = 0x01, UDO_LAYOUT_NHWC = 0x01, + /// data layout (4D): NCHW (batch-channel-height-width) + SNPE_UDO_LAYOUT_NCHW = 0x02, UDO_LAYOUT_NCHW = 0x02, + /// data layout (5D): NDHWC (batch-depth-height-width-channel) + SNPE_UDO_LAYOUT_NDHWC = 0x04, UDO_LAYOUT_NDHWC = 0x04, + SNPE_UDO_LAYOUT_GPU_OPTIMAL1 = 0x08, UDO_LAYOUT_GPU_OPTIMAL1 = 0x08, + SNPE_UDO_LAYOUT_GPU_OPTIMAL2 = 0x10, UDO_LAYOUT_GPU_OPTIMAL2 = 0x10, + SNPE_UDO_LAYOUT_DSP_OPTIMAL1 = 0x11, UDO_LAYOUT_DSP_OPTIMAL1 = 0x11, + SNPE_UDO_LAYOUT_DSP_OPTIMAL2 = 0x12, UDO_LAYOUT_DSP_OPTIMAL2 = 0x12, + // Indicates no data will be allocated for this tensor. + // Used to specify optional inputs/outputs positionally. + SNPE_UDO_LAYOUT_NULL = 0x13, UDO_LAYOUT_NULL = 0x13, + SNPE_UDO_LAYOUT_LAST = 0xFFFFFFFF, UDO_LAYOUT_LAST = 0xFFFFFFFF +} SnpeUdo_TensorLayout_t; + +typedef SnpeUdo_TensorLayout_t Udo_TensorLayout_t; + +/** + * An enum which holds the UDO library Core type . + * Designed to be used as single values or combined into a bitfield parameter + * (0x1, 0x2, 0x4, etc) + */ +typedef enum +{ + /// Library target IP Core is undefined + SNPE_UDO_CORETYPE_UNDEFINED = 0x00, UDO_CORETYPE_UNDEFINED = 0x00, + /// Library target IP Core is CPU + SNPE_UDO_CORETYPE_CPU = 0x01, UDO_CORETYPE_CPU = 0x01, + /// Library target IP Core is GPU + SNPE_UDO_CORETYPE_GPU = 0x02, UDO_CORETYPE_GPU = 0x02, + /// Library target IP Core is DSP + SNPE_UDO_CORETYPE_DSP = 0x04, UDO_CORETYPE_DSP = 0x04, + SNPE_UDO_CORETYPE_LAST = 0xFFFFFFFF, UDO_CORETYPE_LAST = 0xFFFFFFFF +} SnpeUdo_CoreType_t; + +typedef SnpeUdo_CoreType_t Udo_CoreType_t; + +/** + * An enum to specify the parameter type : Scalar or Tensor + */ +typedef enum +{ + /// UDO static param type: scalar + SNPE_UDO_PARAMTYPE_SCALAR = 0x00, UDO_PARAMTYPE_SCALAR = 0x00, + /// UDO static param type: string + SNPE_UDO_PARAMTYPE_STRING = 0x01, UDO_PARAMTYPE_STRING = 0x01, + /// UDO static param type: tensor + SNPE_UDO_PARAMTYPE_TENSOR = 0x02, UDO_PARAMTYPE_TENSOR = 0x02, + SNPE_UDO_PARAMTYPE_LAST = 0xFFFFFFFF, UDO_PARAMTYPE_LAST = 0xFFFFFFFF +} SnpeUdo_ParamType_t; + +typedef SnpeUdo_ParamType_t Udo_ParamType_t; + +/** + * An enum to specify quantization type + */ +typedef enum +{ + /// Tensor Quantization type: NONE. Signifies unquantized tensor data + SNPE_UDO_QUANTIZATION_NONE = 0x00, UDO_QUANTIZATION_NONE = 0x00, + /// Tensor Quantization type: Tensorflow-style + SNPE_UDO_QUANTIZATION_TF = 0x01, UDO_QUANTIZATION_TF = 0x01, + SNPE_UDO_QUANTIZATION_QMN = 0x02, UDO_QUANTIZATION_QMN = 0x02, + SNPE_UDO_QUANTIZATION_LAST = 0xFFFFFFFF, UDO_QUANTIZATION_LAST = 0xFFFFFFFF +} SnpeUdo_QuantizationType_t; + +typedef SnpeUdo_QuantizationType_t Udo_QuantizationType_t; + +/** + * @brief A struct which is used to provide a version number using 3 values : major, minor, teeny + * + */ +typedef struct +{ + /// version field: major - for backward-incompatible changes + uint32_t major; + /// version field: minor - for backward-compatible feature updates + uint32_t minor; + /// version field: teeny - for minor bug-fixes and clean-up + uint32_t teeny; +} SnpeUdo_Version_t; + +typedef SnpeUdo_Version_t Udo_Version_t; + +/** + * @brief A struct returned from version query, contains the Library version and API version + * + */ +typedef struct +{ + /// Version of UDO library. Controlled by users + SnpeUdo_Version_t libVersion; + /// Version of SNPE UDO API used in compiling library. Determined by SNPE + SnpeUdo_Version_t apiVersion; +} SnpeUdo_LibVersion_t; + +/** + * @brief A struct returned from version query, contains the package version + * + */ +typedef struct +{ + /// Version of UDO API used in package. + Udo_Version_t apiVersion; +} Udo_PkgVersion_t; + +/** + * @brief A union to hold the value of a generic type. Allows defining a parameter struct + * in a generic way, with a "value" location that holds the data regardless of the type. + * + */ +typedef union +{ + /// value type: float + float floatValue; + /// value type: unsigned 32-bit integer + uint32_t uint32Value; + /// value type: signed 32-bit integer + int32_t int32Value; + /// value type: unsigned 16-bit integer + uint16_t uint16Value; + /// value type: signed 16-bit integer + int16_t int16Value; + /// value type: unsigned 8-bit integer + uint8_t uint8Value; + /// value type: signed 8-bit integer + int8_t int8Value; +} SnpeUdo_Value_t; + +typedef SnpeUdo_Value_t Udo_Value_t; + +/** + * @brief A struct which defines a scalar parameter : name, data type, and union of values + * + */ +typedef struct +{ + /// The parameter data type : float, int, etc. + SnpeUdo_DataType_t dataType; + /// a union of specified type which holds the data + SnpeUdo_Value_t dataValue; +} SnpeUdo_ScalarParam_t; + +typedef SnpeUdo_ScalarParam_t Udo_ScalarParam_t; + +/** + * @brief A struct which defines the quantization parameters in case of Tensorflow style quantization + * + */ +typedef struct +{ + /// minimum value of the quantization range of data + float minValue; + /// maximum value of the quantization range of data + float maxValue; +} SnpeUdo_TFQuantize_t; + +typedef SnpeUdo_TFQuantize_t Udo_TFQuantize_t; + +/** + * @brief A struct which defines the quantization type, and union of supported quantization structs + * + */ +typedef struct +{ + /// quantization type (only TF-style currently supported) + SnpeUdo_QuantizationType_t quantizeType; + union + { + /// TF-style min-max quantization ranges + SnpeUdo_TFQuantize_t TFParams; + }; +} SnpeUdo_QuantizeParams_t; + +typedef SnpeUdo_QuantizeParams_t Udo_QuantizeParams_t; + +/** + * @brief A struct which defines the datatype associated with a specified core-type + * This should be used to denote the datatypes for a single tensor info, depending + * on the intended execution core. + * + */ +typedef struct +{ + /// The IP Core + SnpeUdo_CoreType_t coreType; + /// The associated datatype for this coreType + SnpeUdo_DataType_t dataType; +} SnpeUdo_PerCoreDatatype_t; + +typedef SnpeUdo_PerCoreDatatype_t Udo_PerCoreDatatype_t; + +/** + * @brief A struct which defines a tensor parameter : name, data type, layout, quantization, more. + * Also holds a pointer to the tensor data. + * + */ +typedef struct +{ + /// The maximum allowable dimensions of the tensor. The memory held in + /// _tensorData_ is guaranteed to be large enough for this. + uint32_t* maxDimensions; + /// The current dimensions of the tensor. An operation may modify the current + /// dimensions of its output, to indicate cases where the output has been + /// "resized". + /// Note that for static parameters, the current and max dimensions must + /// match. + uint32_t* currDimensions; + /// Quantization params applicable to the tensor. Currently only supports + /// Tensorflow quantization style. + SnpeUdo_QuantizeParams_t quantizeParams; + /// Number of dimensions to the tensor: 3D, 4D, etc. + uint32_t tensorRank; + /// The parameter data type: float, int, etc. + SnpeUdo_DataType_t dataType; + /// The tensor layout type: NCHW, NHWC, etc. + SnpeUdo_TensorLayout_t layout; + /// Opaque pointer to tensor data. User may be required to re-interpret the pointer + /// based on core-specific definitions. + void* tensorData; +} SnpeUdo_TensorParam_t; + +typedef SnpeUdo_TensorParam_t Udo_TensorParam_t; + +/** + * @brief A struct which defines tensor information for activation tensors only + * + * It describes an activation tensor object using its name, the intended layout and the datatype + * it will take depending on the intended runtime core. The repeated field indicates that + * that the tensor info describes several input/output activation tensors, which all share the + * aforementioned properties. + */ +typedef struct +{ + /// The tensor name + SnpeUdo_String_t tensorName; + /// The tensor layout type: NCHW, NHWC, etc. + SnpeUdo_TensorLayout_t layout; + /// The per core datatype: {SNPE_UDO_DATATYPE, SNPE_UDO_CORE_TYPE} + SnpeUdo_PerCoreDatatype_t* perCoreDatatype; + /// A boolean field indicating that this tensorinfo will be repeated e.x for ops such as Concat or Split + bool repeated; + /// A boolean field indicating whether input is static or not. + bool isStatic; +} SnpeUdo_TensorInfo_t; + +typedef SnpeUdo_TensorInfo_t Udo_TensorInfo_t; + +/** + * @brief struct which defines a UDO parameter - a union of scalar, tensor and string parameters + * + */ +typedef struct +{ + /// Type is scalar or tensor + SnpeUdo_ParamType_t paramType; + /// The param name, for example : "offset", "activation_type" + SnpeUdo_String_t paramName; + union + { + /// scalar param value + SnpeUdo_ScalarParam_t scalarParam; + /// tensor param value + SnpeUdo_TensorParam_t tensorParam; + /// string param value + SnpeUdo_String_t stringParam; + }; +} SnpeUdo_Param_t; + +typedef SnpeUdo_Param_t Udo_Param_t; + +/** + * @brief A struct which defines Operation information which is specific for IP core (CPU, GPU, DSP ...) + * + */ +typedef struct +{ + /// The IP Core + SnpeUdo_CoreType_t udoCoreType; + /// Bitmask, defines supported internal calculation types (like FLOAT_32, etc) + /// Based on SnpeUdo_DataType + SnpeUdo_Bitmask_t operationCalculationTypes; +} SnpeUdo_OpCoreInfo_t; + +typedef SnpeUdo_OpCoreInfo_t Udo_OpCoreInfo_t; + +/** + * @brief A struct which defines the common and core-specific Operation information + * + */ +typedef struct +{ + /// Operation type + SnpeUdo_String_t operationType; + /// A bitmask describing which IP Cores (CPU, GPU, DSP ...) support this operation + /// Translated based on SnpeUdo_CoreType + SnpeUdo_Bitmask_t supportedByCores; + /// Number of static parameters defined by the op + uint32_t numOfStaticParams; + /// Array of static parameters. Can be scalar or tensor params + SnpeUdo_Param_t* staticParams; + /// Number of input tensors this op receives + uint32_t numOfInputs; + /// Array of input tensor names to this operation + SnpeUdo_String_t* inputNames; + /// Number of output tensors this op receives + uint32_t numOfOutputs; + /// Array of output tensor names to this operation + SnpeUdo_String_t* outputNames; + /// Number of cores that the op can execute on + uint32_t numOfCoreInfo; + /// Array of per-core information entries + SnpeUdo_OpCoreInfo_t* opPerCoreInfo; + /// Array of input tensor infos for this operation + SnpeUdo_TensorInfo_t* inputInfos; + /// Array of output tensor infos for this operation + SnpeUdo_TensorInfo_t* outputInfos; +} SnpeUdo_OperationInfo_t; + +typedef SnpeUdo_OperationInfo_t Udo_OperationInfo_t; + +/** + * @brief A struct which provides the implementation library info : type, name + * + */ +typedef struct +{ + /// Defines the IP Core that this implementation library is targeting + SnpeUdo_CoreType_t udoCoreType; + /// library name. will be looked at in the standard library path + SnpeUdo_String_t libraryName; +} SnpeUdo_LibraryInfo_t; + +typedef SnpeUdo_LibraryInfo_t Udo_LibraryInfo_t; + +/** + * @brief A struct returned by the registration library and contains information on the UDO package : + * name, operations, libraries, etc. + * + */ +typedef struct +{ + /// A string containing the package name + SnpeUdo_String_t packageName; + /// A bitmask describing supported IP cores (CPU, GPU, DSP ...) + /// Translated based on SnpeUdo_CoreType + SnpeUdo_Bitmask_t supportedCoreTypes; + /// The number of implementation libraries in the package + uint32_t numOfImplementationLib; + /// Array of implementation libraries names/types + SnpeUdo_LibraryInfo_t* implementationLib; + /// A string containing all operation types separated by space + SnpeUdo_String_t operationsString; + /// Number of supported operations + uint32_t numOfOperations; + /// Array of Operation info structs. Each entry describes one + /// Operation (name, params, inputs, outputs) + SnpeUdo_OperationInfo_t* operationsInfo; +} SnpeUdo_RegInfo_t; + +typedef SnpeUdo_RegInfo_t Udo_RegInfo_t; + +/** +* @brief A struct returned by the implementation library and contains information on the +* specific library: name, IP Core, operations, etc. +* +*/ +typedef struct +{ + /// Defines the IP Core that this implementation library is targeting + SnpeUdo_CoreType_t udoCoreType; + /// A string containing the package name + SnpeUdo_String_t packageName; + /// A string containing all operation types separated by space + SnpeUdo_String_t operationsString; + /// Number of supported operations + uint32_t numOfOperations; +} SnpeUdo_ImpInfo_t; + +typedef SnpeUdo_ImpInfo_t Udo_ImpInfo_t; + +/** + * @brief This struct defines an operation. It is used for validation + * or creation of an operation. + * In case of using it for creation, the static params which are tensors + * contain pointers to the real data (weights, for example), and input/output + * tensors also include pointers to the buffers used. + */ +typedef struct +{ + /// The IP Core that the operation is defined for - CPU, GPU, DSP... + SnpeUdo_CoreType_t udoCoreType; + /// Operation type + SnpeUdo_String_t operationType; + /// The number of static parameters provided in the staticParams array. + /// this number has to match the number provided by the UDO Registration library information + uint32_t numOfStaticParams; + /// Array of static parameters + SnpeUdo_Param_t* staticParams; + /// The number of input parameters provided in inputs array. + /// this number has to match the number provided by the UDO Registration library information + uint32_t numOfInputs; + /// Array of input tensors, providing layout, data type, sizes, etc + /// When used to create an operation, also contains the initial location of the data + SnpeUdo_TensorParam_t* inputs; + /// The number of output parameters provided in inputs array. + /// this number has to match the number provided by the UDO Registration library information + uint32_t numOfOutputs; + /// Array of output tensors, providing layout, data type, sizes, etc + /// When used to create an operation, also contains the initial location of the data + SnpeUdo_TensorParam_t* outputs; +} SnpeUdo_OpDefinition_t; + +typedef SnpeUdo_OpDefinition_t Udo_OpDefinition_t; + +/** @} */ /* end_addtogroup c_plus_plus_apis C++ */ + +#endif //SNPE_UDO_BASE_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoReg.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoReg.h new file mode 100644 index 00000000..2166be59 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoReg.h @@ -0,0 +1,117 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2019-2020 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef SNPE_UDO_REG_H +#define SNPE_UDO_REG_H + +#include "SnpeUdo/UdoShared.h" + +#ifdef __cplusplus +extern "C" +{ +#endif + +/** @addtogroup c_plus_plus_apis C++ +@{ */ + +/** + * @brief Initialize the shared library's data structures. Calling any other + * library function before this one will result in an error being returned. + * + * @return Error code + */ +SnpeUdo_ErrorType_t +SnpeUdo_initRegLibrary(void); + +typedef SnpeUdo_ErrorType_t +(*SnpeUdo_InitRegLibraryFunction_t)(void); + +/** + * @brief A function to query the API version of the UDO registration library. + * The function populates a SnpeUdo_LibVersion_t struct, which contains a SnpeUdo_Version_t + * struct for API version and library version. + * + * @param[in, out] version A pointer to struct which contains major, minor, teeny information for + * library and api versions. + * + * @return Error code + */ +SnpeUdo_ErrorType_t +SnpeUdo_getRegLibraryVersion(SnpeUdo_LibVersion_t** version); + +typedef SnpeUdo_ErrorType_t +(*SnpeUdo_getRegLibraryVersion_t)(SnpeUdo_LibVersion_t** version); + +/** + * @brief Release the shared library's data structures, and invalidate any + * handles returned by the library. The behavior of any outstanding + * asynchronous calls made to this library when this function is called + * are undefined. All library functions (except SnpeUdo_InitRegLibrary) will + * return an error after this function has been successfully called. + * + * It should be possible to call SnpeUdo_InitRegLibrary after calling this + * function, and re-initialize the library. + * + * @return Error code + */ +SnpeUdo_ErrorType_t +SnpeUdo_terminateRegLibrary(void); + +typedef SnpeUdo_ErrorType_t +(*SnpeUdo_TerminateRegLibraryFunction_t)(void); + + +/** + * @brief A function to query the info on the UDO set. + * The function populates a structure which contains information about + * the package and operations contained in it. + * + * @param[in, out] registrationInfo A struct which contains information on the set of UDOs + * + * @return Error code + * + */ +SnpeUdo_ErrorType_t +SnpeUdo_getRegInfo(SnpeUdo_RegInfo_t** registrationInfo); + +typedef SnpeUdo_ErrorType_t +(*SnpeUdo_GetRegInfoFunction_t)(SnpeUdo_RegInfo_t** registrationInfo); + +/** + * @brief A function to validate that a set of params is supported by an operation + * The function receives an operation definition struct, and returns if this configuration is + * supported (e.g. if an operation can be created using this configuration) + * + * @param[in] opDefinition A struct of SnpeUdo_OpDefinition type, containing the information needed to + * validate that an operation can be created with this configuration. + * + * @return Error code, indicating is the operation can be created on this set or not. + * + */ +SnpeUdo_ErrorType_t +SnpeUdo_validateOperation(SnpeUdo_OpDefinition_t* opDefinition); + +typedef SnpeUdo_ErrorType_t +(*SnpeUdo_ValidateOperationFunction_t)(SnpeUdo_OpDefinition_t* opDefinition); + +/** @} */ /* end_addtogroup c_plus_plus_apis C++ */ + +#ifdef __cplusplus +} // extern "C" +#endif + +#endif //SNPE_UDO_REG_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoShared.h b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoShared.h new file mode 100644 index 00000000..816a8a74 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/SnpeUdo/UdoShared.h @@ -0,0 +1,57 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +//============================================================================== +// +// Copyright (c) 2019-2021 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================== + +#ifndef SNPE_UDO_SHARED_H +#define SNPE_UDO_SHARED_H + +#include "SnpeUdo/UdoBase.h" + +#ifdef __cplusplus +extern "C" +{ +#endif + +/** @addtogroup c_plus_plus_apis C++ +@{ */ + +/** + * @brief A function to return the various versions as they relate to the UDO + * The function returns a struct containing the the following: + * libVersion: the version of the implementation library compiled for the UDO. Set by user + * apiVersion: the version of the UDO API used in compiling the implementation library. + * Set by SNPE + * + * @param[in, out] version A pointer to Version struct of type SnpeUdo_LibVersion_t + * + * @return Error code + * + */ +SnpeUdo_ErrorType_t +SnpeUdo_getVersion (SnpeUdo_LibVersion_t** version); + +typedef SnpeUdo_ErrorType_t +(*SnpeUdo_GetVersionFunction_t) (SnpeUdo_LibVersion_t** version); + +typedef SnpeUdo_GetVersionFunction_t Udo_GetVersionFunction_t; + +#ifdef __cplusplus +} // extern "C" +#endif + +/** @} */ /* end_addtogroup c_plus_plus_apis C++ */ + +#endif // SNPE_UDO_SHARED_H diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/Wrapper.hpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/Wrapper.hpp new file mode 100644 index 00000000..5f908f15 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inc/zdl/Wrapper.hpp @@ -0,0 +1,449 @@ +//============================================================================= +// +// Copyright (c) 2023 Qualcomm Technologies, Inc. +// All Rights Reserved. +// Confidential and Proprietary - Qualcomm Technologies, Inc. +// +//============================================================================= +#pragma once + +#define SNPE_WRAPPER_TYPES + +#include +#include +#include +#include + +#include + +#include + + +#include "DlSystem/DlError.h" + +// Put type aliases in zdl::namespace +#define ALIAS_IN_ZDL_NAMESPACE(ns, type) namespace zdl{ namespace ns { using type = ::ns::type; }} + + +// Uncomment to print info from the Wrapper base class +//#define WRAPPER_DEBUG_PRINTS + + +#ifdef WRAPPER_DEBUG_PRINTS + +#ifdef _MSC_VER +#define WRAPPER_FUNCTION_NAME __FUNCTION__ +#define WRAPPER_TRACE() std::cout << __LINE__ << ":\t" << WRAPPER_FUNCTION_NAME << std::endl +#define WRAPPER_ETRACE() std::cout << __LINE__ << ":\t" << WRAPPER_FUNCTION_NAME << std::endl +#else +#define WRAPPER_FUNCTION_NAME __PRETTY_FUNCTION__ +#define WRAPPER_TRACE() std::cout << "\e[33m" << __LINE__ << ":\t" << WRAPPER_FUNCTION_NAME << "\e[0m" << std::endl +#define WRAPPER_ETRACE() std::cout << "\e[31m" << __LINE__ << ":\t" << WRAPPER_FUNCTION_NAME << "\e[0m" << std::endl +#endif + +#include +#else +#define WRAPPER_TRACE() do{}while(0) +#define WRAPPER_ETRACE() do{}while(0) +#endif + + +namespace WrapperDetail { + + +template +using GetterFuncType = MemberType(*)(HandleType); + +template +using SetterFuncType = Snpe_ErrorCode_t(*)(HandleType, MemberType); + + + +// Allow Wrappers to have members that require CAPI calls for access +template GetterFunc, + SetterFuncType SetterFunc +> +class GenericMemberReference{ + OwnerType& owner; +public: + + + ~GenericMemberReference() = default; + GenericMemberReference() = delete; + + GenericMemberReference(const GenericMemberReference&) = delete; + GenericMemberReference(GenericMemberReference&&) noexcept = default; + + GenericMemberReference(OwnerType& owner) + : owner{owner} + { } + explicit GenericMemberReference(OwnerType& owner, MemberType member) + : owner{owner} + { + operator=(member); + } + GenericMemberReference& operator=(MemberType member){ + SetterFunc(owner.handle(), member); + return *this; + } + + operator MemberType() const{ + return GetterFunc(owner.handle()); + } + + GenericMemberReference& + operator=(const GenericMemberReference& other){ + return operator=(other.operator MemberType()); + } + + MemberType operator()() const{ + return operator MemberType(); + } + +}; + +// Allow Wrappers to have members that require CAPI calls for access +template GetterFunc +> +class GenericConstMemberReference{ + + OwnerType& owner; + +public: + ~GenericConstMemberReference() = default; + GenericConstMemberReference() = delete; + + GenericConstMemberReference(const GenericConstMemberReference&) = delete; + GenericConstMemberReference(GenericConstMemberReference&&) noexcept = default; + + GenericConstMemberReference(OwnerType& owner) + : owner{owner} + { } + + operator MemberType() const{ + return GetterFunc(owner.handle()); + } + + + template::value,int>::Type=0> + operator const char*() const{ + thread_local std::string tlss; + tlss = operator MemberType(); + return tlss.c_str(); + } + + MemberType operator()() const{ + return operator MemberType(); + } + +}; + + + +// Allows returning references to literals through the CAPI's _Get and _Set functions +template +using GetterIndexedFuncType = MemberType(*)(HandleType, IndexType); + +template +using SetterIndexedFuncType = Snpe_ErrorCode_t(*)(HandleType, IndexType, MemberType); + +template GetterFunc, + SetterIndexedFuncType SetterFunc +> +class MemberIndexedReference{ + OwnerType& owner; + IndexType idx; + +public: + MemberIndexedReference(OwnerType& owner, IndexType idx) + : owner{owner}, + idx{idx} + { } + MemberIndexedReference(const MemberIndexedReference&) noexcept = default; + MemberIndexedReference(MemberIndexedReference&&) noexcept = default; + + MemberIndexedReference& operator=(const MemberIndexedReference&) noexcept = default; + MemberIndexedReference& operator=(MemberIndexedReference&&) noexcept = default; + + MemberIndexedReference operator=(MemberType member){ + SetterFunc(owner.handle(), idx, member); + return *this; + } + + operator MemberType() const{ + return GetterFunc(owner.handle(), idx); + } + +}; + + + +// Allow moving ownership of handles +template +struct HandleMover { + Handle handle; + bool isReference; +}; + +template +HandleMover moveHandle(Handle handle, bool isReference = false){ + return {handle, isReference}; +} + +// Virtual base class to allow for WrapperStorage to hold pointers to any Wrapper type +class WrapperBase{ +public: + virtual ~WrapperBase() = default; +}; + +// Storage type for Wrappers. Will have a set if the CAPI type is capable of creating reference handles +template +struct WrapperStorage{ + Handle handle; + bool isReference; + constexpr WrapperStorage(Handle handle = {}, bool isReference = false) noexcept + : handle{handle}, + isReference{isReference} + { } +}; + +template +struct WrapperStorage{ + Handle handle; + bool isReference; + mutable std::set> referencedObjects; + WrapperStorage(Handle handle = {}, bool isReference = false) noexcept + : handle{handle}, + isReference{isReference} + { } +}; + +// Allow a handle to be unbound from a Wrapper +struct HandleReleaser{ + template + static typename WrapperType::HandleType release(WrapperType& wrapper){ + auto toret = wrapper.m_Storage.handle; + wrapper.m_Storage.handle = {}; + return toret; + } +}; + +} // ns WrapperDetail + + + +// The base class for all Wrappers around the CAPI +// NOTE: This Wrapper class leverages the Curiously Recurring Template Pattern (CRTP) +template +class Wrapper : public WrapperDetail::WrapperBase{ + friend struct WrapperDetail::HandleReleaser; + // Allow certain types to access getHandle() and handle() + template + friend class Wrapper; + + template, + WrapperDetail::SetterIndexedFuncType> + friend class WrapperDetail::MemberIndexedReference; + + template> + friend class WrapperDetail::GenericConstMemberReference; + + template, WrapperDetail::SetterFuncType> + friend class WrapperDetail::GenericMemberReference; + + + +protected: + using HandleType = Handle; + using BaseType = Wrapper; + using DeleteFunctionType = Snpe_ErrorCode_t(*)(Handle); + + using StorageType = WrapperDetail::WrapperStorage; + + + template Getter> + static WrapperValueType CastingGetter(HandleType handle){ + return static_cast(Getter(handle)); + } + template Setter> + static Snpe_ErrorCode_t CastingSetter(HandleType handle, WrapperValueType value){ + return Setter(handle,static_cast(value)); + } + + + template + struct WrapperMemberReference{ + Derived& owner; + + WrapperMemberReference(Derived& owner) + : owner{owner} + { } + WrapperMemberReference(Derived& owner, const RlType& other) + : owner{owner} + { + operator=(other); + } + + WrapperMemberReference& operator=(const RlType& rl){ + Setter(getHandle(owner), getHandle(rl)); + return *this; + } + + operator RlType&() { + return *owner.template makeReference( Getter(getHandle(owner)) ); + } + operator RlType&() const { + return *owner.template makeReference( Getter(getHandle(owner)) ); + } + + RlType& operator()(){ + return operator RlType&(); + } + const RlType& operator()() const{ + return operator RlType&(); + } + }; + + // For Factory/Singleton types, we need a way for the deleter to do nothing + static Snpe_ErrorCode_t NoOpDeleter(Handle){ + return SNPE_SUCCESS; + } + + // Simplify calls to WrapperDetail::moveHandle. Can be removed, but will require updating all calls to moveHandle + template + static WrapperDetail::HandleMover moveHandle(H handle, bool isReference = false){ + return WrapperDetail::moveHandle(handle, isReference); + } + + + HandleType& handle() noexcept{ return m_Storage.handle; } + const HandleType& handle() const noexcept{ return m_Storage.handle; } + + bool isReference() const noexcept{ return m_Storage.isReference; } + + void Dtor(){ + if(!isReference() && !handle()){ + if(Derived::DeleteFunction != NoOpDeleter){ + WRAPPER_ETRACE(); + } + } + if(!isReference() && handle()){ + WRAPPER_TRACE(); +#ifdef WRAPPER_DEBUG_PRINTS + auto status = Derived::DeleteFunction(handle()); + if(status != SNPE_SUCCESS){ + WRAPPER_ETRACE(); + } +#else + Derived::DeleteFunction(handle()); +#endif + + handle() = nullptr; + } else { + WRAPPER_TRACE(); + } + } + +protected: + + // Only compile these if the class creates references. This will save memory and time + template::type=0> + void addReference(WrapperBase* wrapperBase) const{ // accesses mutable member + if(!wrapperBase){ + WRAPPER_ETRACE(); + } + m_Storage.referencedObjects.insert(std::unique_ptr(wrapperBase)); + } + + template::type=0> + T* makeReference(H referenceHandle) const{ + if(!referenceHandle){ + WRAPPER_ETRACE(); + return nullptr; + } + auto refObj = new T(moveHandle(referenceHandle, true)); + addReference(refObj); + return refObj; + } + + // This will be used to access another Wrapped type's handles once handle() is made protected + template + static OtherHandle getHandle(const Wrapper& otherObject){ + return otherObject.handle(); + } + + template + static OtherHandle getHandle(const Wrapper* otherObject){ + if(!otherObject) return {}; + return getHandle(*otherObject); + } + + template + static std::unique_ptr makeUnique(H handle){ + if(!handle) return {}; + return std::unique_ptr(new T(moveHandle(handle))); + } + + +public: + ~Wrapper(){ + Dtor(); + } +protected: + // Only derived types should have access to this + Wrapper(HandleType handle, bool isReference = false) + : m_Storage{handle, isReference} + { WRAPPER_TRACE(); } + +public: + // We should never have an empty wrapper + Wrapper() = delete; + + // Move semantics are essentially free for all wrapper types + Wrapper(Wrapper&& other) noexcept + : m_Storage{std::move(other.m_Storage)} + { + WRAPPER_TRACE(); + other.handle() = nullptr; + } + Wrapper(const Wrapper&) = delete; + + + Wrapper& operator=(Wrapper&& other) noexcept{ + WRAPPER_TRACE(); + if(this != &other){ + std::swap(m_Storage, other.m_Storage); + other.Dtor(); + } + return *this; + } + Wrapper& operator=(const Wrapper&) = delete; + + + // Allow a CAPI handle to be taken over by a Wrapper + Wrapper(WrapperDetail::HandleMover handleMover) noexcept + : Wrapper(handleMover.handle, handleMover.isReference) + { WRAPPER_TRACE(); } + +protected: + // Simplify Derived's move assignment operators + Derived& moveAssign(Derived&& other) noexcept{ WRAPPER_TRACE(); + return static_cast(operator=(std::move(other))); + } + + +private: + StorageType m_Storage; + +}; diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference.cpp new file mode 100644 index 00000000..d0cd64da --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference.cpp @@ -0,0 +1,194 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include "android/log.h" + +#include "hpp/CheckRuntime.hpp" +#include "hpp/SetBuilderOptions.hpp" +#include "hpp/Util.hpp" +#include "LoadContainer.hpp" +#include "CreateUserBuffer.hpp" +#include "LoadInputTensor.hpp" + +#include +#include +#include + +std::unique_ptr snpe_HRNET; +std::unique_ptr snpe_BB; + +std::mutex mtx; +static zdl::DlSystem::Runtime_t runtime = zdl::DlSystem::Runtime_t::CPU; +static zdl::DlSystem::RuntimeList runtimeList; +bool useUserSuppliedBuffers = true; +bool useIntBuffer = false; + +zdl::DlSystem::UserBufferMap inputMap, outputMap; +std::vector > snpeUserBackedInputBuffers, snpeUserBackedOutputBuffers; +std::unordered_map > applicationOutputBuffers; +std::unordered_map > applicationInputBuffers; +int bitWidth = 32; + + +#include +#include + +std::string build_network_BB(const uint8_t * dlc_buffer_BB, const size_t dlc_size_BB, const char runtime_arg, ModelName modelName) +{ + std::string outputLogger; + bool usingInitCaching = false; //shubham: TODO check with true + + std::unique_ptr container_BB = nullptr ; + + container_BB = loadContainerFromBuffer(dlc_buffer_BB, dlc_size_BB); + + if (container_BB == nullptr) { + LOGE("Error while opening the container file."); + return "Error while opening the container file.\n"; + } + + runtimeList.clear(); + LOGI("runtime arg %c",runtime_arg); + zdl::DlSystem::Runtime_t runtime = zdl::DlSystem::Runtime_t::CPU; + if (runtime_arg == 'D'){ + runtime = zdl::DlSystem::Runtime_t::DSP; + LOGI("Added DSP"); + } + else if (runtime_arg == 'G') + { + runtime = zdl::DlSystem::Runtime_t::GPU_FLOAT32_16_HYBRID; //can be written as GPU + LOGI("Added GPU"); + } + + if(runtime != zdl::DlSystem::Runtime_t::UNSET) + { + bool ret = runtimeList.add(checkRuntime(runtime)); + if(ret == false){ + LOGE("Cannot set runtime"); + return outputLogger + "\nCannot set runtime"; + } + } else { + return outputLogger + "\nCannot set runtime"; + } + + + mtx.lock(); + snpe_BB = setBuilderOptions(container_BB, runtime, runtimeList, useUserSuppliedBuffers, usingInitCaching, modelName); + mtx.unlock(); + + if (snpe_BB == nullptr) { + LOGE("SNPE Prepare failed: Builder option failed for BB"); + outputLogger += "Model Prepare failed for BB"; + return outputLogger + "SNPE Prepare failed for BB"; + } + + outputLogger += "\nBB Model Network Prepare success !!!\n"; + + //Creating Buffer + createInputBufferMap(inputMap, applicationInputBuffers, snpeUserBackedInputBuffers, snpe_BB, useIntBuffer, bitWidth); + createOutputBufferMap(outputMap, applicationOutputBuffers, snpeUserBackedOutputBuffers, snpe_BB, useIntBuffer, bitWidth); + return outputLogger; +} + + + +bool executeDLC(cv::Mat &img, int orig_width, int orig_height, int &numberofobj, std::vector> &BB_coords, std::vector &BB_names, Model *modelobj) { + + LOGI("execute_net_BB"); + ATrace_beginSection("preprocessing"); + + struct timeval start_time, end_time; + float milli_time, seconds, useconds; + + mtx.lock(); + assert(snpe_BB!=nullptr); + + if(!loadInputUserBuffer(applicationInputBuffers, snpe_BB, img, inputMap, bitWidth, modelobj)) + { + LOGE("Failed to load Input UserBuffer"); + mtx.unlock(); + return false; + } + + //std::string name_out_boxes = "885"; + //std::string name_out_classes = "877"; + + // get output tensor names of the network that need to be populated + const auto &outputNamesOpt = snpe_BB->getOutputTensorNames(); + if (!outputNamesOpt) throw std::runtime_error("Error obtaining output tensor names"); + const zdl::DlSystem::StringList &outputNames = *outputNamesOpt; + assert(outputNames.size() > 0); + + if (outputNames.size()) LOGI("Preprocessing and loading in application Output Buffer for BB"); + + std::string name_out_boxes; + + //YoloX is using only single output tensor + if (modelobj->model_name != YoloX) { + name_out_boxes = outputNames.at(1); + LOGI("Filling %s buffer name_out_boxes", name_out_boxes.c_str()); + } + + + std::string name_out_classes = outputNames.at(0); + LOGI("Filling %s buffer name_out_classes", name_out_classes.c_str()); + + ATrace_endSection(); + gettimeofday(&start_time, NULL); + ATrace_beginSection("inference time"); + + bool execStatus = snpe_BB->execute(inputMap, outputMap); + ATrace_endSection(); + ATrace_beginSection("postprocessing time"); + gettimeofday(&end_time, NULL); + seconds = end_time.tv_sec - start_time.tv_sec; //seconds + useconds = end_time.tv_usec - start_time.tv_usec; //milliseconds + milli_time = ((seconds) * 1000 + useconds/1000.0); + //LOGI("Inference time %f ms", milli_time); + + if(execStatus== true){ + LOGI("Exec BB status is true"); + } + else{ + LOGE("Exec BB status is false"); + mtx.unlock(); + return false; + } + + std::vector BBout_boxcoords; + + //YoloX is using only single output tensor + if (modelobj->model_name != YoloX) { + LOGI("reading output name_out_boxes"); + BBout_boxcoords = applicationOutputBuffers.at(name_out_boxes); + } + + LOGI("reading output name_out_classes"); + std::vector BBout_class = applicationOutputBuffers.at(name_out_classes); + //LOGI("reading output done. Calling postprocess"); + + modelobj->postprocess(orig_width, orig_height, numberofobj, BB_coords, BB_names, BBout_boxcoords, BBout_class, milli_time); + + ATrace_endSection(); + mtx.unlock(); + return true; +} + diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference_helper.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference_helper.cpp new file mode 100644 index 00000000..bef75104 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/inference_helper.cpp @@ -0,0 +1,288 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +#include +#include +#include +#include +#include +#include +#include "android/log.h" + +#include "zdl/SNPE/SNPE.hpp" +#include "zdl/SNPE/SNPEFactory.hpp" +#include "zdl/DlSystem/DlVersion.hpp" +#include "zdl/DlSystem/DlEnums.hpp" +#include "zdl/DlSystem/String.hpp" +#include "zdl/DlContainer/IDlContainer.hpp" +#include "zdl/SNPE/SNPEBuilder.hpp" +#include "zdl/DlSystem/ITensor.hpp" +#include "zdl/DlSystem/StringList.hpp" +#include "zdl/DlSystem/TensorMap.hpp" +#include "zdl/DlSystem/TensorShape.hpp" +#include "DlSystem/ITensorFactory.hpp" + +#include "hpp/LoadInputTensor.hpp" +#include "hpp/Util.hpp" +#include "inference.h" + +bool SetAdspLibraryPath(std::string nativeLibPath) { + nativeLibPath += ";/data/local/tmp/mv_dlc;/vendor/lib/rfsa/adsp;/vendor/dsp/cdsp;/system/lib/rfsa/adsp;/system/vendor/lib/rfsa/adsp;/dsp"; + + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "ADSP Lib Path = %s \n", nativeLibPath.c_str()); + std::cout << "ADSP Lib Path = " << nativeLibPath << std::endl; + + return setenv("ADSP_LIBRARY_PATH", nativeLibPath.c_str(), 1 /*override*/) == 0; +} + + +std::unique_ptr loadContainerFromBuffer(const uint8_t * buffer, const size_t size) +{ + std::unique_ptr container; + container = zdl::DlContainer::IDlContainer::open(buffer, size); + return container; +} + + +zdl::DlSystem::Runtime_t checkRuntime(zdl::DlSystem::Runtime_t runtime) +{ + static zdl::DlSystem::Version_t Version = zdl::SNPE::SNPEFactory::getLibraryVersion(); + + LOGI("SNPE Version = %s", Version.asString().c_str()); //Print Version number + + if (!zdl::SNPE::SNPEFactory::isRuntimeAvailable(runtime)) { + LOGE("Selected runtime not present. Falling back to GPU."); + runtime = zdl::DlSystem::Runtime_t::GPU; + if (!zdl::SNPE::SNPEFactory::isRuntimeAvailable(runtime)){ + LOGE("Selected runtime not present. Falling back to CPU."); + runtime = zdl::DlSystem::Runtime_t::CPU; + } + } + + return runtime; +} + +std::unique_ptr setBuilderOptions(std::unique_ptr & container, + zdl::DlSystem::Runtime_t runtime, + zdl::DlSystem::RuntimeList runtimeList, + bool useUserSuppliedBuffers, + bool useCaching, + ModelName modelName) +{ + std::unique_ptr snpe; + zdl::SNPE::SNPEBuilder snpeBuilder(container.get()); + + if(runtimeList.empty()) + { + runtimeList.add(runtime); + } + + std::string platformOptionStr = "useAdaptivePD:ON"; +// if (isSignedStatus == UNSIGNED_PD) { + // use unsignedPD feature for untrusted app. + // platformOptionStr += "unsignedPD:ON"; +// } + zdl::DlSystem::PlatformConfig platformConfig; + bool setSuccess = platformConfig.setPlatformOptions(platformOptionStr); + if (!setSuccess) + LOGE("=========> failed to set platformconfig: %s", platformOptionStr.c_str()); + else + LOGI("=========> platformconfig set: %s", platformOptionStr.c_str()); + + bool isValid = platformConfig.isOptionsValid(); + if (!isValid) + LOGE("=========> platformconfig option is invalid"); + else + LOGI("=========> platformconfig option: valid"); + + + zdl::DlSystem::StringList stringruntime = runtimeList.getRuntimeListNames(); + for (const char *name : stringruntime) + LOGI("runtime sh %s", name); + + snpe = snpeBuilder.setOutputLayers({}) + .setPerformanceProfile(zdl::DlSystem::PerformanceProfile_t::BURST) + .setExecutionPriorityHint( + zdl::DlSystem::ExecutionPriorityHint_t::HIGH) + .setRuntimeProcessorOrder(runtimeList) + .setUseUserSuppliedBuffers(useUserSuppliedBuffers) + .setPlatformConfig(platformConfig) + .setInitCacheMode(useCaching) + .setUnconsumedTensorsAsOutputs(true) + .build(); + + return snpe; + +} + +// ==============================User Buffer func=================================== // +// ================================================================================= // + + +//CreateUserbuffer INPUT/OUTPUT for BB +void createUserBuffer(zdl::DlSystem::UserBufferMap& userBufferMap, + std::unordered_map>& applicationBuffers, + std::vector>& snpeUserBackedBuffers, + std::unique_ptr& snpe, + const char * name, + const bool isTfNBuffer, + int bitWidth) +{ + + auto bufferAttributesOpt = snpe->getInputOutputBufferAttributes(name); + if (!bufferAttributesOpt) throw std::runtime_error(std::string("Error obtaining attributes for input tensor ") + name); + + // calculate the size of buffer required by the input tensor + const zdl::DlSystem::TensorShape& bufferShape = (*bufferAttributesOpt)->getDims(); + + size_t bufferElementSize = 0; + if (isTfNBuffer) { + bufferElementSize = bitWidth / 8; + } + else { + bufferElementSize = sizeof(float); + } + + // Calculate the stride based on buffer strides. + // Note: Strides = Number of bytes to advance to the next element in each dimension. + // For example, if a float tensor of dimension 2x4x3 is tightly packed in a buffer of 96 bytes, then the strides would be (48,12,4) + // Note: Buffer stride is usually known and does not need to be calculated. + + int num_dims = bufferShape.rank(); //bufferShape rank is generally 1 more than expected, as it add 1 for batchSize, so 320x320x3 will look like 1x320x320x3 + std::vector strides(num_dims); + strides[strides.size() - 1] = bufferElementSize; + size_t stride = strides[strides.size() - 1]; + for (size_t i = num_dims - 1; i > 0; i--) { + stride *= bufferShape[i]; + strides[i - 1] = stride; + //LOGI("\nstrides[%d]: %d",i,stride); + //LOGI("\nbuffershape[%d]: %d",i,bufferShape[i]); + } + + size_t bufSize=bufferElementSize; + for(int i=0;i userBufferEncoding; + if (isTfNBuffer) + userBufferEncoding = std::unique_ptr( + new zdl::DlSystem::UserBufferEncodingTfN(0, 1.0, bitWidth)); + else + userBufferEncoding = std::unique_ptr( + new zdl::DlSystem::UserBufferEncodingFloat()); + + // create user-backed storage to load input data onto it + applicationBuffers.emplace(name, std::vector(bufSize)); + + // create SNPE user buffer from the user-backed buffer + zdl::DlSystem::IUserBufferFactory &ubFactory = zdl::SNPE::SNPEFactory::getUserBufferFactory(); + snpeUserBackedBuffers.push_back( + ubFactory.createUserBuffer(applicationBuffers.at(name).data(), + bufSize, + strides, + userBufferEncoding.get())); + if (snpeUserBackedBuffers.back() == nullptr) + throw std::runtime_error(std::string("Error while creating user buffer.")); + + // add the user-backed buffer to the inputMap, which is later on fed to the network for execution + userBufferMap.add(name, snpeUserBackedBuffers.back().get()); + +} + +/* + Cretae OutPut Buffer Map for BB + */ +void createOutputBufferMap(zdl::DlSystem::UserBufferMap& outputMap, + std::unordered_map>& applicationBuffers, + std::vector>& snpeUserBackedBuffers, + std::unique_ptr& snpe, + bool isTfNBuffer, + int bitWidth) +{ + //LOGI("Creating Output Buffer for BB"); + const auto& outputNamesOpt = snpe->getOutputTensorNames(); + if (!outputNamesOpt) throw std::runtime_error("Error obtaining output tensor names"); + const zdl::DlSystem::StringList& outputNames = *outputNamesOpt; + + // create SNPE user buffers for each application storage buffer + for (const char *name : outputNames) { + LOGI("Creating output buffer %s", name); + createUserBuffer(outputMap, applicationBuffers, snpeUserBackedBuffers, snpe, name, isTfNBuffer, bitWidth); + } + +} +/* + * Create Input Buffer Map for BB + */ +void createInputBufferMap(zdl::DlSystem::UserBufferMap& inputMap, + std::unordered_map>& applicationBuffers, + std::vector>& snpeUserBackedBuffers, + std::unique_ptr& snpe, + bool isTfNBuffer, + int bitWidth) { + //LOGI("Creating Input Buffer for BB"); + const auto &inputNamesOpt = snpe->getInputTensorNames(); + if (!inputNamesOpt) throw std::runtime_error("Error obtaining input tensor names"); + const zdl::DlSystem::StringList &inputNames = *inputNamesOpt; + assert(inputNames.size() > 0); + + // create SNPE user buffers for each application storage buffer + for (const char *name: inputNames) { + LOGI("Creating Input Buffer = %s", name); + createUserBuffer(inputMap, applicationBuffers, snpeUserBackedBuffers, snpe, name, + isTfNBuffer, bitWidth); + } +} + +//Preprocessing and loading in application Input Buffer for BB +bool loadInputUserBuffer(std::unordered_map>& applicationBuffers, + std::unique_ptr& snpe, + cv::Mat &img, + zdl::DlSystem::UserBufferMap& inputMap, + int bitWidth, Model *modelobj) { + + // get input tensor names of the network that need to be populated + const auto &inputNamesOpt = snpe->getInputTensorNames(); + if (!inputNamesOpt) throw std::runtime_error("Error obtaining input tensor names"); + const zdl::DlSystem::StringList &inputNames = *inputNamesOpt; + assert(inputNames.size() > 0); + + if (inputNames.size()) LOGI("Preprocessing and loading in application Input Buffer for BB"); + + + for (size_t j = 0; j < inputNames.size(); j++) { + const char *name = inputNames.at(j); + LOGI("Filling %s buffer ", name); + + if(bitWidth == 8 || bitWidth == 16) { + LOGE("bitwidth 8 and 16 are NOT DEFINED"); + return false; + } else { + + std::vector dims; + auto bufferAttributesOpt = snpe->getInputOutputBufferAttributes(name); + if (!bufferAttributesOpt) throw std::runtime_error(std::string("Error obtaining attributes for input tensor ") + name); + + const zdl::DlSystem::TensorShape& bufferShape = (*bufferAttributesOpt)->getDims(); + int num_dims = bufferShape.rank(); + for(int i=0;ipreprocess(applicationBuffers.at(name),img, dims); //functions loads data in applicationBuffer + + } + } + return true; +} diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/objectdetectionYoloNas.cpp b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/objectdetectionYoloNas.cpp new file mode 100644 index 00000000..c53cfa54 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/cpp/objectdetectionYoloNas.cpp @@ -0,0 +1,195 @@ +// -*- mode: cpp -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +#include +using namespace cv; +#include +#include +#include +#include +#include + +#include "hpp/inference.h" +#include "hpp/Util.hpp" + +#include "zdl/SNPE/SNPE.hpp" +#include "zdl/SNPE/SNPEFactory.hpp" + +#include "YOLONAS_Model.h" +#include "SSDMobileNetV2_Model.h" +#include "YOLO_X_Model.h" + +Model *modelobj; + +extern "C" JNIEXPORT jstring JNICALL +Java_com_qcom_aistack_1objdetect_SNPEHelper_queryRuntimes( + JNIEnv* env, + jobject /* this */, + jstring native_dir_path) { + const char *cstr = env->GetStringUTFChars(native_dir_path, nullptr); + env->ReleaseStringUTFChars(native_dir_path, cstr); + + std::string runT_Status; + std::string nativeLibPath = std::string(cstr); + +// runT_Status += "\nLibs Path : " + nativeLibPath + "\n"; + + if (!SetAdspLibraryPath(nativeLibPath)) { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "Failed to set ADSP Library Path\n"); + + runT_Status += "\nFailed to set ADSP Library Path\nTerminating"; + return env->NewStringUTF(runT_Status.c_str()); + } + + // ====================================================================================== // + runT_Status = "Querying Runtimes : \n\n"; + // DSP unsignedPD check + if (!zdl::SNPE::SNPEFactory::isRuntimeAvailable(zdl::DlSystem::Runtime_t::DSP,zdl::DlSystem::RuntimeCheckOption_t::UNSIGNEDPD_CHECK)) { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "UnsignedPD DSP runtime : Absent\n"); + runT_Status += "UnsignedPD DSP runtime : Absent\n"; + } + else { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "UnsignedPD DSP runtime : Present\n"); + runT_Status += "UnsignedPD DSP runtime : Present\n"; + } + // DSP signedPD check + if (!zdl::SNPE::SNPEFactory::isRuntimeAvailable(zdl::DlSystem::Runtime_t::DSP)) { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "DSP runtime : Absent\n"); + runT_Status += "DSP runtime : Absent\n"; + } + else { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "DSP runtime : Present\n"); + runT_Status += "DSP runtime : Present\n"; + } + // GPU check + if (!zdl::SNPE::SNPEFactory::isRuntimeAvailable(zdl::DlSystem::Runtime_t::GPU)) { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "GPU runtime : Absent\n"); + runT_Status += "GPU runtime : Absent\n"; + } + else { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "GPU runtime : Present\n"); + runT_Status += "GPU runtime : Present\n"; + } + // CPU check + if (!zdl::SNPE::SNPEFactory::isRuntimeAvailable(zdl::DlSystem::Runtime_t::CPU)) { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "CPU runtime : Absent\n"); + runT_Status += "CPU runtime : Absent\n"; + } + else { + __android_log_print(ANDROID_LOG_INFO, "SNPE ", "CPU runtime : Present\n"); + runT_Status += "CPU runtime : Present\n"; + } + + return env->NewStringUTF(runT_Status.c_str()); +} + + + +//initializing network +extern "C" +JNIEXPORT jstring JNICALL +Java_com_qcom_aistack_1objdetect_SNPEHelper_initSNPE(JNIEnv *env, jobject thiz, jobject asset_manager, jchar runtime, jstring jdlc_name) { + LOGI("Reading SNPE DLC ..."); + std::string result; + + //AAssetManager* mgr = AAssetManager_fromJava(env, asset_manager); + //AAsset* asset_BB = AAssetManager_open(mgr, "Quant_yoloNas_s_320.dlc", AASSET_MODE_UNKNOWN); + + const char *cstr = env->GetStringUTFChars(jdlc_name, 0); + AAssetManager* mgr = AAssetManager_fromJava(env, asset_manager); + AAsset* asset_BB = AAssetManager_open(mgr, cstr, AASSET_MODE_UNKNOWN); + + if(strcmp(cstr,"Quant_yoloNas_s_320.dlc")==0) { + LOGI("Quant_yoloNas_s_320 dlc"); + } + + modelobj= new YOLONAS_Model(); + + //Changing PrePost for different models + if (strcmp(cstr,"ssd_mobilenetV2_without_ABP-NMS_Q.dlc")==0){ + LOGI("ssd_mobilenetV2_without_ABP-NMS_Q dlc"); + modelobj = new SSDMobileNetV2_Model(); + modelobj->msg(); + } + else if(strcmp(cstr,"yolox_x_212_Q.dlc")==0){ + LOGI("YOLO_X dlc"); + modelobj = new YOLO_X_Model(); + modelobj->msg(); + } + + env->ReleaseStringUTFChars(jdlc_name, cstr); + + + if (NULL == asset_BB) { + LOGE("Failed to load ASSET, needed to load DLC\n"); + result = "Failed to load ASSET, needed to load DLC\n"; + return env->NewStringUTF(result.c_str()); + } + + long dlc_size_BB = AAsset_getLength(asset_BB); + LOGI("DLC BB Size = %ld MB\n", dlc_size_BB / (1024*1024)); + result += "DLC BB Size = " + std::to_string(dlc_size_BB); + char* dlc_buffer_BB = (char*) malloc(sizeof(char) * dlc_size_BB); + AAsset_read(asset_BB, dlc_buffer_BB, dlc_size_BB); + + result += "\n\nBuilding Models DLC Network:\n"; + result += build_network_BB(reinterpret_cast(dlc_buffer_BB), dlc_size_BB,runtime, modelobj->model_name); + + return env->NewStringUTF(result.c_str()); +} + + +//inference +extern "C" +JNIEXPORT jint JNICALL +Java_com_qcom_aistack_1objdetect_SNPEHelper_inferSNPE(JNIEnv *env, jobject thiz, jlong inputMat, jint actual_width, jint actual_height, + jobjectArray jboxcoords, jobjectArray objnames) { + + LOGI("infer SNPE S"); + + cv::Mat &img = *(cv::Mat*) inputMat; + std::string bs; + int numberofobj = 0; + std::vector> BB_coords; + std::vector BB_names; + + bool status = executeDLC(img,actual_width, actual_height, numberofobj, BB_coords, BB_names, modelobj); + + if(numberofobj ==0) + { + LOGI("No object detected"); + } + else if (numberofobj == -1){ + LOGE("ERROR in loading model properly"); + return -1; + } + else if(status == false) + { + LOGE("fatal ERROR"); + return 0; + } + else { + //LOGI("number of detected objects: %d",numberofobj); + + for (int z = 0; z < numberofobj; z++){ + jfloatArray boxcoords = (jfloatArray) env->GetObjectArrayElement(jboxcoords, z); + env->SetObjectArrayElement(objnames, z,env->NewStringUTF(BB_names[z].data())); + + + float tempbox[5]; //4 coords and 1 processing time + for(int k=0;k<5;k++) + tempbox[k]=BB_coords[z][k]; + env->SetFloatArrayRegion(boxcoords,0,5,tempbox); + } + //LOGI("executeDLC_returned successfully"); + } + //LOGD("infer SNPE E"); + return numberofobj; + +} \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/ic_launcher-playstore.png b/ai-solutions/android/03-ObjectDetection/app/src/main/ic_launcher-playstore.png new file mode 100644 index 0000000000000000000000000000000000000000..58cdfd91ddaca745b7fc2f02910ffd358ad3b001 GIT binary patch literal 136308 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⁡d#VWVXuk9smii6FgC5q%jGTxL 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 Tc>&8jKr&7ocXeW-gHrzkg*CDl literal 0 HcmV?d00001 diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/CameraFragment.java b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/CameraFragment.java new file mode 100644 index 00000000..4534ddf7 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/CameraFragment.java @@ -0,0 +1,1131 @@ +// -*- mode: java -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= + +package com.qcom.aistack_objdetect; + +import android.Manifest; +import android.app.ProgressDialog; +import android.content.Context; +import android.content.pm.PackageManager; +import android.graphics.Bitmap; +import android.graphics.Matrix; +import android.graphics.SurfaceTexture; +import android.hardware.camera2.CameraAccessException; +import android.hardware.camera2.CameraCaptureSession; +import android.hardware.camera2.CameraDevice; +import android.hardware.camera2.CameraManager; +import android.hardware.camera2.CaptureFailure; +import android.hardware.camera2.CaptureRequest; +import android.hardware.camera2.CaptureResult; +import android.hardware.camera2.TotalCaptureResult; +import android.os.Bundle; +import android.os.Handler; +import android.os.HandlerThread; +import android.support.annotation.NonNull; +import android.support.v4.app.Fragment; +import android.support.v4.content.ContextCompat; +import android.view.LayoutInflater; +import android.view.Surface; +import android.view.TextureView; +import android.view.View; +import android.view.ViewGroup; + +import java.util.ArrayList; +import java.util.Arrays; + + +import android.app.Activity; +import android.app.AlertDialog; +import android.app.Dialog; +import android.content.DialogInterface; +import android.graphics.ImageFormat; +import android.graphics.Point; +import android.graphics.RectF; +import android.hardware.camera2.CameraCharacteristics; +import android.hardware.camera2.CameraMetadata; +import android.hardware.camera2.params.StreamConfigurationMap; +import android.media.Image; +import android.media.ImageReader; +import android.support.v4.app.DialogFragment; +import android.util.Size; +import android.util.SparseIntArray; +import android.widget.Toast; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.nio.ByteBuffer; +import java.util.Collections; +import java.util.Comparator; +import java.util.List; +import java.util.concurrent.Semaphore; +import java.util.concurrent.TimeUnit; + + + +public class CameraFragment extends Fragment +{ + + /** + * Conversion from screen rotation to JPEG orientation. + */ + private SNPEHelper mSnpeHelper; + public long tic = 0,tic2=0; + private boolean mNetworkLoaded; + + private ProgressDialog dialog_model_error; + + private FragmentRender mFragmentRender; + + public int fps=0,frame_count =-1; + public static char runtime_var; + + public static String dlc_name_var; + + private String mCameraId; + private static final SparseIntArray ORIENTATIONS = new SparseIntArray(); + private static final int REQUEST_CAMERA_PERMISSION = 1; + private static final String FRAGMENT_DIALOG = "dialog"; + + static { + ORIENTATIONS.append(Surface.ROTATION_0, 90); + ORIENTATIONS.append(Surface.ROTATION_90, 0); + ORIENTATIONS.append(Surface.ROTATION_180, 270); + ORIENTATIONS.append(Surface.ROTATION_270, 180); + } + + + /** + * Camera state: Showing camera preview. + */ + private static final int STATE_PREVIEW = 0; + + /** + * Camera state: Waiting for the focus to be locked. + */ + private static final int STATE_WAITING_LOCK = 1; + + /** + * Camera state: Waiting for the exposure to be precapture state. + */ + private static final int STATE_WAITING_PRECAPTURE = 2; + + /** + * Camera state: Waiting for the exposure state to be something other than precapture. + */ + private static final int STATE_WAITING_NON_PRECAPTURE = 3; + + /** + * Camera state: Picture was taken. + */ + private static final int STATE_PICTURE_TAKEN = 4; + + /** + * Max preview width that is guaranteed by Camera2 API + */ + private static final int MAX_PREVIEW_WIDTH = 1920; + + /** + * Max preview height that is guaranteed by Camera2 API + */ + private static final int MAX_PREVIEW_HEIGHT = 1080; + + private TextureView mTextureView; + /** + * {@link TextureView.SurfaceTextureListener} handles several lifecycle events on a + * {@link TextureView}. + + */ + private final TextureView.SurfaceTextureListener mSurfaceTextureListener + = new TextureView.SurfaceTextureListener() { + + @Override + public void onSurfaceTextureAvailable(SurfaceTexture texture, int width, int height) { +// System.out.println("Textureiew widthxheight: "+mTextureView.getWidth()+ mTextureView.getHeight()); + openCamera(width, height); + } + + @Override + public void onSurfaceTextureSizeChanged(SurfaceTexture texture, int width, int height) { + configureTransform(width, height); + } + + @Override + public boolean onSurfaceTextureDestroyed(SurfaceTexture texture) { + return true; + } + + @Override + public void onSurfaceTextureUpdated(SurfaceTexture texture) { + } + + }; + + /** + * A {@link CameraCaptureSession } for camera preview. + */ + private CameraCaptureSession mCaptureSession; + + /** + * A reference to the opened {@link CameraDevice}. + */ + private CameraDevice mCameraDevice; + + /** + * The {@link android.util.Size} of camera preview. + */ + private Size mPreviewSize; + + /** + * {@link CameraDevice.StateCallback} is called when {@link CameraDevice} changes its state. + */ + private final CameraDevice.StateCallback mStateCallback = new CameraDevice.StateCallback() { + + @Override + public void onOpened(@NonNull CameraDevice cameraDevice) { + // This method is called when the camera is opened. We start camera preview here. + mCameraOpenCloseLock.release(); + mCameraDevice = cameraDevice; + createCameraPreviewSession(); + } + + @Override + public void onDisconnected(@NonNull CameraDevice cameraDevice) { + mCameraOpenCloseLock.release(); + cameraDevice.close(); + mCameraDevice = null; + } + + @Override + public void onError(@NonNull CameraDevice cameraDevice, int error) { + mCameraOpenCloseLock.release(); + cameraDevice.close(); + mCameraDevice = null; + Activity activity = getActivity(); + if (null != activity) { + activity.finish(); + } + } + + }; + + public static CameraFragment create(Bundle SavedInstanceState) { + System.out.println("==>CameraFragment"); + final CameraFragment fragment = new CameraFragment(); + runtime_var = SavedInstanceState.getChar("key"); + dlc_name_var = SavedInstanceState.getString("selected_dlc_name"); + System.out.println("CameraFragment class got data runtime_var="+runtime_var); + System.out.println("CameraFragment class got data selected_dlc_name="+dlc_name_var); + return fragment; + } + /** + * An additional thread for running tasks that shouldn't block the UI. + */ + private HandlerThread mBackgroundThread; + + /** + * A {@link Handler} for running tasks in the background. + */ + private Handler mBackgroundHandler; + + /** + * An {@link ImageReader} that handles still image capture. + */ + private ImageReader mImageReader; + + /** + * This is the output file for our picture. + */ + private File mFile; + + /** + * This a callback object for the {@link ImageReader}. "onImageAvailable" will be called when a + * still image is ready to be saved. + */ + private final ImageReader.OnImageAvailableListener mOnImageAvailableListener + = new ImageReader.OnImageAvailableListener() { + + @Override + public void onImageAvailable(ImageReader reader) { + mBackgroundHandler.post(new ImageSaver(reader.acquireNextImage(), mFile)); + } + + }; + + /** + * {@link CaptureRequest.Builder} for the camera preview + */ + private CaptureRequest.Builder mPreviewRequestBuilder; + + /** + * {@link CaptureRequest} generated by {@link #mPreviewRequestBuilder} + */ + private CaptureRequest mPreviewRequest; + + /** + * The current state of camera state for taking pictures. + * + * @see #mCaptureCallback + */ + private int mState = STATE_PREVIEW; + + /** + * A {@link Semaphore} to prevent the app from exiting before closing the camera. + */ + private Semaphore mCameraOpenCloseLock = new Semaphore(1); + + /** + * Whether the current camera device supports Flash or not. + */ + private boolean mFlashSupported; + + /** + * Orientation of the camera sensor + */ + private int mSensorOrientation; + /** + * A {@link CameraCaptureSession.CaptureCallback} that handles events related to JPEG capture. + */ + private CameraCaptureSession.CaptureCallback mCaptureCallback + = new CameraCaptureSession.CaptureCallback() { + + private void process(CaptureResult result) { + switch (mState) { + case STATE_PREVIEW: { + // We have nothing to do when the camera preview is working normally. + break; + } + case STATE_WAITING_LOCK: { + Integer afState = result.get(CaptureResult.CONTROL_AF_STATE); + if (afState == null) { + captureStillPicture(); + } else if (CaptureResult.CONTROL_AF_STATE_FOCUSED_LOCKED == afState || + CaptureResult.CONTROL_AF_STATE_NOT_FOCUSED_LOCKED == afState) { + // CONTROL_AE_STATE can be null on some devices + Integer aeState = result.get(CaptureResult.CONTROL_AE_STATE); + if (aeState == null || + aeState == CaptureResult.CONTROL_AE_STATE_CONVERGED) { + mState = STATE_PICTURE_TAKEN; + captureStillPicture(); + } else { + runPrecaptureSequence(); + } + } + break; + } + case STATE_WAITING_PRECAPTURE: { + // CONTROL_AE_STATE can be null on some devices + Integer aeState = result.get(CaptureResult.CONTROL_AE_STATE); + if (aeState == null || + aeState == CaptureResult.CONTROL_AE_STATE_PRECAPTURE || + aeState == CaptureRequest.CONTROL_AE_STATE_FLASH_REQUIRED) { + mState = STATE_WAITING_NON_PRECAPTURE; + } + break; + } + case STATE_WAITING_NON_PRECAPTURE: { + // CONTROL_AE_STATE can be null on some devices + Integer aeState = result.get(CaptureResult.CONTROL_AE_STATE); + if (aeState == null || aeState != CaptureResult.CONTROL_AE_STATE_PRECAPTURE) { + mState = STATE_PICTURE_TAKEN; + captureStillPicture(); + } + break; + } + } + } + + @Override + public void onCaptureProgressed(@NonNull CameraCaptureSession session, + @NonNull CaptureRequest request, + @NonNull CaptureResult partialResult) { + process(partialResult); + } + + @Override + public void onCaptureCompleted(@NonNull CameraCaptureSession session, + @NonNull CaptureRequest request, + @NonNull TotalCaptureResult result) { + process(result); + } + + }; + + + /** + * Shows a {@link Toast} on the UI thread. + * + * @param text The message to show + */ + private void showToast(final String text) { + final Activity activity = getActivity(); + if (activity != null) { + activity.runOnUiThread(new Runnable() { + @Override + public void run() { + Toast.makeText(activity, text, Toast.LENGTH_SHORT).show(); + } + }); + } + } + + /** + * Given {@code choices} of {@code Size}s supported by a camera, choose the smallest one that + * is at least as large as the respective texture view size, and that is at most as large as the + * respective max size, and whose aspect ratio matches with the specified value. If such size + * doesn't exist, choose the largest one that is at most as large as the respective max size, + * and whose aspect ratio matches with the specified value. + * + * @param choices The list of sizes that the camera supports for the intended output + * class + * @param textureViewWidth The width of the texture view relative to sensor coordinate + * @param textureViewHeight The height of the texture view relative to sensor coordinate + * @param maxWidth The maximum width that can be chosen + * @param maxHeight The maximum height that can be chosen + * @param aspectRatio The aspect ratio + * @return The optimal {@code Size}, or an arbitrary one if none were big enough + */ + private static Size chooseOptimalSize(Size[] choices, int textureViewWidth, + int textureViewHeight, int maxWidth, int maxHeight, Size aspectRatio) { + + // Collect the supported resolutions that are at least as big as the preview Surface + List bigEnough = new ArrayList<>(); + // Collect the supported resolutions that are smaller than the preview Surface + List notBigEnough = new ArrayList<>(); + int w = aspectRatio.getWidth(); + int h = aspectRatio.getHeight(); + for (Size option : choices) { + if (option.getWidth() <= maxWidth && option.getHeight() <= maxHeight && + option.getHeight() == option.getWidth() * h / w) { + if (option.getWidth() >= textureViewWidth && + option.getHeight() >= textureViewHeight) { + bigEnough.add(option); + } else { + notBigEnough.add(option); + } + } + } + + // Pick the smallest of those big enough. If there is no one big enough, pick the + // largest of those not big enough. + if (bigEnough.size() > 0) { + return Collections.min(bigEnough, new CompareSizesByArea()); + } else if (notBigEnough.size() > 0) { + return Collections.max(notBigEnough, new CompareSizesByArea()); + } else { + return choices[0]; + } + } + + public static CameraFragment newInstance() { + return new CameraFragment(); + } + + @Override + public View onCreateView(LayoutInflater inflater, ViewGroup container, + Bundle savedInstanceState) { + return inflater.inflate(R.layout.fragment_camera, container, false); + } + + @Override + public void onViewCreated(final View view, Bundle savedInstanceState) { + super.onViewCreated(view, savedInstanceState); + mTextureView = view.findViewById(R.id.surface); + mTextureView.setSurfaceTextureListener(mSurfaceTextureListener); + mFragmentRender = view.findViewById(R.id.fragmentRender); + + } + + @Override + public void onResume() { + super.onResume(); + + startBackgroundThread(); + + // When the screen is turned off and turned back on, the SurfaceTexture is already + // available, and "onSurfaceTextureAvailable" will not be called. In that case, we can open + // a camera and start preview from here (otherwise, we wait until the surface is ready in + // the SurfaceTextureListener). + if (mTextureView.isAvailable()) { + openCamera(mTextureView.getWidth(), mTextureView.getHeight()); + } else { + mTextureView.setSurfaceTextureListener(mSurfaceTextureListener); + } + + ensureNetCreated(); + + } + + @Override + public void onPause() { + closeCamera(); + stopBackgroundThread(); + super.onPause(); + } + + @Override + public void onDestroy() { + stopBackgroundThread(); + closeCamera(); + super.onDestroy(); + } + + private void requestCameraPermission() { + if (shouldShowRequestPermissionRationale(Manifest.permission.CAMERA)) { + new ConfirmationDialog().show(getChildFragmentManager(), FRAGMENT_DIALOG); + } else { + requestPermissions(new String[]{Manifest.permission.CAMERA}, REQUEST_CAMERA_PERMISSION); + } + } + + @Override + public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, + @NonNull int[] grantResults) { + if (requestCode == REQUEST_CAMERA_PERMISSION) { + if (grantResults.length != 1 || grantResults[0] != PackageManager.PERMISSION_GRANTED) { + ErrorDialog.newInstance(getString(R.string.request_permission)) + .show(getChildFragmentManager(), FRAGMENT_DIALOG); + } + } else { + super.onRequestPermissionsResult(requestCode, permissions, grantResults); + } + } + + + /** + * Sets up member variables related to camera. + * + * @param width The width of available size for camera preview + * @param height The height of available size for camera preview + */ + @SuppressWarnings("SuspiciousNameCombination") + private void setUpCameraOutputs(int width, int height) { + Activity activity = getActivity(); + CameraManager mCameraManager = (CameraManager) activity.getSystemService(Context.CAMERA_SERVICE); + try { + for (String cameraId : mCameraManager.getCameraIdList()) { + CameraCharacteristics characteristics + = mCameraManager.getCameraCharacteristics(cameraId); + + // We don't use a front facing camera in this sample. + Integer facing = characteristics.get(CameraCharacteristics.LENS_FACING); + if (facing != null && facing == CameraCharacteristics.LENS_FACING_FRONT) { + continue; + } + + StreamConfigurationMap map = characteristics.get( + CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP); + if (map == null) { + continue; + } + + // For still image captures, we use the largest available size. + Size largest = Collections.max( + Arrays.asList(map.getOutputSizes(ImageFormat.JPEG)), + new CompareSizesByArea()); + mImageReader = ImageReader.newInstance(largest.getWidth(), largest.getHeight(), + ImageFormat.JPEG, /*maxImages*/2); + mImageReader.setOnImageAvailableListener( + mOnImageAvailableListener, mBackgroundHandler); + + // Find out if we need to swap dimension to get the preview size relative to sensor + // coordinate. + int displayRotation = activity.getWindowManager().getDefaultDisplay().getRotation(); + //noinspection ConstantConditions + mSensorOrientation = characteristics.get(CameraCharacteristics.SENSOR_ORIENTATION); + boolean swappedDimensions = false; + switch (displayRotation) { + case Surface.ROTATION_0: + case Surface.ROTATION_180: + if (mSensorOrientation == 90 || mSensorOrientation == 270) { + swappedDimensions = true; + } + break; + case Surface.ROTATION_90: + case Surface.ROTATION_270: + if (mSensorOrientation == 0 || mSensorOrientation == 180) { + swappedDimensions = true; + } + break; + default: + System.out.println("Display rotation is invalid: " + displayRotation); + } + + Point displaySize = new Point(); + activity.getWindowManager().getDefaultDisplay().getSize(displaySize); + int rotatedPreviewWidth = width; + int rotatedPreviewHeight = height; + int maxPreviewWidth = displaySize.x; + int maxPreviewHeight = displaySize.y; + + if (swappedDimensions) { + rotatedPreviewWidth = height; + rotatedPreviewHeight = width; + maxPreviewWidth = displaySize.y; + maxPreviewHeight = displaySize.x; + } + + if (maxPreviewWidth > MAX_PREVIEW_WIDTH) { + maxPreviewWidth = MAX_PREVIEW_WIDTH; + } + + if (maxPreviewHeight > MAX_PREVIEW_HEIGHT) { + maxPreviewHeight = MAX_PREVIEW_HEIGHT; + } + + // Danger, W.R.! Attempting to use too large a preview size could exceed the camera + // bus' bandwidth limitation, resulting in gorgeous previews but the storage of + // garbage capture data. + mPreviewSize = chooseOptimalSize(map.getOutputSizes(SurfaceTexture.class), + rotatedPreviewWidth, rotatedPreviewHeight, maxPreviewWidth, + maxPreviewHeight, largest); + + // Check if the flash is supported. + Boolean available = characteristics.get(CameraCharacteristics.FLASH_INFO_AVAILABLE); + mFlashSupported = available == null ? false : available; + + mCameraId = cameraId; + return; + } + } catch (CameraAccessException e) { + e.printStackTrace(); + } catch (NullPointerException e) { + // Currently an NPE is thrown when the Camera2API is used but not supported on the + // device this code runs. + ErrorDialog.newInstance(getString(R.string.camera_error)) + .show(getChildFragmentManager(), FRAGMENT_DIALOG); + } + } + + /** + * Opens the camera specified by {@link CameraFragment#mCameraId}. + */ + private void openCamera(int width, int height) { + if (ContextCompat.checkSelfPermission(getActivity(), Manifest.permission.CAMERA) + != PackageManager.PERMISSION_GRANTED) { + requestCameraPermission(); + return; + } + setUpCameraOutputs(width, height); + configureTransform(width, height); + Activity activity = getActivity(); + CameraManager manager = (CameraManager) activity.getSystemService(Context.CAMERA_SERVICE); + try { + if (!mCameraOpenCloseLock.tryAcquire(2500, TimeUnit.MILLISECONDS)) { + throw new RuntimeException("Time out waiting to lock camera opening."); + } + manager.openCamera(mCameraId, mStateCallback, mBackgroundHandler); + } catch (CameraAccessException e) { + e.printStackTrace(); + } catch (InterruptedException e) { + throw new RuntimeException("Interrupted while trying to lock camera opening.", e); + } + } + + /** + * Closes the current {@link CameraDevice}. + */ + private void closeCamera() { + try { + mCameraOpenCloseLock.acquire(); + if (null != mCaptureSession) { + mCaptureSession.close(); + mCaptureSession = null; + } + if (null != mCameraDevice) { + mCameraDevice.close(); + mCameraDevice = null; + } + if (null != mImageReader) { + mImageReader.close(); + mImageReader = null; + } + } catch (InterruptedException e) { + throw new RuntimeException("Interrupted while trying to lock camera closing.", e); + } finally { + mCameraOpenCloseLock.release(); + } + } + + /** + * Starts a background thread and its {@link Handler}. + */ + private void startBackgroundThread() { + mBackgroundThread = new HandlerThread("CameraBackground"); + mBackgroundThread.start(); + mBackgroundHandler = new Handler(mBackgroundThread.getLooper()); + } + + /** + * Stops the background thread and its {@link Handler}. + */ + private void stopBackgroundThread() { + + if (mBackgroundThread!=null) { + + mBackgroundThread.quitSafely(); + + mBackgroundThread = null; + mBackgroundHandler = null; + + } + } + + /** + * Creates a new {@link CameraCaptureSession} for camera preview. + */ + private void createCameraPreviewSession() { + try { + SurfaceTexture texture = mTextureView.getSurfaceTexture(); + assert texture != null; + + // We configure the size of default buffer to be the size of camera preview we want. + texture.setDefaultBufferSize(mPreviewSize.getWidth(), mPreviewSize.getHeight()); + + // This is the output Surface we need to start preview. + Surface surface = new Surface(texture); + + // We set up a CaptureRequest.Builder with the output Surface. + mPreviewRequestBuilder + = mCameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW); + mPreviewRequestBuilder.addTarget(surface); + + //Shubham + try { + mCameraDevice.createCaptureSession(Arrays.asList(surface), new CameraCapture(), + null); + } catch (CameraAccessException e) { + e.printStackTrace(); + } + } catch (CameraAccessException e) { + e.printStackTrace(); + } + } + + /** + * Configures the necessary {@link android.graphics.Matrix} transformation to `mTextureView`. + * This method should be called after the camera preview size is determined in + * setUpCameraOutputs and also the size of `mTextureView` is fixed. + * + * @param viewWidth The width of `mTextureView` + * @param viewHeight The height of `mTextureView` + */ + private void configureTransform(int viewWidth, int viewHeight) { + Activity activity = getActivity(); + if (null == mTextureView || null == mPreviewSize || null == activity) { + return; + } + int rotation = activity.getWindowManager().getDefaultDisplay().getRotation(); + Matrix matrix = new Matrix(); + RectF viewRect = new RectF(0, 0, viewWidth, viewHeight); + RectF bufferRect = new RectF(0, 0, mPreviewSize.getHeight(), mPreviewSize.getWidth()); + float centerX = viewRect.centerX(); + float centerY = viewRect.centerY(); + if (Surface.ROTATION_90 == rotation || Surface.ROTATION_270 == rotation) { + bufferRect.offset(centerX - bufferRect.centerX(), centerY - bufferRect.centerY()); + matrix.setRectToRect(viewRect, bufferRect, Matrix.ScaleToFit.FILL); + float scale = Math.max( + (float) viewHeight / mPreviewSize.getHeight(), + (float) viewWidth / mPreviewSize.getWidth()); + matrix.postScale(scale, scale, centerX, centerY); + matrix.postRotate(90 * (rotation - 2), centerX, centerY); + } else if (Surface.ROTATION_180 == rotation) { + matrix.postRotate(180, centerX, centerY); + } + mTextureView.setTransform(matrix); + } + + /** + * Lock the focus as the first step for a still image capture. + */ + private void lockFocus() { + try { + // This is how to tell the camera to lock focus. + mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AF_TRIGGER, + CameraMetadata.CONTROL_AF_TRIGGER_START); + // Tell #mCaptureCallback to wait for the lock. + mState = STATE_WAITING_LOCK; + mCaptureSession.capture(mPreviewRequestBuilder.build(), mCaptureCallback, + mBackgroundHandler); + } catch (CameraAccessException e) { + e.printStackTrace(); + } + } + + /** + * Run the precapture sequence for capturing a still image. This method should be called when + * we get a response in {@link #mCaptureCallback} from {@link #lockFocus()}. + */ + private void runPrecaptureSequence() { + try { + // This is how to tell the camera to trigger. + mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AE_PRECAPTURE_TRIGGER, + CaptureRequest.CONTROL_AE_PRECAPTURE_TRIGGER_START); + // Tell #mCaptureCallback to wait for the precapture sequence to be set. + mState = STATE_WAITING_PRECAPTURE; + mCaptureSession.capture(mPreviewRequestBuilder.build(), mCaptureCallback, + mBackgroundHandler); + } catch (CameraAccessException e) { + e.printStackTrace(); + } + } + + /** + * Capture a still picture. This method should be called when we get a response in + * {@link #mCaptureCallback} from both {@link #lockFocus()}. + */ + private void captureStillPicture() { + try { + final Activity activity = getActivity(); + if (null == activity || null == mCameraDevice) { + return; + } + // This is the CaptureRequest.Builder that we use to take a picture. + final CaptureRequest.Builder captureBuilder = + mCameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_STILL_CAPTURE); + captureBuilder.addTarget(mImageReader.getSurface()); + + // Use the same AE and AF modes as the preview. + captureBuilder.set(CaptureRequest.CONTROL_AF_MODE, + CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE); + setAutoFlash(captureBuilder); + + // Orientation + int rotation = activity.getWindowManager().getDefaultDisplay().getRotation(); + captureBuilder.set(CaptureRequest.JPEG_ORIENTATION, getOrientation(rotation)); + + CameraCaptureSession.CaptureCallback CaptureCallback + = new CameraCaptureSession.CaptureCallback() { + + @Override + public void onCaptureCompleted(@NonNull CameraCaptureSession session, + @NonNull CaptureRequest request, + @NonNull TotalCaptureResult result) { + showToast("Saved: " + mFile); + unlockFocus(); + } + }; + + mCaptureSession.stopRepeating(); + mCaptureSession.abortCaptures(); + mCaptureSession.capture(captureBuilder.build(), CaptureCallback, null); + } catch (CameraAccessException e) { + e.printStackTrace(); + } + } + + /** + * Retrieves the JPEG orientation from the specified screen rotation. + * + * @param rotation The screen rotation. + * @return The JPEG orientation (one of 0, 90, 270, and 360) + */ + private int getOrientation(int rotation) { + // Sensor orientation is 90 for most devices, or 270 for some devices (eg. Nexus 5X) + // We have to take that into account and rotate JPEG properly. + // For devices with orientation of 90, we simply return our mapping from ORIENTATIONS. + // For devices with orientation of 270, we need to rotate the JPEG 180 degrees. + return (ORIENTATIONS.get(rotation) + mSensorOrientation + 270) % 360; + } + + /** + * Unlock the focus. This method should be called when still image capture sequence is + * finished. + */ + private void unlockFocus() { + try { + // Reset the auto-focus trigger + mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AF_TRIGGER, + CameraMetadata.CONTROL_AF_TRIGGER_CANCEL); + setAutoFlash(mPreviewRequestBuilder); + mCaptureSession.capture(mPreviewRequestBuilder.build(), mCaptureCallback, + mBackgroundHandler); + // After this, the camera will go back to the normal state of preview. + mState = STATE_PREVIEW; + mCaptureSession.setRepeatingRequest(mPreviewRequest, mCaptureCallback, + mBackgroundHandler); + } catch (CameraAccessException e) { + e.printStackTrace(); + } + } + + private void setAutoFlash(CaptureRequest.Builder requestBuilder) { + if (mFlashSupported) { + requestBuilder.set(CaptureRequest.CONTROL_AE_MODE, + CaptureRequest.CONTROL_AE_MODE_ON_AUTO_FLASH); + } + } + + /** + * Saves a JPEG {@link Image} into the specified {@link File}. + */ + private static class ImageSaver implements Runnable { + + /** + * The JPEG image + */ + private final Image mImage; + /** + * The file we save the image into. + */ + private final File mFile; + + ImageSaver(Image image, File file) { + mImage = image; + mFile = file; + } + + @Override + public void run() { + ByteBuffer buffer = mImage.getPlanes()[0].getBuffer(); + byte[] bytes = new byte[buffer.remaining()]; + buffer.get(bytes); + FileOutputStream output = null; + try { + output = new FileOutputStream(mFile); + output.write(bytes); + } catch (IOException e) { + e.printStackTrace(); + } finally { + mImage.close(); + if (null != output) { + try { + output.close(); + } catch (IOException e) { + e.printStackTrace(); + } + } + } + } + + } + + /** + * Compares two {@code Size}s based on their areas. + */ + static class CompareSizesByArea implements Comparator { + + @Override + public int compare(Size lhs, Size rhs) { + // We cast here to ensure the multiplications won't overflow + return Long.signum((long) lhs.getWidth() * lhs.getHeight() - + (long) rhs.getWidth() * rhs.getHeight()); + } + + } + + /** + * Shows an error message dialog. + */ + public static class ErrorDialog extends DialogFragment { + + private static final String ARG_MESSAGE = "message"; + + public static ErrorDialog newInstance(String message) { + ErrorDialog dialog = new ErrorDialog(); + Bundle args = new Bundle(); + args.putString(ARG_MESSAGE, message); + dialog.setArguments(args); + return dialog; + } + + @NonNull + @Override + public Dialog onCreateDialog(Bundle savedInstanceState) { + final Activity activity = getActivity(); + return new AlertDialog.Builder(activity) + .setMessage(getArguments().getString(ARG_MESSAGE)) + .setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialogInterface, int i) { + activity.finish(); + } + }) + .create(); + } + + } + + /** + * Shows OK/Cancel confirmation dialog about camera permission. + */ + public static class ConfirmationDialog extends DialogFragment { + + @NonNull + @Override + public Dialog onCreateDialog(Bundle savedInstanceState) { + final Fragment parent = getParentFragment(); + return new AlertDialog.Builder(getActivity()) + .setMessage(R.string.request_permission) + .setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialog, int which) { + parent.requestPermissions(new String[]{Manifest.permission.CAMERA}, + REQUEST_CAMERA_PERMISSION); + } + }) + .setNegativeButton(android.R.string.cancel, + new DialogInterface.OnClickListener() { + @Override + public void onClick(DialogInterface dialog, int which) { + Activity activity = parent.getActivity(); + if (activity != null) { + activity.finish(); + } + } + }) + .create(); + } + } + + public class CameraCapture extends android.hardware.camera2.CameraCaptureSession.StateCallback { + @Override + public void onConfigured(@NonNull android.hardware.camera2.CameraCaptureSession + cameraCaptureSession) { + mCaptureSession = cameraCaptureSession; + + try { + // Auto focus should be continuous for camera preview. + mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AF_MODE, + CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE); + // Flash is automatically enabled when necessary. + setAutoFlash(mPreviewRequestBuilder); + // Finally, we start displaying the camera preview. + mPreviewRequest = mPreviewRequestBuilder.build(); + cameraCaptureSession.setRepeatingRequest(mPreviewRequestBuilder.build(), new CameraSession(), + mBackgroundHandler); + } catch (Exception e) { + e.printStackTrace(); + } + } + + @Override + public void onConfigureFailed(@NonNull android.hardware.camera2.CameraCaptureSession + cameraCaptureSession) { + } + + } + + private class CameraSession extends android.hardware.camera2.CameraCaptureSession.CaptureCallback { + + @Override + public void onCaptureCompleted(@NonNull CameraCaptureSession session, @NonNull + CaptureRequest request, @NonNull TotalCaptureResult result) { + + super.onCaptureCompleted(session, request, result); +// int rotation = getActivity().getWindowManager().getDefaultDisplay().getRotation(); + frame_count+=1; + + try { + if (frame_count == 0) { + tic = System.currentTimeMillis(); + } else { + tic2 = System.currentTimeMillis(); + fps = (int) (1000 / (tic2 - tic)); + tic = System.currentTimeMillis(); + } + } + catch (Exception e) { + e.printStackTrace(); + } + + System.out.println("mNetworkLoaded: "+mNetworkLoaded +" runtime_var: "+runtime_var+" dlc_name_var: "+dlc_name_var); + + if (mNetworkLoaded == true ) { + + Bitmap mBitmap = mTextureView.getBitmap(mTextureView.getWidth(),mTextureView.getHeight()); + + ArrayList BBlist = new ArrayList<>(); + System.out.println("calling inference"); + + int infer_result = mSnpeHelper.snpeInference(mBitmap, fps, BBlist); + + //sanjeev - added temp dialogue info box to graceful exit if model not loaded properly + if (infer_result == -1) + { + if (dialog_model_error==null) { + + dialog_model_error = new ProgressDialog(getActivity()); + + getActivity().runOnUiThread(new Runnable() { + + @Override + public void run() { + + if (dialog_model_error!=null) { + try { + dialog_model_error.setMessage(getString(R.string.model_loading_error)); + dialog_model_error.show(); + }catch (Exception e){ + e.printStackTrace(); + Toast.makeText(getContext(),getString(R.string.model_loading_error), Toast.LENGTH_SHORT).show(); + } + } + } + }); + + } + + } + + mFragmentRender.setCoordsList(BBlist); + } + } + + @Override + public void onCaptureFailed(@NonNull CameraCaptureSession session, @NonNull + CaptureRequest request, @NonNull CaptureFailure failure) { + super.onCaptureFailed(session, request, failure); + } + + @Override + public void onCaptureProgressed(@NonNull CameraCaptureSession session, @NonNull + CaptureRequest request, @NonNull CaptureResult partialResult) { + super.onCaptureProgressed(session, request, partialResult); + } + + @Override + public void onCaptureStarted(@NonNull CameraCaptureSession session, @NonNull + CaptureRequest request, long timestamp, long frameNumber) { + super.onCaptureStarted(session, request, timestamp, frameNumber); + + } + + } + + /** + * Method to ensure if neural network is loaded + * + * @return + */ + private boolean ensureNetCreated() { + if (mSnpeHelper == null) { + // load the neural network for object detection with SNPE + mSnpeHelper = new SNPEHelper(getActivity().getApplication()); + + // Show UI dialog till the model is not loaded yet. + ProgressDialog dialog=new ProgressDialog(getActivity()); + dialog.setMessage("Loading Model.."); + dialog.show(); + + //TODO for time being disabling + new Thread() { + public void run() { + mNetworkLoaded = mSnpeHelper.loadingMODELS(runtime_var, dlc_name_var); + //dismiss dialog when the model is loaded + dialog.dismiss(); + } + }.start() ; + + } + return mNetworkLoaded; + } + +} diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/FragmentRender.java b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/FragmentRender.java new file mode 100644 index 00000000..a353b005 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/FragmentRender.java @@ -0,0 +1,104 @@ +// -*- mode: java -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +package com.qcom.aistack_objdetect; + +import android.content.Context; +import android.graphics.Canvas; +import android.graphics.Color; +import android.graphics.Paint; +import android.graphics.Typeface; +import android.support.annotation.Nullable; +import android.util.AttributeSet; +import android.view.View; + + +import java.util.ArrayList; +import java.util.concurrent.locks.ReentrantLock; + +/** + * FragmentRender class is utility for making boxes on camera frames. + * FragmentRender has utility in fragment_camera.xml and CameraFragment Class + */ +public class FragmentRender extends View { + + private ReentrantLock mLock = new ReentrantLock(); + private ArrayList boxlist = new ArrayList<>(); + + private Paint mTextColor= new Paint(); + private Paint mBorderColor= new Paint(); + + public FragmentRender(Context context, @Nullable AttributeSet attrs) { + super(context, attrs); + init(); + } + + + public void setCoordsList(ArrayList t_boxlist) { + mLock.lock(); + postInvalidate(); + + if (boxlist==null) + { + mLock.unlock(); + return; + } + boxlist.clear(); + for(int j=0;j parent) { + System.out.println("Nothing"); + } + }); + + } + + public void startManinCameraActivity(View v) + { + Intent i =new Intent(this, MainActivity.class); + + Bundle args = new Bundle(); + args.putChar("key", runtime_var); + args.putCharSequence("selected_dlc_name", dlc_name); + i.putExtras(args); + + startActivity(i); + } + +} diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/MainActivity.java b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/MainActivity.java new file mode 100644 index 00000000..ee572868 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/MainActivity.java @@ -0,0 +1,138 @@ +// -*- mode: java -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +package com.qcom.aistack_objdetect; + +import android.Manifest; +import android.content.Intent; +import android.content.pm.PackageManager; +import android.os.Build; +import android.support.v4.app.FragmentTransaction; +import android.support.v7.app.AppCompatActivity; +import android.os.Bundle; +import android.view.WindowManager; +import android.widget.TextView; +import org.opencv.android.OpenCVLoader; +import java.util.HashMap; +import java.util.Map; + +/** + * MainActivity class displays the info of Selected Model,Runtime, classes supported by the model on UI through main_activity.xml + * It also passes the selected model and runtime info to the CameraFragment for making inference using selected Model and Runtime. + */ +public class MainActivity extends AppCompatActivity { + + static final Map model_name=new HashMap(); + static Map runtime_name=new HashMap(); + static Map class_count=new HashMap(); + + static final String[] modeloptions = {"YOLONAS", "SSDMobilenetV2", "YoloX"}; + static final String[] modeldlcname = {"Quant_yoloNas_s_320.dlc", "ssd_mobilenetV2_without_ABP-NMS_Q.dlc", "yolox_x_212_Q.dlc"}; + + static final char[] runtimeoptions = {'C', 'G', 'D'}; + static final String[] runtimename = {"CPU", "GPU", "DSP"}; + + static final String[] classcount = {"80", "21", "80"}; + + static { + //System.loadLibrary("objectdetectionYoloNas"); + + for (int i=0;i= Build.VERSION_CODES.M) { + passToFragment = MainActivity.this.checkSelfPermission(Manifest.permission.CAMERA) == PackageManager.PERMISSION_GRANTED; + } + else{ + passToFragment = true; + } + if (passToFragment) { + FragmentTransaction transaction = getSupportFragmentManager().beginTransaction(); + Bundle args = new Bundle(); + args.putChar("key", runtime_value); + args.putCharSequence("selected_dlc_name", selected_dlc_name); + transaction.add(R.id.main_content, CameraFragment.create(args)); + transaction.commit(); + } else { + cameraPermission(); + } + } + + @Override + protected void onResume() { + super.onResume(); + overToCamera(runtime_var, dlc_name); + } + + @Override + protected void onStop() { + super.onStop(); + } +} \ No newline at end of file diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/RectangleBox.java b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/RectangleBox.java new file mode 100644 index 00000000..c0dcb7d0 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/RectangleBox.java @@ -0,0 +1,36 @@ +// -*- mode: java -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +package com.qcom.aistack_objdetect; + +import java.util.ArrayList; +/** + * RectangleBox class defines the property associated with each box like coordinates + * labels, confidence etc. + * Can also create copy of boxes. + */ +public class RectangleBox { + + public float top; + public float bottom; + public float left; + public float right; + + public int fps; + public String processing_time; + public String label; + public static ArrayList createBoxes(int num) { + final ArrayList boxes; + boxes = new ArrayList<>(); + for (int i = 0; i < num; ++i) { + boxes.add(new RectangleBox()); + } + return boxes; + } +} diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/SNPEHelper.java b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/SNPEHelper.java new file mode 100644 index 00000000..fa16fa86 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/java/com/qcom/aistack_objdetect/SNPEHelper.java @@ -0,0 +1,104 @@ +// -*- mode: java -*- +// ============================================================================= +// @@-COPYRIGHT-START-@@ +// +// Copyright (c) 2023 of Qualcomm Innovation Center, Inc. All rights reserved. +// SPDX-License-Identifier: BSD-3-Clause +// +// @@-COPYRIGHT-END-@@ +// ============================================================================= +package com.qcom.aistack_objdetect; + +import android.app.Application; +import android.content.res.AssetManager; +import android.graphics.Bitmap; +import android.util.Log; +import org.opencv.android.Utils; +import org.opencv.core.Mat; +import java.util.ArrayList; + +public class SNPEHelper { + private final Application mApplication; + private AssetManager assetManager; + + // Constructor + public SNPEHelper(Application application) { + mApplication = application; + } + + //Native functions + public native String queryRuntimes(String a); + public native String initSNPE(AssetManager assetManager, char a, String dlc_name); + public native int inferSNPE(long inputmataddress, int width,int height, float[][]boxcoords, String[] classname); + + + /** + * This method loads ML models on selected runtime + */ + public boolean loadingMODELS(char runtime_var, String dlc_name) { + + assetManager = mApplication.getAssets(); + String nativeDirPath = mApplication.getApplicationInfo().nativeLibraryDir; + String res_query = queryRuntimes(nativeDirPath); + System.out.println(res_query); + String tt = initSNPE(assetManager, runtime_var, dlc_name); + System.out.println("RESULT:"+tt); + + int success_count = tt.split("success", -1).length -1; + + if(success_count==1) + { + System.out.println("Model built successfully"); + return true; + } + + return false; + } + + /* + This method makes inference on bitmap. + */ + public int snpeInference(Bitmap modelInputBitmap, int fps, ArrayList BBlist) { + + int result=0; + + try{ + + Mat inputMat = new Mat(); + Utils.bitmapToMat(modelInputBitmap, inputMat); + + float[][] boxCoords = new float[100][5]; //Stores box coords for all person, MAXLIMIT is 100, last coords i.e. boxCoords[k][4] stores confidence value <-- IMP + String[] boxnames = new String[100]; + + //System.out.println("call inferSNPE input_Width="+modelInputBitmap.getWidth()+" input_Height="+modelInputBitmap.getHeight()); + + int numhuman = inferSNPE(inputMat.getNativeObjAddr(), modelInputBitmap.getWidth(), modelInputBitmap.getHeight(), boxCoords,boxnames); + + if (numhuman == -1) + { + Log.e("SNPEHelper", "Error loading model properly. Return error.."); + return -1; + } + + for(int k=0;k + + + + + + + + + + diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/res/drawable/ic_launcher_background.xml b/ai-solutions/android/03-ObjectDetection/app/src/main/res/drawable/ic_launcher_background.xml new file mode 100644 index 00000000..50ae786e --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/res/drawable/ic_launcher_background.xml @@ -0,0 +1,74 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/ai-solutions/android/03-ObjectDetection/app/src/main/res/layout/activity_home_screen.xml b/ai-solutions/android/03-ObjectDetection/app/src/main/res/layout/activity_home_screen.xml new file mode 100644 index 00000000..14e99e71 --- /dev/null +++ b/ai-solutions/android/03-ObjectDetection/app/src/main/res/layout/activity_home_screen.xml @@ -0,0 +1,78 @@ + + + + + + + + + + + + + + + + + + + +