From f0f739c516cec26d487e23a24997f7c4a77c78e6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?D=C3=A1vid=20H=C3=A1zi?= Date: Thu, 7 Mar 2024 22:01:59 +0100 Subject: [PATCH] applications: Add object-detection example Object detection example is an ML vision example on the Cortex-M systems. Currently running only on Corstone-315 to demonstrate the usage of ISP, NPU, HDLCD together. Signed-off-by: Gabor Abonyi --- README.md | 10 + applications/object_detection/CMakeLists.txt | 177 + applications/object_detection/blink_task.c | 93 + applications/object_detection/blink_task.h | 32 + .../object_detection/configs/CMakeLists.txt | 10 + .../configs/app_config/CMakeLists.txt | 10 + .../configs/app_config/app_config.h | 59 + .../configs/arm_2d_config/CMakeLists.txt | 10 + .../configs/arm_2d_config/arm_2d_cfg.h | 225 ++ .../configs/aws_configs/CMakeLists.txt | 38 + .../aws_configs/aws_clientcredential.h | 51 + .../configs/aws_configs/aws_mqtt_config.h | 69 + .../aws_configs/aws_secure_sockets_config.h | 57 + .../configs/aws_configs/core_mqtt_config.h | 78 + .../configs/aws_configs/core_pkcs11_config.h | 160 + .../configs/aws_configs/demo_config.h | 82 + .../iot_default_root_certificates.h | 197 + .../aws_configs/mqtt_agent_demo_config.h | 129 + .../mqtt_demo_mutual_auth_config.h | 112 + .../configs/aws_configs/ota_config.h | 206 ++ .../configs/aws_configs/ota_demo_config.h | 103 + .../configs/freertos_config/CMakeLists.txt | 20 + .../configs/freertos_config/FreeRTOSConfig.h | 237 ++ .../freertos_config/FreeRTOSIPConfig.h | 316 ++ .../configs/isp_config/CMakeLists.txt | 15 + .../isp_config/acamera_firmware_config.h | 238 ++ .../isp_config/acamera_interface_config.h | 49 + .../configs/mbedtls_config/CMakeLists.txt | 18 + .../mbedtls_config/aws_mbedtls_config.h | 3274 +++++++++++++++++ .../configs/mbedtls_config/threading_alt.h | 61 + .../configs/tfm_config/project_config.h | 86 + .../object_detection/isp/CMakeLists.txt | 23 + .../object_detection/isp/acamera_callbacks.c | 213 ++ applications/object_detection/isp/isp_app.c | 318 ++ .../object_detection/isp/isp_config.c | 459 +++ .../object_detection/isp/isp_config.h | 93 + applications/object_detection/main.c | 324 ++ applications/object_detection/ml_interface.cc | 512 +++ applications/object_detection/ml_interface.h | 71 + .../object_detection/resources/test.frm | Bin 0 -> 663616 bytes .../resources/use_case_resources.json | 14 + .../object_detection/tests/fail_output.log | 0 .../object_detection/tests/pass_output.log | 10 + docs/object_detection.md | 488 +++ tools/ci/license/license.schema | 2 +- tools/scripts/build.sh | 13 +- tools/scripts/run.sh | 23 +- tools/tests/conftest.py | 124 +- 48 files changed, 8870 insertions(+), 39 deletions(-) create mode 100644 applications/object_detection/CMakeLists.txt create mode 100644 applications/object_detection/blink_task.c create mode 100644 applications/object_detection/blink_task.h create mode 100644 applications/object_detection/configs/CMakeLists.txt create mode 100644 applications/object_detection/configs/app_config/CMakeLists.txt create mode 100644 applications/object_detection/configs/app_config/app_config.h create mode 100644 applications/object_detection/configs/arm_2d_config/CMakeLists.txt create mode 100644 applications/object_detection/configs/arm_2d_config/arm_2d_cfg.h create mode 100644 applications/object_detection/configs/aws_configs/CMakeLists.txt create mode 100644 applications/object_detection/configs/aws_configs/aws_clientcredential.h create mode 100644 applications/object_detection/configs/aws_configs/aws_mqtt_config.h create mode 100644 applications/object_detection/configs/aws_configs/aws_secure_sockets_config.h create mode 100644 applications/object_detection/configs/aws_configs/core_mqtt_config.h create mode 100644 applications/object_detection/configs/aws_configs/core_pkcs11_config.h create mode 100644 applications/object_detection/configs/aws_configs/demo_config.h create mode 100644 applications/object_detection/configs/aws_configs/iot_default_root_certificates.h create mode 100644 applications/object_detection/configs/aws_configs/mqtt_agent_demo_config.h create mode 100644 applications/object_detection/configs/aws_configs/mqtt_demo_mutual_auth_config.h create mode 100644 applications/object_detection/configs/aws_configs/ota_config.h create mode 100644 applications/object_detection/configs/aws_configs/ota_demo_config.h create mode 100644 applications/object_detection/configs/freertos_config/CMakeLists.txt create mode 100644 applications/object_detection/configs/freertos_config/FreeRTOSConfig.h create mode 100644 applications/object_detection/configs/freertos_config/FreeRTOSIPConfig.h create mode 100644 applications/object_detection/configs/isp_config/CMakeLists.txt create mode 100644 applications/object_detection/configs/isp_config/acamera_firmware_config.h create mode 100644 applications/object_detection/configs/isp_config/acamera_interface_config.h create mode 100644 applications/object_detection/configs/mbedtls_config/CMakeLists.txt create mode 100644 applications/object_detection/configs/mbedtls_config/aws_mbedtls_config.h create mode 100644 applications/object_detection/configs/mbedtls_config/threading_alt.h create mode 100644 applications/object_detection/configs/tfm_config/project_config.h create mode 100644 applications/object_detection/isp/CMakeLists.txt create mode 100644 applications/object_detection/isp/acamera_callbacks.c create mode 100644 applications/object_detection/isp/isp_app.c create mode 100644 applications/object_detection/isp/isp_config.c create mode 100644 applications/object_detection/isp/isp_config.h create mode 100644 applications/object_detection/main.c create mode 100755 applications/object_detection/ml_interface.cc create mode 100644 applications/object_detection/ml_interface.h create mode 100644 applications/object_detection/resources/test.frm create mode 100644 applications/object_detection/resources/use_case_resources.json create mode 100644 applications/object_detection/tests/fail_output.log create mode 100644 applications/object_detection/tests/pass_output.log create mode 100644 docs/object_detection.md diff --git a/README.md b/README.md index ce6b8c10..149e3e3c 100644 --- a/README.md +++ b/README.md @@ -24,6 +24,9 @@ CPU and Ethos NPU, alongside a range of other components in a scalable and flexible reference package. This enables designers to build secure, AI-capable SoCs faster. +* [Corstone-315](https://developer.arm.com/Processors/Corstone-315) + * Arm Cortex-M85 CPU, Ethos-U65 NPU and Mali-C55 ISP + * Supported versions: 11.24.13 and above * [Corstone-310](https://developer.arm.com/Processors/Corstone-310) * Arm Cortex-M85 CPU and Ethos-U55 NPU * Supported versions: 11.24.13 and above @@ -65,6 +68,8 @@ This reference integration contains following two examples: * Demonstrates detecting keywords from an audio source using Machine Learning. * [Speech Recognition](docs/speech_recognition.md) * Demonstrates detecting sentences from an audio source using Machine Learning. +* [Object Detection](docs/object_detection.md) + * Demonstrates detection of faces from image source using Machine Learning. The Keyword-Detection and Speech-Recognition applications demonstrate [secure connectivity](#secure-tls-connection) to AWS IoT core using [Mbed TLS](#mbed-tls), [PKCS#11 PSA Shim](#pkcs11-psa-shim) and @@ -152,6 +157,11 @@ This implementation maps the AWS OTA PAL APIs to the PSA Firmware Update and PSA Cryptography APIs. The writing, verification and activation of the update image are protected by the PSA secure services. +### Mali-C55 Versatile Image Signal Processor for Computer Vision and Smart Display Systems + +Implementation of [Arm® Mali™-C55 bare-metal driver](https://gitlab.arm.com/iot/m-class/drivers/isp_mali-c55), +that demonstrates the usage of the Mali-C55 ISP with the Corstone M85 processor. + ## Contributing See [CONTRIBUTING](CONTRIBUTING.md) for more information. diff --git a/applications/object_detection/CMakeLists.txt b/applications/object_detection/CMakeLists.txt new file mode 100644 index 00000000..9e64e9d9 --- /dev/null +++ b/applications/object_detection/CMakeLists.txt @@ -0,0 +1,177 @@ +# Copyright 2023-2024 Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +cmake_minimum_required(VERSION 3.21.0 FATAL_ERROR) + +set(ML_INFERENCE_ENGINE "ETHOS" CACHE STRING "Machine Learning inference engine (ETHOS)") + +set(AUDIO_SOURCE "ROM" CACHE STRING "Source of audio data (ROM | VSI)") + +# From: ota-for-aws-iot-embedded-sdk/source/include/ota_appversion32.h +# struct version +# { +# uint8_t major; /*!< @brief Major version number of the firmware (X in firmware version X.Y.Z). */ +# uint8_t minor; /*!< @brief Minor version number of the firmware (Y in firmware version X.Y.Z). */ +# +# uint16_t build; /*!< @brief Build of the firmware (Z in firmware version X.Y.Z). */ +# } x; /*!< @brief Version number of the firmware. */ + +# AWS OTA client does not use the SemVer PATCH version. +# Because of this, if only PATCH version is changed then the OTA will be rejected +# due to same firmware version. +# We will therefore change the build version from TF-M. +set(MCUBOOT_IMAGE_VERSION_NS "0.0.1+10") +set(MCUBOOT_IMAGE_VERSION_NS_UPDATE "0.0.1+20") + +if (${ML_INFERENCE_ENGINE} STREQUAL "ETHOS") + set(ETHOS_U_NPU_ENABLED ON) + set(ETHOS_U_NPU_TIMING_ADAPTER_ENABLED OFF) +else() + set(ETHOS_U_NPU_ENABLED OFF) +endif() +set(ML_USE_CASE "object_detection") +set(ML_MODEL "GenerateObjectDetectionModel") +set(ML_USE_CASE_RESOURCES_FILE "${CMAKE_CURRENT_LIST_DIR}/resources/use_case_resources.json") +set(TFM_PLATFORM_UPGRADE_STRATEGY "SWAP_USING_SCRATCH") +set(TFM_PLATFORM_CONFIRM_IMAGE ON) + +# Trusted Firmware-M setup +set(TFM_CMAKE_APP_ARGS + -DPROJECT_CONFIG_HEADER_FILE=${IOT_REFERENCE_ARM_CORSTONE3XX_SOURCE_DIR}/applications/object_detection/configs/tfm_config/project_config.h + -DMCUBOOT_CONFIRM_IMAGE=${TFM_PLATFORM_CONFIRM_IMAGE} + -DMCUBOOT_UPGRADE_STRATEGY=${TFM_PLATFORM_UPGRADE_STRATEGY} + -DMCUBOOT_IMAGE_VERSION_NS=${MCUBOOT_IMAGE_VERSION_NS} + -DCONFIG_TFM_HALT_ON_CORE_PANIC=ON + -DMCUBOOT_DATA_SHARING=ON + -DPLATFORM_HAS_FIRMWARE_UPDATE_SUPPORT=ON + -DTFM_PARTITION_FIRMWARE_UPDATE=ON + -DTFM_PARTITION_LOG_LEVEL=TFM_PARTITION_LOG_LEVEL_INFO +) + +project(object-detection LANGUAGES C CXX) + +# Set global optimization level to reduce code size while keeping the debug experience. +if(${CMAKE_C_COMPILER_ID} STREQUAL "GNU") + add_compile_options(-Og) +elseif(${CMAKE_C_COMPILER_ID} STREQUAL "ARMClang") + add_compile_options(-O1) +endif() + + +add_subdirectory(${IOT_REFERENCE_ARM_CORSTONE3XX_SOURCE_DIR} ${CMAKE_BINARY_DIR}/iot_reference_arm_corstone3xx) + +list(APPEND CMAKE_MODULE_PATH ${IOT_REFERENCE_ARM_CORSTONE3XX_SOURCE_DIR}/components/aws_iot/cmake) +list(APPEND CMAKE_MODULE_PATH ${IOT_REFERENCE_ARM_CORSTONE3XX_SOURCE_DIR}/components/ai/ml_embedded_evaluation_kit/integration/cmake) +list(APPEND CMAKE_MODULE_PATH ${IOT_REFERENCE_ARM_CORSTONE3XX_SOURCE_DIR}/components/security/trusted_firmware-m/integration/cmake) +include(SetupMlEmbeddedEvaluationKitLibraries) +include(GenerateAWSUpdateDigestAndSignature) +include(MergeTfmImages) +include(SignTfmImage) + +add_subdirectory(configs ${CMAKE_BINARY_DIR}/Config) +add_subdirectory(isp) +add_subdirectory(../helpers helpers) + +add_executable(object-detection + blink_task.c + main.c + ml_interface.cc +) + +target_include_directories(object-detection + PUBLIC + ../helpers/provisioning + ${CMAKE_CURRENT_LIST_DIR} +) + +if (${ML_INFERENCE_ENGINE} STREQUAL "ETHOS") + target_compile_definitions(object-detection PRIVATE USE_ETHOS) + target_sources(object-detection PRIVATE ../helpers/ethosu/src/ethosu_platform_adaptation.c) +endif() + +target_compile_options(object-detection + PRIVATE + $<$:-std=c++14> + $<$:-std=c99> +) + +#FIXME USE RTE_USART2 for control tool +target_compile_definitions(arm-corstone-platform-bsp + PUBLIC + RTE_USART1=1 +) + +# Trusted Firmware-M must be built before the application, because +# the application depends on the NS interface and the BL2 signing scripts, +# both of which are generated as parts of the Trusted Firmware-M build process. +add_dependencies(object-detection trusted_firmware-m-build) +# The provision data must be built before the application because +# it provides credentials to connect to AWS. +add_dependencies(object-detection provisioning_data_bin) + +target_link_libraries(object-detection + PRIVATE + arm-2d + backoff-algorithm + connectivity-stack + coremqtt + coremqtt-agent + corepkcs11 + freertos_kernel + freertos-ota-pal-psa + fri-bsp + helpers-events + helpers-hdlcd + isp-config + isp_platform_driver + mbedtls + mbedtls-threading-freertos + ota-for-aws-iot-embedded-sdk + provisioning-lib + tfm-ns-interface + toolchain-override + object_detection_api + object_detection_model +) + +include(${IOT_REFERENCE_ARM_CORSTONE3XX_SOURCE_DIR}/bsp/cmake/SetLinkerOptions.cmake) +set_linker_script(object-detection) + +# The non-secure application image should be padded while being signed +# Hence, passing "TRUE" as the input parameter to the pad option of sign function. +iot_reference_arm_corstone3xx_tf_m_sign_image( + object-detection + object-detection_signed + ${MCUBOOT_IMAGE_VERSION_NS} + TRUE +) + +# The update image is not padded to fill the whole slot (no --pad), because +# 1) the image to download is smaller without padding +# 2) the trailer that keeps track of boot and update statuses should not be overwritten +# Hence, passing "FALSE" as the input parameter for the pad option to the sign function. +iot_reference_arm_corstone3xx_tf_m_sign_image( + object-detection + object-detection-update_signed + ${MCUBOOT_IMAGE_VERSION_NS_UPDATE} + FALSE +) + + +# A user project that consumes the ARM FIR needs to explicitly provide +# addresses in order to merge images for TF-M. The addresses cannot +# be easily programmatically extracted as they are defined in the linker +# scripts. +iot_reference_arm_corstone3xx_tf_m_merge_images( + object-detection + ${NS_PROVISIONING_BUNDLE_LOAD_ADDRESS} + ${CMAKE_BINARY_DIR}/helpers/provisioning/provisioning_data.bin +) + +iot_reference_arm_corstone3xx_generate_aws_update_digest_and_signature( + object-detection + object-detection-update_signed + update-digest + update-signature +) diff --git a/applications/object_detection/blink_task.c b/applications/object_detection/blink_task.c new file mode 100644 index 00000000..313b9361 --- /dev/null +++ b/applications/object_detection/blink_task.c @@ -0,0 +1,93 @@ +/* Copyright 2021-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include "FreeRTOS.h" +#include "blink_task.h" +#include "mps3_leds.h" +#include "task.h" + +#include +#include + +/* Include header that defines log levels. */ +#include "logging_levels.h" + +/* Configure name and log level. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "Blink" +#endif +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif +#include "logging_stack.h" + +#define blinky_taskBLINK_TIMER_PERIOD_MS 250 + +enum +{ + LED1 = 1 << 0, + LED2 = 1 << 1, + LED3 = 1 << 2, + LED4 = 1 << 3, + LED5 = 1 << 4, + LED6 = 1 << 5, + LED_ALIVE = LED6, + LED7 = 1 << 6, + LED8 = 1 << 7, + LED9 = 1 << 8, + LED10 = 1 << 9, + LED_ALL = 0xFF +}; + + +void vStartBlinkTask( void ) +{ + if( + xTaskCreate( + vBlinkTask, + "BLINK_TASK ", + appCONFIG_BLINK_TASK_STACK_SIZE, + NULL, + appCONFIG_BLINK_TASK_PRIORITY, + NULL + ) != pdPASS + ) + { + LogError( ( "Failed to create Blink Task\r\n" ) ); + } +} + +void vBlinkTask( void * pvParameters ) +{ + ( void ) pvParameters; + + LogInfo( ( "Blink task started\r\n" ) ); + + mps3_leds_init(); + + if( mps3_leds_turn_off( LED_ALL ) != true ) + { + LogError( ( "Failed to turn all LEDs off\r\n" ) ); + return; + } + + const uint32_t ulTicksInterval = blinky_taskBLINK_TIMER_PERIOD_MS * configTICK_RATE_HZ / 1000; + + while( 1 ) + { + if( ulTicksInterval == 0U ) + { + return; + } + + vTaskDelay( ulTicksInterval ); + + if( mps3_leds_toggle( LED_ALIVE ) != true ) + { + LogError( ( "Failed to toggle LED_ALIVE\r\n" ) ); + return; + } + } +} diff --git a/applications/object_detection/blink_task.h b/applications/object_detection/blink_task.h new file mode 100644 index 00000000..7c76d002 --- /dev/null +++ b/applications/object_detection/blink_task.h @@ -0,0 +1,32 @@ +/* Copyright 2021-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#ifndef BLINK_TASK_H + #define BLINK_TASK_H + + #ifdef __cplusplus + extern "C" { + #endif + +/** + * @brief Create blink task. + */ + void vStartBlinkTask( void ); + +/** + * @brief Blinks LEDs according to ML processing. + * + * LED1 ON and LED2 OFF => heard YES + * LED1 OFF and LED2 OFF => heard NO + * LED1 OFF and LED2 blinking => no/unknown input + * @param pvParameters Contextual data for the task. + */ + void vBlinkTask( void * pvParameters ); + + #ifdef __cplusplus + } + #endif + +#endif /* ! BLINK_TASK_H*/ diff --git a/applications/object_detection/configs/CMakeLists.txt b/applications/object_detection/configs/CMakeLists.txt new file mode 100644 index 00000000..0a308861 --- /dev/null +++ b/applications/object_detection/configs/CMakeLists.txt @@ -0,0 +1,10 @@ +# Copyright 2023-2024, Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +add_subdirectory(app_config) +add_subdirectory(arm_2d_config) +add_subdirectory(aws_configs) +add_subdirectory(isp_config) +add_subdirectory(freertos_config) +add_subdirectory(mbedtls_config) diff --git a/applications/object_detection/configs/app_config/CMakeLists.txt b/applications/object_detection/configs/app_config/CMakeLists.txt new file mode 100644 index 00000000..24bd2482 --- /dev/null +++ b/applications/object_detection/configs/app_config/CMakeLists.txt @@ -0,0 +1,10 @@ +# Copyright 2023-2024, Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +add_library(app-config INTERFACE) + +target_include_directories(app-config + INTERFACE + . +) diff --git a/applications/object_detection/configs/app_config/app_config.h b/applications/object_detection/configs/app_config/app_config.h new file mode 100644 index 00000000..ef917056 --- /dev/null +++ b/applications/object_detection/configs/app_config/app_config.h @@ -0,0 +1,59 @@ +/* Copyright 2023-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +/** @brief Set logging task as high priority task */ +#define appCONFIG_LOGGING_TASK_PRIORITY ( configMAX_PRIORITIES - 1 ) +#define appCONFIG_LOGGING_TASK_STACK_SIZE ( 2048 ) +#define appCONFIG_LOGGING_MESSAGE_QUEUE_LENGTH ( 32 ) + + +/** + * @brief Stack size and priority for OTA MQTT agent task. + * Stack size is capped to an adequate value based on requirements from MbedTLS stack + * for establishing a TLS connection. Task priority of OTA MQTT agent is set to a priority + * higher than other MQTT application tasks, so that the agent can drain the queue + * as work is being produced. + */ +#define appCONFIG_OTA_MQTT_AGENT_TASK_STACK_SIZE ( 4096 ) +#define appCONFIG_OTA_MQTT_AGENT_TASK_PRIORITY ( tskIDLE_PRIORITY + 3 ) + +/** + * @brief Stack size and priority for MQTT agent task. + * Stack size is capped to an adequate value based on requirements from MbedTLS stack + * for establishing a TLS connection. Task priority of MQTT agent is set to a priority + * higher than other MQTT application tasks, so that the agent can drain the queue + * as work is being produced. + */ +#define appCONFIG_MQTT_AGENT_TASK_STACK_SIZE ( 4096 ) +#define appCONFIG_MQTT_AGENT_TASK_PRIORITY ( tskIDLE_PRIORITY + 2 ) + + +#define appCONFIG_ML_TASK_STACK_SIZE ( 8192 ) +#define appCONFIG_ML_TASK_PRIORITY ( tskIDLE_PRIORITY + 1 ) + +#define appCONFIG_DSP_TASK_STACK_SIZE ( 8192 ) +#define appCONFIG_DSP_TASK_PRIORITY ( tskIDLE_PRIORITY + 1 ) + + +#define appCONFIG_ML_MQTT_TASK_STACK_SIZE ( configMINIMAL_STACK_SIZE ) +#define appCONFIG_ML_MQTT_TASK_PRIORITY ( tskIDLE_PRIORITY + 1 ) + +/** + * @brief Stack size and priority for Blink task. + * Stack size is capped to an minimal value. Task priority of Blink is set to a priority + * lower than other ML task. + */ +#define appCONFIG_BLINK_TASK_STACK_SIZE ( configMINIMAL_STACK_SIZE ) +#define appCONFIG_BLINK_TASK_PRIORITY ( tskIDLE_PRIORITY ) + +#define appCONFIG_VSI_CALLBACK_TASK_STACK_SIZE ( configMINIMAL_STACK_SIZE ) +#define appCONFIG_VSI_CALLBACK_TASK_PRIORITY ( tskIDLE_PRIORITY + 2 ) + + +/** @brief Increase backoff algorithm timeout by 8 seconds when device advisor + * test is active. + */ + +#define appCONFIG_DEVICE_ADVISOR_TEST_ACTIVE 0 diff --git a/applications/object_detection/configs/arm_2d_config/CMakeLists.txt b/applications/object_detection/configs/arm_2d_config/CMakeLists.txt new file mode 100644 index 00000000..6fd33498 --- /dev/null +++ b/applications/object_detection/configs/arm_2d_config/CMakeLists.txt @@ -0,0 +1,10 @@ +# Copyright 2024 Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +add_library(arm_2d-config INTERFACE) + +target_include_directories(arm_2d-config + INTERFACE + . +) diff --git a/applications/object_detection/configs/arm_2d_config/arm_2d_cfg.h b/applications/object_detection/configs/arm_2d_config/arm_2d_cfg.h new file mode 100644 index 00000000..892a6ace --- /dev/null +++ b/applications/object_detection/configs/arm_2d_config/arm_2d_cfg.h @@ -0,0 +1,225 @@ +/* + * Copyright (c) 2009-2024, Arm Limited. All rights reserved. + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 (the License); you may + * not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an AS IS BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef __ARM_2D_USER_CFG_H__ + #define __ARM_2D_USER_CFG_H__ + +/*============================ INCLUDES ======================================*/ + +/*#include "RTE_Components.h" */ + + #ifdef __cplusplus + extern "C" { + #endif + +/*============================ MACROS ========================================*/ +/*============================ MACROFIED FUNCTIONS ===========================*/ + +/*-------- <<< Use Configuration Wizard in Context Menu >>> ----------------- */ +/* */ +/* Arm-2D General Configurations */ +/* ======================= */ +/* */ +/* Enable Asynchronous Programmers' model support */ +/* Note that if you don't have any hardware accelerator, disable this feature can reduce code size and gain a small */ +/* performance uplift. This feature is enabled by default. */ + #ifndef __ARM_2D_HAS_ASYNC__ + #define __ARM_2D_HAS_ASYNC__ 0 + #endif + +/* Enable anti-alias support for all tranform operations. */ +/* Note that enabling this feature suffers a non-negligible performance drop. */ +/* This feature is disabled by default. */ + #ifndef __ARM_2D_HAS_ANTI_ALIAS_TRANSFORM__ + #define __ARM_2D_HAS_ANTI_ALIAS_TRANSFORM__ 0 + #endif + +/* Enable support for accessing individual colour channels */ +/* Note that enabling this feature will add the support for a special colour type: ARM_2D_CHANNEL_8in32 */ +/* This feature is disabled by default to save code size */ + #ifndef __ARM_2D_CFG_SUPPORT_COLOUR_CHANNEL_ACCESS__ + #define __ARM_2D_CFG_SUPPORT_COLOUR_CHANNEL_ACCESS__ 1 + #endif +/* */ + +/* Patches for improving performance */ +/* ======================= */ +/* */ +/* Do NOT treat alpha value 255 as completely opaque in mask related operations */ +/* When define this macro, alpha value 0xFF will not be treated as opaque in mask related operations and you can */ +/* barely see the background. Defining this macro improves performance. */ +/*#define __ARM_2D_CFG_UNSAFE_IGNORE_ALPHA_255_COMPENSATION__ */ +/* */ + +/* Ignore calibration for small angles in transform operations */ +/* This option is used to speed up M-cores without DSP support. It skips saturation in the QADD/QDADD/QDSUB involved */ +/* in the rotation. The chances of overflow remain low as elements involved are using non-accumulating Q15.16 format and */ +/* integer parts are in the range of the screen size providing enough margin. */ +/*#define __ARM_2D_CFG_UNSAFE_IGNORE_CALIB_IN_TRANSFORM__ */ +/* */ + +/* Ignore saturation protection in fixed-point operations */ +/* This option is used to remove calibration in angle computations to gain a better performance, small error might */ +/* be noticeable for angles like 90, 180, 270 etc. */ +/*#define __ARM_2D_CFG_UNSAFE_NO_SATURATION_IN_FIXED_POINT__ */ +/* */ + +/* Optimize the scaler version of transform operations for pointer-like resources */ +/* This feature is enabled by default. There is no guarantee that the performance will increase or decrease. It is */ +/* all depends your applications. In most of the case, enabling it helps. This feature has no meaning when the */ +/* anti-alias transform is disabled or the helium acceleration is available. */ + #ifndef __ARM_2D_CFG_OPTIMIZE_FOR_POINTER_LIKE_SHAPES_IN_TRANSFORM__ + #define __ARM_2D_CFG_OPTIMIZE_FOR_POINTER_LIKE_SHAPES_IN_TRANSFORM__ 1 + #endif +/* */ + +/* Extra Components */ +/* ======================= */ +/* */ + #ifndef __GLCD_CFG_COLOUR_DEPTH__ +/* Select the screen colour depth */ +/* <8=> 8 Bits */ +/* <16=> 16Bits */ +/* <32=> 32Bits */ +/* The colour depth of your LCD */ +/* Default: 16 */ + #define __GLCD_CFG_COLOUR_DEPTH__ 16 + #endif + +/* Width of the screen <8-32767> */ +/* The width of your screen */ +/* Default: 320 */ + #ifndef __GLCD_CFG_SCREEN_WIDTH__ + #define __GLCD_CFG_SCREEN_WIDTH__ 320 + #endif + +/* Height of the screen <8-32767> */ +/* The height of your screen */ +/* Default: 240 */ + #ifndef __GLCD_CFG_SCREEN_HEIGHT__ + #define __GLCD_CFG_SCREEN_HEIGHT__ 240 + #endif + +/* The size of the LCD printf text buffer <16-65535> */ +/* The text buffer size for the lcd printf service. It determins how many character you can use in one printf */ +/* string. */ + #ifndef __LCD_PRINTF_CFG_TEXT_BUFFER_SIZE__ + #define __LCD_PRINTF_CFG_TEXT_BUFFER_SIZE__ 64 + #endif + +/* Benchmark */ +/* Width of the PFB block <8-32767> */ +/* The width of your PFB block size used in arm-2d benchmark */ + #ifndef BENCHMARK_PFB_BLOCK_WIDTH + #define BENCHMARK_PFB_BLOCK_WIDTH 320 + #endif + +/* Height of the PFB block <8-32767> */ +/* The height of your PFB block size used in arm-2d benchmark */ + #ifndef BENCHMARK_PFB_BLOCK_HEIGHT + #define BENCHMARK_PFB_BLOCK_HEIGHT 240 + #endif + +/* PFB Block Count <1-65535> */ +/* The number of blocks in the PFB pool. */ + #ifndef BENCHMARK_PFB_HEAP_SIZE + #define BENCHMARK_PFB_HEAP_SIZE 1 + #endif + +/* Width Alignment of generated PFBs */ +/* <0=> 1 pixel */ +/* <1=> 2 pixel */ +/* <2=> 4 pixel */ +/* <3=> 8 pixel */ +/* <4=> 16 pixel */ +/* <5=> 32 pixel */ +/* <6=> 64 pixel */ +/* <7=> 128 pixel */ +/* Make sure the x and width of the PFB is always aligned to 2^n pixels */ + #ifndef __BENCHMARK_PFB_PIXEL_ALIGN_WIDTH__ + #define __BENCHMARK_PFB_PIXEL_ALIGN_WIDTH__ 0 + #endif + +/* Height Alignment of generated PFBs */ +/* <0=> 1 pixel */ +/* <1=> 2 pixel */ +/* <2=> 4 pixel */ +/* <3=> 8 pixel */ +/* <4=> 16 pixel */ +/* <5=> 32 pixel */ +/* <6=> 64 pixel */ +/* <7=> 128 pixel */ +/* Make sure the y and height of the PFB is always aligned to 2^n pixels */ + #ifndef __BENCHMARK_PFB_PIXEL_ALIGN_HEIGHT__ + #define __BENCHMARK_PFB_PIXEL_ALIGN_HEIGHT__ 0 + #endif + +/* Swap the high and low bytes */ +/* Swap the high and low bytes of the 16bit-pixels */ + #ifndef __ARM_2D_CFG_BENCHMARK_SWAP_RGB16_HIGH_AND_LOW_BYTES__ + #define __ARM_2D_CFG_BENCHMARK_SWAP_RGB16_HIGH_AND_LOW_BYTES__ 0 + #endif + +/* Enable the helper service for Asynchronous Flushing */ +/* Please select this option when using asynchronous flushing, e.g. DMA + ISR */ + #ifndef __ARM_2D_CFG_BENCHMARK_ENABLE_ASYNC_FLUSHING__ + #define __ARM_2D_CFG_BENCHMARK_ENABLE_ASYNC_FLUSHING__ 0 + #endif + +/* Number of iterations <1-2000> */ +/* run number of iterations in arm-2d benchmark before calculating the result. */ + #ifndef ITERATION_CNT + #define ITERATION_CNT 1000 + #endif + +/* Use Tiny mode to run benchmark */ +/* Enable this mode to reduce the benchmark memory footprint (removing background picture etc.) */ +/* This feature is disabled by default. */ + #ifndef __ARM_2D_CFG_BENCHMARK_TINY_MODE__ + #define __ARM_2D_CFG_BENCHMARK_TINY_MODE__ 0 + #endif + +/* Enable Stopwatch mode in the Benchmark:Watch-panel */ +/* Only update the second-hand (i.e. red pointer) every second in the watch-panel demo */ +/* This feature is disabled by default. */ + #ifndef __ARM_2D_CFG_WATCH_PANEL_STOPWATCH_MODE__ + #define __ARM_2D_CFG_WATCH_PANEL_STOPWATCH_MODE__ 0 + #endif + +/* Exit benchmark when finished */ +/* Exit the arm_2d_run_benchmark() after running specified iterations */ +/* This feature is disabled by default. */ + #ifndef __ARM_2D_CFG_BENCHMARK_EXIT_WHEN_FINISH__ + #define __ARM_2D_CFG_BENCHMARK_EXIT_WHEN_FINISH__ 0 + #endif + +/* */ +/* */ + +/* <<< end of configuration section >>> */ + +/*============================ TYPES =========================================*/ +/*============================ GLOBAL VARIABLES ==============================*/ +/*============================ LOCAL VARIABLES ===============================*/ +/*============================ PROTOTYPES ====================================*/ + + #ifdef __cplusplus + } + #endif + +#endif /* ifndef __ARM_2D_USER_CFG_H__ */ diff --git a/applications/object_detection/configs/aws_configs/CMakeLists.txt b/applications/object_detection/configs/aws_configs/CMakeLists.txt new file mode 100644 index 00000000..d8c60fcf --- /dev/null +++ b/applications/object_detection/configs/aws_configs/CMakeLists.txt @@ -0,0 +1,38 @@ +# Copyright 2023-2024, Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +target_include_directories(coremqtt-config + INTERFACE + . +) + +target_link_libraries(coremqtt-config + INTERFACE + helpers-logging +) + +target_compile_definitions(coremqtt-agent-config + INTERFACE + MQTT_AGENT_DO_NOT_USE_CUSTOM_CONFIG +) + +target_include_directories(corepkcs11-config + INTERFACE + . +) + +target_include_directories(freertos-ota-pal-psa-config + INTERFACE + . +) + +target_include_directories(freertos-pkcs11-psa-config + INTERFACE + . +) + +target_include_directories(ota-for-aws-iot-embedded-sdk-config + INTERFACE + . +) diff --git a/applications/object_detection/configs/aws_configs/aws_clientcredential.h b/applications/object_detection/configs/aws_configs/aws_clientcredential.h new file mode 100644 index 00000000..df1dd08a --- /dev/null +++ b/applications/object_detection/configs/aws_configs/aws_clientcredential.h @@ -0,0 +1,51 @@ +/* Copyright 2023-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#ifndef __AWS_CLIENTCREDENTIAL__H__ +#define __AWS_CLIENTCREDENTIAL__H__ + +/** + * @brief Endpoint of the MQTT broker to connect to. + * + * This demo application can be run with any MQTT broker, that supports mutual + * authentication. + * + * For AWS IoT MQTT broker, this is the Thing's REST API Endpoint. + * + * @note Replace the used dummy value "dummy.endpointid.amazonaws.com" by your + * AWS IoT Core endpoint which can be found in the AWS IoT console under + * Settings/Custom Endpoint, or using the describe-endpoint REST API (with + * AWS CLI command line tool). + * + */ + +#define clientcredentialMQTT_BROKER_ENDPOINT "dummy.endpointid.amazonaws.com" + +/** + * @brief The MQTT client identifier used in this example. Each client identifier + * must be unique; so replace the used dummy value "dummy_thingname" as required to + * ensure that no two clients connecting to the same broker use the same client identifier. + * + * Value is defined in "aws_clientcredential.h". + */ + +#define clientcredentialIOT_THING_NAME "dummy_thingname" + +/** + * @brief The port to use for the demo. + * + * In general, port 8883 is for secured MQTT connections. + * + * @note Port 443 requires use of the ALPN TLS extension with the ALPN protocol + * name. When using port 8883, ALPN is not required. + * + */ +#define clientcredentialMQTT_BROKER_PORT 8883 + +#define clientcredentialGREENGRASS_DISCOVERY_PORT 8443 +#define clientcredentialWIFI_SSID "" +#define clientcredentialWIFI_PASSWORD "" +#define clientcredentialWIFI_SECURITY eWiFiSecurityWPA2 +#endif /* ifndef __AWS_CLIENTCREDENTIAL__H__ */ diff --git a/applications/object_detection/configs/aws_configs/aws_mqtt_config.h b/applications/object_detection/configs/aws_configs/aws_mqtt_config.h new file mode 100644 index 00000000..efef52a5 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/aws_mqtt_config.h @@ -0,0 +1,69 @@ +/* + * Amazon FreeRTOS V1.1.4 + * Copyright (C) 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://aws.amazon.com/freertos + * http://www.FreeRTOS.org + */ + +/** + * @file aws_mqtt_config.h + * @brief MQTT config options. + */ + +#ifndef _AWS_MQTT_CONFIG_H_ +#define _AWS_MQTT_CONFIG_H_ + +/** + * @brief Define assert for test project. + */ +#define mqttconfigASSERT( x ) if( ( x ) == 0 ) __asm( "b . " ); + +/** + * @brief Enable subscription management. + * + * This gives the user flexibility of registering a callback per subscription. + */ +#define mqttconfigENABLE_SUBSCRIPTION_MANAGEMENT ( 1 ) + +/** + * @brief Maximum length of the topic which can be stored in subscription + * manager. + */ +#define mqttconfigSUBSCRIPTION_MANAGER_MAX_TOPIC_LENGTH ( 128 ) + +/** + * @brief Maximum number of subscriptions which can be stored in subscription + * manager. + */ +#define mqttconfigSUBSCRIPTION_MANAGER_MAX_SUBSCRIPTIONS ( 8 ) + +/* + * Uncomment the following two lines to enable asserts. + */ +/* extern void vAssertCalled( const char *pcFile, uint32_t ulLine ); */ +/* #define mqttconfigASSERT( x ) if( ( x ) == 0 ) vAssertCalled( __FILE__, __LINE__ ) */ + +/** + * @brief Set this macro to 1 for enabling debug logs. + */ +#define mqttconfigENABLE_DEBUG_LOGS ( 0 ) + +#endif /* _AWS_MQTT_CONFIG_H_ */ diff --git a/applications/object_detection/configs/aws_configs/aws_secure_sockets_config.h b/applications/object_detection/configs/aws_configs/aws_secure_sockets_config.h new file mode 100644 index 00000000..a3924669 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/aws_secure_sockets_config.h @@ -0,0 +1,57 @@ +/* + * FreeRTOS V202107.00 + * Copyright (C) 2021 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * Copyright (c) 2022-2024, Arm Limited and Contributors. All rights reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://aws.amazon.com/freertos + * http://www.FreeRTOS.org + */ + +#ifndef IOT_SECURE_SOCKETS_CONFIG_H +#define IOT_SECURE_SOCKETS_CONFIG_H + +#include "RTOS_config.h" + +/** + * @brief Byte order of the target MCU. + * + * Valid values are pdLITTLE_ENDIAN and pdBIG_ENDIAN. + */ +#define socketsconfigBYTE_ORDER pdLITTLE_ENDIAN + +/** + * @brief Default socket send timeout. + */ +#define socketsconfigDEFAULT_SEND_TIMEOUT ( 10000 ) + +/** + * @brief Default socket receive timeout. + */ +#define socketsconfigDEFAULT_RECV_TIMEOUT ( 10000 ) + +/** + * @brief Enable metrics of secure socket. + */ +#define AWS_IOT_SECURE_SOCKETS_METRICS_ENABLED ( 0 ) + +#define socketsconfigDEFAULT_MAX_NUM_SECURE_SOCKETS 4 +#define socketsconfigRECEIVE_CALLBACK_TASK_STACK_DEPTH 512 + +#endif /*IOT_SECURE_SOCKETS_CONFIG_H*/ diff --git a/applications/object_detection/configs/aws_configs/core_mqtt_config.h b/applications/object_detection/configs/aws_configs/core_mqtt_config.h new file mode 100644 index 00000000..8ea1f7bc --- /dev/null +++ b/applications/object_detection/configs/aws_configs/core_mqtt_config.h @@ -0,0 +1,78 @@ +/* + * Copyright (C) 2021 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + */ +#ifndef CORE_MQTT_CONFIG_H +#define CORE_MQTT_CONFIG_H + +#include "logging_levels.h" + +/* Logging configuration for the MQTT library. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "MQTT" +#endif + +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif + +#include "logging_stack.h" + +/** + * @brief The maximum number of MQTT PUBLISH messages that may be pending + * acknowledgment at any time. + * + * QoS 1 and 2 MQTT PUBLISHes require acknowledgment from the server before + * they can be completed. While they are awaiting the acknowledgment, the + * client must maintain information about their state. The value of this + * macro sets the limit on how many simultaneous PUBLISH states an MQTT + * context maintains. + */ +#define MQTT_STATE_ARRAY_MAX_COUNT ( 20U ) +/* #define MQTT_RECV_POLLING_TIMEOUT_MS ( 250 ) */ + +/*_RB_ To document and add to the mqtt config defaults header file. */ +#define MQTT_AGENT_COMMAND_QUEUE_LENGTH ( 32 ) +#define MQTT_COMMAND_CONTEXTS_POOL_SIZE ( 32 ) + +/** + * @brief The maximum number of subscriptions to track for a single connection. + * + * @note The MQTT agent keeps a record of all existing MQTT subscriptions. + * MQTT_AGENT_MAX_SIMULTANEOUS_SUBSCRIPTIONS sets the maximum number of + * subscriptions records that can be maintained at one time. The higher this + * number is the greater the agent's RAM consumption will be. + */ +#define MQTT_AGENT_MAX_SIMULTANEOUS_SUBSCRIPTIONS ( 10 ) + +/** + * @brief Size of statically allocated buffers for holding subscription filters. + * + * @note Subscription filters are strings such as "/my/topicname/#". These + * strings are limited to a maximum of MQTT_AGENT_MAX_SUBSCRIPTION_FILTER_LENGTH + * characters. The higher this number is the greater the agent's RAM consumption + * will be. + */ +#define MQTT_AGENT_MAX_SUBSCRIPTION_FILTER_LENGTH ( 100 ) + +#endif /* ifndef CORE_MQTT_CONFIG_H */ diff --git a/applications/object_detection/configs/aws_configs/core_pkcs11_config.h b/applications/object_detection/configs/aws_configs/core_pkcs11_config.h new file mode 100644 index 00000000..9cdf1322 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/core_pkcs11_config.h @@ -0,0 +1,160 @@ +/* + * Amazon FreeRTOS V1.1.4 + * Copyright (C) 2018 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * Copyright (c) 2022-2024, Arm Limited and Contributors. All rights reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://aws.amazon.com/freertos + * http://www.FreeRTOS.org + */ + +/** + * @file aws_pkcs11_config.h + * @brief PCKS#11 config options. + */ + + +#ifndef _AWS_PKCS11_CONFIG_H_ +#define _AWS_PKCS11_CONFIG_H_ + +#include + +/** + * @brief Malloc API used by core_pkcs11.h + */ +extern void * pvPortMalloc( size_t xWantedSize ); +#define PKCS11_MALLOC pvPortMalloc + +/** + * @brief Free API used by core_pkcs11.h + */ +extern void vPortFree( void * pv ); +#define PKCS11_FREE vPortFree + +/* A non-standard version of C_INITIALIZE should be used by this port. */ +/* #define pkcs11configC_INITIALIZE_ALT */ + +/** + * @brief PKCS #11 default user PIN. + * + * The PKCS #11 standard specifies the presence of a user PIN. That feature is + * sensible for applications that have an interactive user interface and memory + * protections. However, since typical microcontroller applications lack one or + * both of those, the user PIN is assumed to be used herein for interoperability + * purposes only, and not as a security feature. + */ +#define configPKCS11_DEFAULT_USER_PIN "0000" + +/** + * @brief Maximum length (in characters) for a PKCS #11 CKA_LABEL + * attribute. + */ +#define pkcs11configMAX_LABEL_LENGTH 32 + +/** + * @brief Maximum number of token objects that can be stored + * by the PKCS #11 module. + */ +#define pkcs11configMAX_NUM_OBJECTS 6 + +/** + * @brief Set to 1 if a PAL destroy object is implemented. + * + * If set to 0, no PAL destroy object is implemented, and this functionality + * is implemented in the common PKCS #11 layer. + */ +#define pkcs11configPAL_DESTROY_SUPPORTED 0 + +/** + * @brief Set to 1 if OTA image verification via PKCS #11 module is supported. + * + * If set to 0, OTA code signing certificate is built in via + * aws_ota_codesigner_certificate.h. + */ +#define pkcs11configOTA_SUPPORTED 0 + +/** + * @brief Set to 1 if PAL supports storage for JITP certificate, + * code verify certificate, and trusted server root certificate. + * + * If set to 0, PAL does not support storage mechanism for these, and + * they are accessed via headers compiled into the code. + */ +#define pkcs11configJITP_CODEVERIFY_ROOT_CERT_SUPPORTED 0 + +/** + * @brief The PKCS #11 label for device private key. + * + * Private key for connection to AWS IoT endpoint. The corresponding + * public key should be registered with the AWS IoT endpoint. + */ +#define pkcs11configLABEL_DEVICE_PRIVATE_KEY_FOR_TLS "Device Priv TLS Key" + +/** + * @brief The PKCS #11 label for device public key. + * + * The public key corresponding to pkcs11configLABEL_DEVICE_PRIVATE_KEY_FOR_TLS. + */ +#define pkcs11configLABEL_DEVICE_PUBLIC_KEY_FOR_TLS "Device Pub TLS Key" + +/** + * @brief The PKCS #11 label for the device certificate. + * + * Device certificate corresponding to pkcs11configLABEL_DEVICE_PRIVATE_KEY_FOR_TLS. + */ +#define pkcs11configLABEL_DEVICE_CERTIFICATE_FOR_TLS "Device Cert" + +/** + * @brief The PKCS #11 label for the object to be used for code verification. + * + * Used by over-the-air update code to verify an incoming signed image. + */ +#define pkcs11configLABEL_CODE_VERIFICATION_KEY "Code Verify Key" + +/** + * @brief The PKCS #11 label for Just-In-Time-Provisioning. + * + * The certificate corresponding to the issuer of the device certificate + * (pkcs11configLABEL_DEVICE_CERTIFICATE_FOR_TLS) when using the JITR or + * JITP flow. + */ +#define pkcs11configLABEL_JITP_CERTIFICATE "JITP Cert" + +/** + * @brief The PKCS #11 label for the AWS Trusted Root Certificate. + * + * @see aws_default_root_certificates.h + */ +#define pkcs11configLABEL_ROOT_CERTIFICATE "Root Cert" + +/* + * Define the key ID of the device keys which will be saved as + * persistent keys in TF-M. The key ID servers as the a name. + */ +#define PSA_DEVICE_PRIVATE_KEY_ID 0x01 +#define PSA_DEVICE_PUBLIC_KEY_ID 0x10 + +/* */ +/* FIXME: are these needed? */ +/* extern void *MPU_pvPortMalloc( size_t xSize ); */ +/* extern void MPU_vPortFree( void *pv ); */ + +/* #define pvPortMalloc MPU_pvPortMalloc */ +/* #define vPortFree MPU_vPortFree */ +#endif /* _AWS_PKCS11_CONFIG_H_ include guard. */ diff --git a/applications/object_detection/configs/aws_configs/demo_config.h b/applications/object_detection/configs/aws_configs/demo_config.h new file mode 100644 index 00000000..e38a35d8 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/demo_config.h @@ -0,0 +1,82 @@ +/* + * FreeRTOS V202010.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + */ + +#ifndef DEMO_CONFIG_H +#define DEMO_CONFIG_H + +#include "core_mqtt.h" +#include "aws_clientcredential.h" +#include "iot_default_root_certificates.h" + +/** + * To use this demo, please configure the client's certificate and private key + * in aws_configs/aws_clientcredential_keys.h + * and credentials in aws_configs/aws_clientcredential.h. + * + * For the AWS IoT MQTT broker, refer to the AWS documentation below for details + * regarding client authentication. + * https://docs.aws.amazon.com/iot/latest/developerguide/client-authentication.html + */ + +/** + * @brief The name of the operating system that the application is running on. + * The current value is given as an example. Please update for your specific + * operating system. + */ +#define democonfigOS_NAME "FreeRTOS" + +/** + * @brief The version of the operating system that the application is running + * on. The current value is given as an example. Please update for your specific + * operating system version. + */ +#define democonfigOS_VERSION "V10.4.3" + +/** + * @brief The name of the MQTT library used and its version, following an "@" + * symbol. + */ +#define democonfigMQTT_LIB "core-mqtt@" MQTT_LIBRARY_VERSION + +/** + * @brief The MQTT metrics string expected by AWS IoT. + */ +#define democonfigMQTT_USERNAME \ + "?SDK=" democonfigOS_NAME "&Version=" democonfigOS_VERSION \ + "&MQTTLib=" democonfigMQTT_LIB + +/** + * @brief The length of the MQTT metrics string expected by AWS IoT. + */ +#define democonfigMQTT_USERNAME_LENGTH ( ( uint16_t ) ( sizeof( democonfigMQTT_USERNAME ) - 1 ) ) + +/* these are set in "aws_clientcredential.h and iot_default_root_certificates.h" */ +#define democonfigCLIENT_IDENTIFIER clientcredentialIOT_THING_NAME +#define democonfigMQTT_BROKER_ENDPOINT clientcredentialMQTT_BROKER_ENDPOINT +#define democonfigMQTT_BROKER_PORT clientcredentialMQTT_BROKER_PORT +#define democonfigROOT_CA_PEM tlsATS1_ROOT_CERTIFICATE_PEM + +#endif /* DEMO_CONFIG_H */ diff --git a/applications/object_detection/configs/aws_configs/iot_default_root_certificates.h b/applications/object_detection/configs/aws_configs/iot_default_root_certificates.h new file mode 100644 index 00000000..f8243e7b --- /dev/null +++ b/applications/object_detection/configs/aws_configs/iot_default_root_certificates.h @@ -0,0 +1,197 @@ +/* + * FreeRTOS Common V1.2.0 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://aws.amazon.com/freertos + * http://www.FreeRTOS.org + */ + +#ifndef __DEFAULT__ROOT__CERTIFICATES__H__ +#define __DEFAULT__ROOT__CERTIFICATES__H__ + +/* + * https://www.amazontrust.com/repository/AmazonRootCA1.pem + * RSA 2048-bit + */ +static const char tlsATS1_ROOT_CERTIFICATE_PEM[] = + "-----BEGIN CERTIFICATE-----\n" + "MIIDQTCCAimgAwIBAgITBmyfz5m/jAo54vB4ikPmljZbyjANBgkqhkiG9w0BAQsF\n" + "ADA5MQswCQYDVQQGEwJVUzEPMA0GA1UEChMGQW1hem9uMRkwFwYDVQQDExBBbWF6\n" + "b24gUm9vdCBDQSAxMB4XDTE1MDUyNjAwMDAwMFoXDTM4MDExNzAwMDAwMFowOTEL\n" + "MAkGA1UEBhMCVVMxDzANBgNVBAoTBkFtYXpvbjEZMBcGA1UEAxMQQW1hem9uIFJv\n" + "b3QgQ0EgMTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBALJ4gHHKeNXj\n" + "ca9HgFB0fW7Y14h29Jlo91ghYPl0hAEvrAIthtOgQ3pOsqTQNroBvo3bSMgHFzZM\n" + "9O6II8c+6zf1tRn4SWiw3te5djgdYZ6k/oI2peVKVuRF4fn9tBb6dNqcmzU5L/qw\n" + "IFAGbHrQgLKm+a/sRxmPUDgH3KKHOVj4utWp+UhnMJbulHheb4mjUcAwhmahRWa6\n" + "VOujw5H5SNz/0egwLX0tdHA114gk957EWW67c4cX8jJGKLhD+rcdqsq08p8kDi1L\n" + "93FcXmn/6pUCyziKrlA4b9v7LWIbxcceVOF34GfID5yHI9Y/QCB/IIDEgEw+OyQm\n" + "jgSubJrIqg0CAwEAAaNCMEAwDwYDVR0TAQH/BAUwAwEB/zAOBgNVHQ8BAf8EBAMC\n" + "AYYwHQYDVR0OBBYEFIQYzIU07LwMlJQuCFmcx7IQTgoIMA0GCSqGSIb3DQEBCwUA\n" + "A4IBAQCY8jdaQZChGsV2USggNiMOruYou6r4lK5IpDB/G/wkjUu0yKGX9rbxenDI\n" + "U5PMCCjjmCXPI6T53iHTfIUJrU6adTrCC2qJeHZERxhlbI1Bjjt/msv0tadQ1wUs\n" + "N+gDS63pYaACbvXy8MWy7Vu33PqUXHeeE6V/Uq2V8viTO96LXFvKWlJbYK8U90vv\n" + "o/ufQJVtMVT8QtPHRh8jrdkPSHCa2XV4cdFyQzR1bldZwgJcJmApzyMZFo6IQ6XU\n" + "5MsI+yMRQ+hDKXJioaldXgjUkK642M4UwtBV8ob2xJNDd2ZhwLnoQdeXeGADbkpy\n" + "rqXRfboQnoZsG4q5WTP468SQvvG5\n" + "-----END CERTIFICATE-----\n"; +static const uint32_t tlsATS1_ROOT_CERTIFICATE_LENGTH = sizeof( tlsATS1_ROOT_CERTIFICATE_PEM ); + +/* + * https://www.amazontrust.com/repository/AmazonRootCA2.pem + * RSA 4096-bit + */ +static const char tlsATS2_ROOT_CERTIFICATE_PEM[] = + "-----BEGIN CERTIFICATE-----\n" + "MIIFQTCCAymgAwIBAgITBmyf0pY1hp8KD+WGePhbJruKNzANBgkqhkiG9w0BAQwF\n" + "ADA5MQswCQYDVQQGEwJVUzEPMA0GA1UEChMGQW1hem9uMRkwFwYDVQQDExBBbWF6\n" + "b24gUm9vdCBDQSAyMB4XDTE1MDUyNjAwMDAwMFoXDTQwMDUyNjAwMDAwMFowOTEL\n" + "MAkGA1UEBhMCVVMxDzANBgNVBAoTBkFtYXpvbjEZMBcGA1UEAxMQQW1hem9uIFJv\n" + "b3QgQ0EgMjCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAK2Wny2cSkxK\n" + "gXlRmeyKy2tgURO8TW0G/LAIjd0ZEGrHJgw12MBvIITplLGbhQPDW9tK6Mj4kHbZ\n" + "W0/jTOgGNk3Mmqw9DJArktQGGWCsN0R5hYGCrVo34A3MnaZMUnbqQ523BNFQ9lXg\n" + "1dKmSYXpN+nKfq5clU1Imj+uIFptiJXZNLhSGkOQsL9sBbm2eLfq0OQ6PBJTYv9K\n" + "8nu+NQWpEjTj82R0Yiw9AElaKP4yRLuH3WUnAnE72kr3H9rN9yFVkE8P7K6C4Z9r\n" + "2UXTu/Bfh+08LDmG2j/e7HJV63mjrdvdfLC6HM783k81ds8P+HgfajZRRidhW+me\n" + "z/CiVX18JYpvL7TFz4QuK/0NURBs+18bvBt+xa47mAExkv8LV/SasrlX6avvDXbR\n" + "8O70zoan4G7ptGmh32n2M8ZpLpcTnqWHsFcQgTfJU7O7f/aS0ZzQGPSSbtqDT6Zj\n" + "mUyl+17vIWR6IF9sZIUVyzfpYgwLKhbcAS4y2j5L9Z469hdAlO+ekQiG+r5jqFoz\n" + "7Mt0Q5X5bGlSNscpb/xVA1wf+5+9R+vnSUeVC06JIglJ4PVhHvG/LopyboBZ/1c6\n" + "+XUyo05f7O0oYtlNc/LMgRdg7c3r3NunysV+Ar3yVAhU/bQtCSwXVEqY0VThUWcI\n" + "0u1ufm8/0i2BWSlmy5A5lREedCf+3euvAgMBAAGjQjBAMA8GA1UdEwEB/wQFMAMB\n" + "Af8wDgYDVR0PAQH/BAQDAgGGMB0GA1UdDgQWBBSwDPBMMPQFWAJI/TPlUq9LhONm\n" + "UjANBgkqhkiG9w0BAQwFAAOCAgEAqqiAjw54o+Ci1M3m9Zh6O+oAA7CXDpO8Wqj2\n" + "LIxyh6mx/H9z/WNxeKWHWc8w4Q0QshNabYL1auaAn6AFC2jkR2vHat+2/XcycuUY\n" + "+gn0oJMsXdKMdYV2ZZAMA3m3MSNjrXiDCYZohMr/+c8mmpJ5581LxedhpxfL86kS\n" + "k5Nrp+gvU5LEYFiwzAJRGFuFjWJZY7attN6a+yb3ACfAXVU3dJnJUH/jWS5E4ywl\n" + "7uxMMne0nxrpS10gxdr9HIcWxkPo1LsmmkVwXqkLN1PiRnsn/eBG8om3zEK2yygm\n" + "btmlyTrIQRNg91CMFa6ybRoVGld45pIq2WWQgj9sAq+uEjonljYE1x2igGOpm/Hl\n" + "urR8FLBOybEfdF849lHqm/osohHUqS0nGkWxr7JOcQ3AWEbWaQbLU8uz/mtBzUF+\n" + "fUwPfHJ5elnNXkoOrJupmHN5fLT0zLm4BwyydFy4x2+IoZCn9Kr5v2c69BoVYh63\n" + "n749sSmvZ6ES8lgQGVMDMBu4Gon2nL2XA46jCfMdiyHxtN/kHNGfZQIG6lzWE7OE\n" + "76KlXIx3KadowGuuQNKotOrN8I1LOJwZmhsoVLiJkO/KdYE+HvJkJMcYr07/R54H\n" + "9jVlpNMKVv/1F2Rs76giJUmTtt8AF9pYfl3uxRuw0dFfIRDH+fO6AgonB8Xx1sfT\n" + "4PsJYGw=\n" + "-----END CERTIFICATE-----\n"; +static const uint32_t tlsATS2_ROOT_CERTIFICATE_LENGTH = sizeof( tlsATS2_ROOT_CERTIFICATE_PEM ); + +/* + * https://www.amazontrust.com/repository/AmazonRootCA3.pem + * EC 256-bit + */ +static const char tlsATS3_ROOT_CERTIFICATE_PEM[] = + "-----BEGIN CERTIFICATE-----\n" + "MIIBtjCCAVugAwIBAgITBmyf1XSXNmY/Owua2eiedgPySjAKBggqhkjOPQQDAjA5\n" + "MQswCQYDVQQGEwJVUzEPMA0GA1UEChMGQW1hem9uMRkwFwYDVQQDExBBbWF6b24g\n" + "Um9vdCBDQSAzMB4XDTE1MDUyNjAwMDAwMFoXDTQwMDUyNjAwMDAwMFowOTELMAkG\n" + "A1UEBhMCVVMxDzANBgNVBAoTBkFtYXpvbjEZMBcGA1UEAxMQQW1hem9uIFJvb3Qg\n" + "Q0EgMzBZMBMGByqGSM49AgEGCCqGSM49AwEHA0IABCmXp8ZBf8ANm+gBG1bG8lKl\n" + "ui2yEujSLtf6ycXYqm0fc4E7O5hrOXwzpcVOho6AF2hiRVd9RFgdszflZwjrZt6j\n" + "QjBAMA8GA1UdEwEB/wQFMAMBAf8wDgYDVR0PAQH/BAQDAgGGMB0GA1UdDgQWBBSr\n" + "ttvXBp43rDCGB5Fwx5zEGbF4wDAKBggqhkjOPQQDAgNJADBGAiEA4IWSoxe3jfkr\n" + "BqWTrBqYaGFy+uGh0PsceGCmQ5nFuMQCIQCcAu/xlJyzlvnrxir4tiz+OpAUFteM\n" + "YyRIHN8wfdVoOw==\n" + "-----END CERTIFICATE-----\n"; +static const uint32_t tlsATS3_ROOT_CERTIFICATE_LENGTH = sizeof( tlsATS3_ROOT_CERTIFICATE_PEM ); + +/* + * https://www.amazontrust.com/repository/AmazonRootCA4.pem + * EC 384-bit + */ +static const char tlsATS4_ROOT_CERTIFICATE_PEM[] = + "-----BEGIN CERTIFICATE-----\n" + "MIIB8jCCAXigAwIBAgITBmyf18G7EEwpQ+Vxe3ssyBrBDjAKBggqhkjOPQQDAzA5\n" + "MQswCQYDVQQGEwJVUzEPMA0GA1UEChMGQW1hem9uMRkwFwYDVQQDExBBbWF6b24g\n" + "Um9vdCBDQSA0MB4XDTE1MDUyNjAwMDAwMFoXDTQwMDUyNjAwMDAwMFowOTELMAkG\n" + "A1UEBhMCVVMxDzANBgNVBAoTBkFtYXpvbjEZMBcGA1UEAxMQQW1hem9uIFJvb3Qg\n" + "Q0EgNDB2MBAGByqGSM49AgEGBSuBBAAiA2IABNKrijdPo1MN/sGKe0uoe0ZLY7Bi\n" + "9i0b2whxIdIA6GO9mif78DluXeo9pcmBqqNbIJhFXRbb/egQbeOc4OO9X4Ri83Bk\n" + "M6DLJC9wuoihKqB1+IGuYgbEgds5bimwHvouXKNCMEAwDwYDVR0TAQH/BAUwAwEB\n" + "/zAOBgNVHQ8BAf8EBAMCAYYwHQYDVR0OBBYEFNPsxzplbszh2naaVvuc84ZtV+WB\n" + "MAoGCCqGSM49BAMDA2gAMGUCMDqLIfG9fhGt0O9Yli/W651+kI0rz2ZVwyzjKKlw\n" + "CkcO8DdZEv8tmZQoTipPNU0zWgIxAOp1AE47xDqUEpHJWEadIRNyp4iciuRMStuW\n" + "1KyLa2tJElMzrdfkviT8tQp21KW8EA==\n" + "-----END CERTIFICATE-----\n"; +static const uint32_t tlsATS4_ROOT_CERTIFICATE_LENGTH = sizeof( tlsATS4_ROOT_CERTIFICATE_PEM ); + +/* + * Starfield Cross-signing CA + */ +static const char tlsSTARFIELD_ROOT_CERTIFICATE_PEM[] = + "-----BEGIN CERTIFICATE-----\n" + "MIIEDzCCAvegAwIBAgIBADANBgkqhkiG9w0BAQUFADBoMQswCQYDVQQGEwJVUzEl\n" + "MCMGA1UEChMcU3RhcmZpZWxkIFRlY2hub2xvZ2llcywgSW5jLjEyMDAGA1UECxMp\n" + "U3RhcmZpZWxkIENsYXNzIDIgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkwHhcNMDQw\n" + "NjI5MTczOTE2WhcNMzQwNjI5MTczOTE2WjBoMQswCQYDVQQGEwJVUzElMCMGA1UE\n" + "ChMcU3RhcmZpZWxkIFRlY2hub2xvZ2llcywgSW5jLjEyMDAGA1UECxMpU3RhcmZp\n" + "ZWxkIENsYXNzIDIgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkwggEgMA0GCSqGSIb3\n" + "DQEBAQUAA4IBDQAwggEIAoIBAQC3Msj+6XGmBIWtDBFk385N78gDGIc/oav7PKaf\n" + "8MOh2tTYbitTkPskpD6E8J7oX+zlJ0T1KKY/e97gKvDIr1MvnsoFAZMej2YcOadN\n" + "+lq2cwQlZut3f+dZxkqZJRRU6ybH838Z1TBwj6+wRir/resp7defqgSHo9T5iaU0\n" + "X9tDkYI22WY8sbi5gv2cOj4QyDvvBmVmepsZGD3/cVE8MC5fvj13c7JdBmzDI1aa\n" + "K4UmkhynArPkPw2vCHmCuDY96pzTNbO8acr1zJ3o/WSNF4Azbl5KXZnJHoe0nRrA\n" + "1W4TNSNe35tfPe/W93bC6j67eA0cQmdrBNj41tpvi/JEoAGrAgEDo4HFMIHCMB0G\n" + "A1UdDgQWBBS/X7fRzt0fhvRbVazc1xDCDqmI5zCBkgYDVR0jBIGKMIGHgBS/X7fR\n" + "zt0fhvRbVazc1xDCDqmI56FspGowaDELMAkGA1UEBhMCVVMxJTAjBgNVBAoTHFN0\n" + "YXJmaWVsZCBUZWNobm9sb2dpZXMsIEluYy4xMjAwBgNVBAsTKVN0YXJmaWVsZCBD\n" + "bGFzcyAyIENlcnRpZmljYXRpb24gQXV0aG9yaXR5ggEAMAwGA1UdEwQFMAMBAf8w\n" + "DQYJKoZIhvcNAQEFBQADggEBAAWdP4id0ckaVaGsafPzWdqbAYcaT1epoXkJKtv3\n" + "L7IezMdeatiDh6GX70k1PncGQVhiv45YuApnP+yz3SFmH8lU+nLMPUxA2IGvd56D\n" + "eruix/U0F47ZEUD0/CwqTRV/p2JdLiXTAAsgGh1o+Re49L2L7ShZ3U0WixeDyLJl\n" + "xy16paq8U4Zt3VekyvggQQto8PT7dL5WXXp59fkdheMtlb71cZBDzI0fmgAKhynp\n" + "VSJYACPq4xJDKVtHCN2MQWplBqjlIapBtJUhlbl90TSrE9atvNziPTnNvT51cKEY\n" + "WQPJIrSPnNVeKtelttQKbfi3QBFGmh95DmK/D5fs4C8fF5Q=\n" + "-----END CERTIFICATE-----\n"; +static const uint32_t tlsSTARFIELD_ROOT_CERTIFICATE_LENGTH = sizeof( tlsSTARFIELD_ROOT_CERTIFICATE_PEM ); + +/* + * VeriSign-Class 3-Public-Primary-Certification-Authority-G5 + */ +static const char tlsVERISIGN_ROOT_CERTIFICATE_PEM[] = + "-----BEGIN CERTIFICATE-----\n" + "MIIE0zCCA7ugAwIBAgIQGNrRniZ96LtKIVjNzGs7SjANBgkqhkiG9w0BAQUFADCB\n" + "yjELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDlZlcmlTaWduLCBJbmMuMR8wHQYDVQQL\n" + "ExZWZXJpU2lnbiBUcnVzdCBOZXR3b3JrMTowOAYDVQQLEzEoYykgMjAwNiBWZXJp\n" + "U2lnbiwgSW5jLiAtIEZvciBhdXRob3JpemVkIHVzZSBvbmx5MUUwQwYDVQQDEzxW\n" + "ZXJpU2lnbiBDbGFzcyAzIFB1YmxpYyBQcmltYXJ5IENlcnRpZmljYXRpb24gQXV0\n" + "aG9yaXR5IC0gRzUwHhcNMDYxMTA4MDAwMDAwWhcNMzYwNzE2MjM1OTU5WjCByjEL\n" + "MAkGA1UEBhMCVVMxFzAVBgNVBAoTDlZlcmlTaWduLCBJbmMuMR8wHQYDVQQLExZW\n" + "ZXJpU2lnbiBUcnVzdCBOZXR3b3JrMTowOAYDVQQLEzEoYykgMjAwNiBWZXJpU2ln\n" + "biwgSW5jLiAtIEZvciBhdXRob3JpemVkIHVzZSBvbmx5MUUwQwYDVQQDEzxWZXJp\n" + "U2lnbiBDbGFzcyAzIFB1YmxpYyBQcmltYXJ5IENlcnRpZmljYXRpb24gQXV0aG9y\n" + "aXR5IC0gRzUwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCvJAgIKXo1\n" + "nmAMqudLO07cfLw8RRy7K+D+KQL5VwijZIUVJ/XxrcgxiV0i6CqqpkKzj/i5Vbex\n" + "t0uz/o9+B1fs70PbZmIVYc9gDaTY3vjgw2IIPVQT60nKWVSFJuUrjxuf6/WhkcIz\n" + "SdhDY2pSS9KP6HBRTdGJaXvHcPaz3BJ023tdS1bTlr8Vd6Gw9KIl8q8ckmcY5fQG\n" + "BO+QueQA5N06tRn/Arr0PO7gi+s3i+z016zy9vA9r911kTMZHRxAy3QkGSGT2RT+\n" + "rCpSx4/VBEnkjWNHiDxpg8v+R70rfk/Fla4OndTRQ8Bnc+MUCH7lP59zuDMKz10/\n" + "NIeWiu5T6CUVAgMBAAGjgbIwga8wDwYDVR0TAQH/BAUwAwEB/zAOBgNVHQ8BAf8E\n" + "BAMCAQYwbQYIKwYBBQUHAQwEYTBfoV2gWzBZMFcwVRYJaW1hZ2UvZ2lmMCEwHzAH\n" + "BgUrDgMCGgQUj+XTGoasjY5rw8+AatRIGCx7GS4wJRYjaHR0cDovL2xvZ28udmVy\n" + "aXNpZ24uY29tL3ZzbG9nby5naWYwHQYDVR0OBBYEFH/TZafC3ey78DAJ80M5+gKv\n" + "MzEzMA0GCSqGSIb3DQEBBQUAA4IBAQCTJEowX2LP2BqYLz3q3JktvXf2pXkiOOzE\n" + "p6B4Eq1iDkVwZMXnl2YtmAl+X6/WzChl8gGqCBpH3vn5fJJaCGkgDdk+bW48DW7Y\n" + "5gaRQBi5+MHt39tBquCWIMnNZBU4gcmU7qKEKQsTb47bDN0lAtukixlE0kF6BWlK\n" + "WE9gyn6CagsCqiUXObXbf+eEZSqVir2G3l6BFoMtEMze/aiCKm0oHw0LxOXnGiYZ\n" + "4fQRbxC1lfznQgUy286dUV4otp6F01vvpX1FQHKOtw5rDgb7MzVIcbidJ4vEZV8N\n" + "hnacRHr2lVz2XTIIM6RUthg/aFzyQkqFOFSDX9HoLPKsEdao7WNq\n" + "-----END CERTIFICATE-----\n"; +static const uint32_t tlsVERISIGN_ROOT_CERTIFICATE_LENGTH = sizeof( tlsVERISIGN_ROOT_CERTIFICATE_PEM ); + +#endif /* ifndef __DEFAULT__ROOT__CERTIFICATES__H__ */ diff --git a/applications/object_detection/configs/aws_configs/mqtt_agent_demo_config.h b/applications/object_detection/configs/aws_configs/mqtt_agent_demo_config.h new file mode 100644 index 00000000..664d07f3 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/mqtt_agent_demo_config.h @@ -0,0 +1,129 @@ +/* + * FreeRTOS V202010.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + */ + +#ifndef MQTT_DEMO_CONNECTION_SHARING_CONFIG_H +#define MQTT_DEMO_CONNECTION_SHARING_CONFIG_H + +/**************************************************/ +/******* DO NOT CHANGE the following order ********/ +/**************************************************/ + +/* Include logging header files and define logging macros in the following order: + * 1. Include the header file "logging_levels.h". + * 2. Define the LIBRARY_LOG_NAME and LIBRARY_LOG_LEVEL macros depending on + * the logging configuration for DEMO. + * 3. Include the header file "logging_stack.h", if logging is enabled for DEMO. + */ + +#include "logging_levels.h" + +/* Logging configuration for the Demo. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "MQTTDemo" +#endif + +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif +#include "logging_stack.h" + +/************ End of logging configuration ****************/ + +/** + * @brief The MQTT client identifier used in this example. Each client identifier + * must be unique so edit as required to ensure no two clients connecting to the + * same broker use the same client identifier. + * + * #define democonfigCLIENT_IDENTIFIER "insert here." + */ + + +/** + * @brief Endpoint of the MQTT broker to connect to. + * + * This demo application can be run with any MQTT broker, although it is + * recommended to use one that supports mutual authentication. If mutual + * authentication is not used, then #democonfigUSE_TLS should be set to 0. + * + * For AWS IoT MQTT broker, this is the Thing's REST API Endpoint. + * + * @note Your AWS IoT Core endpoint can be found in the AWS IoT console under + * Settings/Custom Endpoint, or using the describe-endpoint REST API (with + * AWS CLI command line tool). + * + * #define democonfigMQTT_BROKER_ENDPOINT "insert here." + */ + + +/** + * @brief The port to use for the demo. + * + * In general, port 8883 is for secured MQTT connections, and port 1883 if not + * using TLS. + * + * @note Port 443 requires use of the ALPN TLS extension with the ALPN protocol + * name. Using ALPN with this demo would require additional changes, including + * setting the `pAlpnProtos` member of the `NetworkCredentials_t` struct before + * forming the TLS connection. When using port 8883, ALPN is not required. + * + * #define democonfigMQTT_BROKER_PORT ( insert here. ) + */ + +/** + * @brief Dimensions the buffer used to serialize and deserialize MQTT packets. + * + * @note Specified in bytes. Must be large enough to hold the maximum + * anticipated MQTT payload. + * + * #define MQTT_AGENT_NETWORK_BUFFER_SIZE ( insert here. ) + */ + +/** + * @brief The length of the queue used to hold commands for the agent. + * + * #define MQTT_AGENT_COMMAND_QUEUE_LENGTH ( insert here. ) + */ + +/** + * @brief Maximum number of subscriptions maintained by the subscription manager + * simultaneously in a list. + * + * #define SUBSCRIPTION_MANAGER_MAX_SUBSCRIPTIONS ( insert here. ) + */ + +/** + * @brief The number of simple subscribe-publish tasks to create for the demo + */ +#define democonfigNUM_SIMPLE_SUB_PUB_TASKS_TO_CREATE 1 +#define democonfigSIMPLE_SUB_PUB_TASK_STACK_SIZE ( configMINIMAL_STACK_SIZE * 4 ) + + +/** + * @brief The maximum number of times to run the demo's task creation loop. + */ +#define democonfigMQTT_MAX_DEMO_COUNT ( 3 ) + +#endif /* MQTT_DEMO_CONNECTION_SHARING_CONFIG_H */ diff --git a/applications/object_detection/configs/aws_configs/mqtt_demo_mutual_auth_config.h b/applications/object_detection/configs/aws_configs/mqtt_demo_mutual_auth_config.h new file mode 100644 index 00000000..6ac43b79 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/mqtt_demo_mutual_auth_config.h @@ -0,0 +1,112 @@ +/* + * FreeRTOS V202010.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + */ + +#ifndef MQTT_DEMO_MUTUAL_AUTH_CONFIG_H +#define MQTT_DEMO_MUTUAL_AUTH_CONFIG_H + +/**************************************************/ +/******* DO NOT CHANGE the following order ********/ +/**************************************************/ + +/* Include logging header files and define logging macros in the following order: + * 1. Include the header file "logging_levels.h". + * 2. Define the LIBRARY_LOG_NAME and LIBRARY_LOG_LEVEL macros depending on + * the logging configuration for DEMO. + * 3. Include the header file "logging_stack.h", if logging is enabled for DEMO. + */ + +#include "logging_levels.h" + +/* Logging configuration for the Demo. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "MQTT_MutualAuth_Demo" +#endif + +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif +#include "logging_stack.h" + +/************ End of logging configuration ****************/ + +/** + * To use this demo, please configure the client's certificate and private key + * in demos/include/aws_clientcredential_keys.h. + * + * For the AWS IoT MQTT broker, refer to the AWS documentation below for details + * regarding client authentication. + * https://docs.aws.amazon.com/iot/latest/developerguide/client-authentication.html + */ + +/** + * @brief The MQTT client identifier used in this example. Each client identifier + * must be unique; so edit as required to ensure that no two clients connecting to + * the same broker use the same client identifier. + * + * #define democonfigCLIENT_IDENTIFIER "insert here." + */ + +/** + * @brief Endpoint of the MQTT broker to connect to. + * + * This demo application can be run with any MQTT broker, that supports mutual + * authentication. + * + * For AWS IoT MQTT broker, this is the Thing's REST API Endpoint. + * + * @note Your AWS IoT Core endpoint can be found in the AWS IoT console under + * Settings/Custom Endpoint, or using the describe-endpoint REST API (with + * AWS CLI command line tool). + * + * #define democonfigMQTT_BROKER_ENDPOINT "...insert here..." + */ + +/** + * @brief The port to use for the demo. + * + * In general, port 8883 is for secured MQTT connections. + * + * @note Port 443 requires use of the ALPN TLS extension with the ALPN protocol + * name. When using port 8883, ALPN is not required. + * + * #define democonfigMQTT_BROKER_PORT "...insert here..." + */ +#define democonfigMQTT_BROKER_PORT 8883 + +/** + * @brief Size of the network buffer for MQTT packets. + */ +#define democonfigNETWORK_BUFFER_SIZE ( 1500 ) + +/** + * @brief The maximum number of times to run the subscribe publish loop in the + * demo. + * + * #define democonfigMQTT_MAX_DEMO_COUNT ( insert here ) + */ +#define democonfigMQTT_MAX_DEMO_COUNT ( 3 ) + +#endif /* MQTT_DEMO_MUTUAL_AUTH_CONFIG_H */ diff --git a/applications/object_detection/configs/aws_configs/ota_config.h b/applications/object_detection/configs/aws_configs/ota_config.h new file mode 100644 index 00000000..829b25f7 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/ota_config.h @@ -0,0 +1,206 @@ +/* + * FreeRTOS V202104.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + */ + +/** + * @file ota_config.h + * @brief OTA user configurable settings. + */ + +#ifndef OTA_CONFIG_H_ +#define OTA_CONFIG_H_ + +/**************************************************/ +/******* DO NOT CHANGE the following order ********/ +/**************************************************/ + +/* Logging related header files are required to be included in the following order: + * 1. Include the header file "logging_levels.h". + * 2. Define LIBRARY_LOG_NAME and LIBRARY_LOG_LEVEL. + * 3. Include the header file "logging_stack.h". + */ + +/* Include header that defines log levels. */ +#include "logging_levels.h" + +/* Configure name and log level for the OTA library. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "OTA" +#endif +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif +#include "logging_stack.h" + +/************ End of logging configuration ****************/ + +/* Include config file used for testing the OTA PAL. This config file defines + * the otapalconfigCODE_SIGNING_CERTIFICATE macro. */ +#include "ota_demo_config.h" + +/** + * @brief Log base 2 of the size of the file data block message (excluding the + * header). + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigLOG2_FILE_BLOCK_SIZE 12UL /* TODO */ + +/** + * @brief Size of the file data block message (excluding the header). + */ +#define otaconfigFILE_BLOCK_SIZE ( 1UL << otaconfigLOG2_FILE_BLOCK_SIZE ) + +/** + * @brief Milliseconds to wait for the self test phase to succeed before we + * force reset. + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigSELF_TEST_RESPONSE_WAIT_MS 16000U /* TODO */ + +/** + * @brief Milliseconds to wait before requesting data blocks from the OTA + * service if nothing is happening. + * + * @note The wait timer is reset whenever a data block is received from the OTA + * service so we will only send the request message after being idle for this + * amount of time. + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigFILE_REQUEST_WAIT_MS 100000U /* TODO */ + +/** + * @brief The maximum allowed length of the thing name used by the OTA agent. + * + * @note AWS IoT requires Thing names to be unique for each device that + * connects to the broker. Likewise, the OTA agent requires the developer to + * construct and pass in the Thing name when initializing the OTA agent. The + * agent uses this size to allocate static storage for the Thing name used in + * all OTA base topics. Namely $aws/things/thingName + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigMAX_THINGNAME_LEN 128U /* TODO */ + +/** + * @brief The maximum number of data blocks requested from OTA streaming + * service. + * + * @note This configuration parameter is sent with data requests and represents + * the maximum number of data blocks the service will send in response. The + * maximum limit for this must be calculated from the maximum data response + * limit (128 KB from service) divided by the block size. For example if block + * size is set as 1 KB then the maximum number of data blocks that we can + * request is 128/1 = 128 blocks. Configure this parameter to this maximum + * limit or lower based on how many data blocks response is expected for each + * data requests. + * + * Possible values: Any unsigned 32 integer value greater than 0.
+ */ +#define otaconfigMAX_NUM_BLOCKS_REQUEST 1U /* TODO */ + +/** + * @brief The maximum number of requests allowed to send without a response + * before we abort. + * + * @note This configuration parameter sets the maximum number of times the + * requests are made over the selected communication channel before aborting + * and returning error. + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigMAX_NUM_REQUEST_MOMENTUM 32U /* TODO */ + +/** + * @brief How frequently the device will report its OTA progress to the cloud. + * + * @note Device will update the job status with the number of blocks it has + * received every certain number of blocks it receives. For example, 64 means + * device will update job status every 64 blocks it receives. + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigOTA_UPDATE_STATUS_FREQUENCY 25U /* TODO */ + +/** + * @brief The number of data buffers reserved by the OTA agent. + * + * @note This configurations parameter sets the maximum number of static data + * buffers used by the OTA agent for job and file data blocks received. + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigMAX_NUM_OTA_DATA_BUFFERS otaconfigMAX_NUM_BLOCKS_REQUEST + 1 /* TODO */ + +/** + * @brief Flag to enable booting into updates that have an identical or lower + * version than the current version. + * + * @note Set this configuration parameter to '1' to disable version checks. + * This allows updates to an identical or lower version. This is provided for + * testing purpose and it's recommended to always update to higher version and + * keep this configuration disabled. + * + * Possible values: Any unsigned 32 integer.
+ */ +#define otaconfigAllowDowngrade 0U /* TODO */ + +/** + * @brief The protocol selected for OTA control operations. + * + * @note This configurations parameter sets the default protocol for all the + * OTA control operations like requesting OTA job, updating the job status etc. + * + * @note Only MQTT is supported at this time for control operations. + * + * Possible values: OTA_CONTROL_OVER_MQTT
+ */ +#define configENABLED_CONTROL_PROTOCOL ( OTA_CONTROL_OVER_MQTT ) + +/** + * @brief The protocol selected for OTA data operations. + * + * @note This configurations parameter sets the protocols selected for the data + * operations like requesting file blocks from the service. + * + * Possible values:
+ * Enable data over MQTT - ( OTA_DATA_OVER_MQTT )
+ * Enable data over HTTP - ( OTA_DATA_OVER_HTTP )
+ * Enable data over both MQTT & HTTP - ( OTA_DATA_OVER_MQTT | OTA_DATA_OVER_HTTP )
+ */ +#define configENABLED_DATA_PROTOCOLS ( OTA_DATA_OVER_MQTT ) + +/** + * @brief The preferred protocol selected for OTA data operations. + * + * @note Primary data protocol will be the protocol used for downloading file + * if more than one protocol is selected while creating OTA job. + * + * Possible values:
+ * Data over MQTT - ( OTA_DATA_OVER_MQTT )
+ * Data over HTTP - ( OTA_DATA_OVER_HTTP )
+ */ + +#define configOTA_PRIMARY_DATA_PROTOCOL ( OTA_DATA_OVER_MQTT ) + +#endif /* OTA_CONFIG_H_ */ diff --git a/applications/object_detection/configs/aws_configs/ota_demo_config.h b/applications/object_detection/configs/aws_configs/ota_demo_config.h new file mode 100644 index 00000000..c6f3aa73 --- /dev/null +++ b/applications/object_detection/configs/aws_configs/ota_demo_config.h @@ -0,0 +1,103 @@ +/* + * FreeRTOS V202104.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + */ + +/** + * @file ota_demo_config.h + * @brief Configuration options for the OTA related demos. + */ + +#ifndef OTA_DEMO_CONFIG_H_ +#define OTA_DEMO_CONFIG_H_ + +/** + * @brief Certificate used for validating code signing signatures in the OTA PAL. + */ +#ifndef otapalconfigCODE_SIGNING_CERTIFICATE + #define otapalconfigCODE_SIGNING_CERTIFICATE "Insert code signing certificate..." /* TODO */ +#endif + +/** + * @brief Whether to use const qualifier for the appFirmwareVersion variable. + * + * @note In some cases the appFirmwareVersion variable cannot be declared as const + * because the version is read out during runtime. + * + * Possible values:
+ * appFirmwareVersion is const - ( 0 )
+ * appFirmwareVersion is non-const - ( 1 )
+ * Default value: '0' + */ +#ifndef OTA_USE_NONCONST_APPVERSION + #define OTA_USE_NONCONST_APPVERSION ( 1U ) +#endif + + +/** + * @brief Major version of the firmware. + * + * This is used in the OTA demo to set the appFirmwareVersion variable + * that is declared in the ota_appversion32.h file in the OTA library. + */ +#ifndef APP_VERSION_MAJOR + #define APP_VERSION_MAJOR 0 /* TODO */ +#endif + +/** + * @brief Minor version of the firmware. + * + * This is used in the OTA demo to set the appFirmwareVersion variable + * that is declared in the ota_appversion32.h file in the OTA library. + */ +#ifndef APP_VERSION_MINOR + #define APP_VERSION_MINOR 0 /* TODO */ +#endif + +/** + * @brief Build version of the firmware. + * + * This is used in the OTA demo to set the appFirmwareVersion variable + * that is declared in the ota_appversion32.h file in the OTA library. + */ +#ifndef APP_VERSION_BUILD + #define APP_VERSION_BUILD 0 /* TODO */ +#endif + +/** + * @brief Timeout for which MQTT library keeps polling the transport interface, + * when no byte is received. + * The timeout is honoured only after the first byte is read and while remaining + * bytes are read from network interface. Keeping this timeout to a sufficiently + * large value so as to account for delay of receipt of a large block of message. + */ +#undef MQTT_RECV_POLLING_TIMEOUT_MS /* Override the one from "core_mqtt_config_defaults.h" */ +#define MQTT_RECV_POLLING_TIMEOUT_MS ( 1000U ) /* TODO Set a timeout in msecs for data received from MQTT. Recommend values > 1 s. */ + +/** + * @brief Maximum time MQTT agent waits in the queue for any pending MQTT + * operations. The wait time is kept smallest possible to increase the + * responsiveness of MQTT agent while processing pending MQTT operations as + * well as receive packets from network. + */ +#undef MQTT_AGENT_MAX_EVENT_QUEUE_WAIT_TIME /* Override the one from "core_mqtt_config_defaults.h" */ +#define MQTT_AGENT_MAX_EVENT_QUEUE_WAIT_TIME ( 1U ) /* TODO */ + +#endif /* OTA_DEMO_CONFIG_H_ */ diff --git a/applications/object_detection/configs/freertos_config/CMakeLists.txt b/applications/object_detection/configs/freertos_config/CMakeLists.txt new file mode 100644 index 00000000..bf3ae06d --- /dev/null +++ b/applications/object_detection/configs/freertos_config/CMakeLists.txt @@ -0,0 +1,20 @@ +# Copyright 2023-2024, Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +target_include_directories(freertos_config SYSTEM + INTERFACE + . +) + +target_compile_definitions(freertos_config + INTERFACE + projCOVERAGE_TEST=0 +) + +target_link_libraries(freertos_config + INTERFACE + tfm-ns-interface + app-config + fri-bsp +) diff --git a/applications/object_detection/configs/freertos_config/FreeRTOSConfig.h b/applications/object_detection/configs/freertos_config/FreeRTOSConfig.h new file mode 100644 index 00000000..847bf7fb --- /dev/null +++ b/applications/object_detection/configs/freertos_config/FreeRTOSConfig.h @@ -0,0 +1,237 @@ +/* + * FreeRTOS Kernel V10.4.1 + * Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * Copyright (c) 2022-2024, Arm Limited and Contributors. All rights reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * http://aws.amazon.com/freertos + * http://www.FreeRTOS.org + */ + +#ifndef FREERTOS_CONFIG_H +#define FREERTOS_CONFIG_H + +/* Here is a good place to include header files that are required across + * your application. */ +/* #include */ +/* #include "aws_secure_sockets_config.h" */ +/* #include "RTOS_config.h" */ + +#include "app_config.h" + +#ifndef __USED + #define __USED __attribute__( ( used ) ) +#endif +#ifndef __WEAK + #define __WEAK __attribute__( ( weak ) ) +#endif + + +extern uint32_t SystemCoreClock; + +/* Memory allocation related definitions. */ +#define configSUPPORT_STATIC_ALLOCATION 1 +#define configSUPPORT_DYNAMIC_ALLOCATION 1 +#define configTOTAL_HEAP_SIZE 0 +#define configAPPLICATION_ALLOCATED_HEAP 0 + +#define configENABLE_MVE 0 +#define configENABLE_FPU 1 +#define configENABLE_MPU 0 +#define configENABLE_TRUSTZONE 0 +#define configRUN_FREERTOS_SECURE_ONLY 0 + +/* The target specific macros `configTICK_RATE_HZ`, `pdMS_TO_TICKS` and + * `TICKS_TO_pdMS` are defined in `FreeRTOSConfig_target.h`. */ +#include "FreeRTOSConfig_target.h" + +#define configMINIMAL_STACK_SIZE 4096 +#define configUSE_16_BIT_TICKS 0 +#define portTICK_TYPE_IS_ATOMIC 1 + +#define configUSE_PREEMPTION 1 +#define configUSE_PORT_OPTIMISED_TASK_SELECTION 0 +#define configUSE_TICKLESS_IDLE 0 +#define configCPU_CLOCK_HZ ( ( unsigned long ) SystemCoreClock ) +#define configMAX_PRIORITIES 56 +#define configMAX_TASK_NAME_LEN 16 +#define configIDLE_SHOULD_YIELD 1 +#define configUSE_TASK_NOTIFICATIONS 1 +#define configTASK_NOTIFICATION_ARRAY_ENTRIES 3 +#define configUSE_MUTEXES 1 +#define configUSE_RECURSIVE_MUTEXES 1 +#define configUSE_COUNTING_SEMAPHORES 1 +#define configUSE_ALTERNATIVE_API 0 /* Deprecated! */ +#define configQUEUE_REGISTRY_SIZE 10 +#define configUSE_QUEUE_SETS 0 +#define configUSE_TIME_SLICING 1 +#define configUSE_NEWLIB_REENTRANT 0 +#define configENABLE_BACKWARD_COMPATIBILITY 0 +#define configNUM_THREAD_LOCAL_STORAGE_POINTERS 5 + +/* Hook function related definitions. */ +#define configUSE_IDLE_HOOK 0 +#define configUSE_TICK_HOOK 0 +#define configCHECK_FOR_STACK_OVERFLOW 1 /* This should only be set for development */ +#define configUSE_MALLOC_FAILED_HOOK 0 +#define configUSE_DAEMON_TASK_STARTUP_HOOK 0 + +/* Run time and task stats gathering related definitions. */ +#define configGENERATE_RUN_TIME_STATS 0 +#define configUSE_TRACE_FACILITY 1 +#define configUSE_STATS_FORMATTING_FUNCTIONS 0 + +/* Co-routine related definitions. */ +#define configUSE_CO_ROUTINES 0 +#define configMAX_CO_ROUTINE_PRIORITIES 1 + +/* Software timer related definitions. */ +#define configUSE_TIMERS 1 +#define configTIMER_TASK_PRIORITY 3 +#define configTIMER_QUEUE_LENGTH 10 +#define configTIMER_TASK_STACK_DEPTH configMINIMAL_STACK_SIZE + +/* Interrupt nesting behaviour configuration. */ +/* FIXME: these were taken from a cortex M4 example project */ +#define configKERNEL_INTERRUPT_PRIORITY 255 /* Should be set to the lowest interrupt priority */ + +/* !!!! configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to zero !!!! + * See http://www.FreeRTOS.org/RTOS-Cortex-M3-M4.html. */ +#define configMAX_SYSCALL_INTERRUPT_PRIORITY ( 5 << ( 8U - 3U ) ) /* Platform implements 3 priority bits */ +#define configMAC_INTERRUPT_PRIORITY 7 /* Used at ethernet init and shifted by CMSIS call */ + +/* Define to trap errors during development. */ +void vAssertCalled( const char * pcFile, + unsigned long ulLine ); +#define configASSERT( x ) if( ( x ) == 0 ) vAssertCalled( __FILE__, __LINE__ ); + +/* FreeRTOS MPU specific definitions. */ +#define configINCLUDE_APPLICATION_DEFINED_PRIVILEGED_FUNCTIONS 0 +#define configTOTAL_MPU_REGIONS 8 /* Default value. */ +#define configTEX_S_C_B_FLASH 0x07UL /* Default value. */ +#define configTEX_S_C_B_SRAM 0x07UL /* Default value. */ +#define configENFORCE_SYSTEM_CALLS_FROM_KERNEL_ONLY 1 + +/* Optional functions - most linkers will remove unused functions anyway. */ +#define INCLUDE_vTaskPrioritySet 1 +#define INCLUDE_uxTaskPriorityGet 1 +#define INCLUDE_vTaskDelete 1 +#define INCLUDE_vTaskSuspend 1 +#define INCLUDE_xResumeFromISR 1 +#define INCLUDE_vTaskDelayUntil 1 +#define INCLUDE_vTaskDelay 1 +#define INCLUDE_xTaskGetSchedulerState 1 +#define INCLUDE_xTaskGetCurrentTaskHandle 1 +#define INCLUDE_uxTaskGetStackHighWaterMark 1 +#define INCLUDE_xTaskGetIdleTaskHandle 0 +#define INCLUDE_eTaskGetState 1 +#define INCLUDE_xEventGroupSetBitFromISR 1 +#define INCLUDE_xTimerPendFunctionCall 1 +#define INCLUDE_xTaskAbortDelay 1 +#define INCLUDE_xTaskGetHandle 0 +#define INCLUDE_xTaskResumeFromISR 1 +#define INCLUDE_xEventGroupSetBitsFromISR 1 +#define INCLUDE_xSemaphoreGetMutexHolder 1 + +/* Cortex-M specific definitions. */ +#ifdef __NVIC_PRIO_BITS + /* __NVIC_PRIO_BITS will be specified when CMSIS is being used. */ + #define configPRIO_BITS __NVIC_PRIO_BITS +#else + #define configPRIO_BITS 3 /* 8 priority levels. */ +#endif + +/* Logging task definitions. */ +void vLoggingPrintf( const char * pcFormat, + ... ); + +/* Map the FreeRTOS printf() to the logging task printf. */ +#define configPRINTF( x ) vLoggingPrintf x + +/* Map the logging task's printf to the board specific output function. */ +#define configPRINT_STRING( x ) printf( "%s", ( x ) ) + +/* Sets the length of the buffers into which logging messages are written - so + * also defines the maximum length of each log message. */ +#if ( appCONFIG_DEVICE_ADVISOR_TEST_ACTIVE == 1 ) + #define configLOGGING_MAX_MESSAGE_LENGTH 20480 +#else + #define configLOGGING_MAX_MESSAGE_LENGTH 1024 +#endif +/* Prepend each log message with a message number, the task name and a time stamp. */ +#define configLOGGING_INCLUDE_TIME_AND_TASK_NAME 1 + +/* A header file that defines trace macro can be included here. */ + +/* FIXME: this is just to build freeRTOS demo */ + +/* The address of an echo server that will be used by the two demo echo client + * tasks: + * http://www.freertos.org/FreeRTOS-Plus/FreeRTOS_Plus_TCP/TCP_Echo_Clients.html, + * http://www.freertos.org/FreeRTOS-Plus/FreeRTOS_Plus_TCP/UDP_Echo_Clients.html. */ +#define configECHO_SERVER_ADDR0 192 +#define configECHO_SERVER_ADDR1 168 +#define configECHO_SERVER_ADDR2 0 +#define configECHO_SERVER_ADDR3 105 +#define configTCP_ECHO_CLIENT_PORT 45000 + +/* Default MAC address configuration. The demo creates a virtual network + * connection that uses this MAC address by accessing the raw Ethernet/WiFi data + * to and from a real network connection on the host PC. See the + * configNETWORK_INTERFACE_TO_USE definition above for information on how to + * configure the real network connection to use. */ +#define configMAC_ADDR0 0x00 +#define configMAC_ADDR1 0x11 +#define configMAC_ADDR2 0x22 +#define configMAC_ADDR3 0x33 +#define configMAC_ADDR4 0x44 +#define configMAC_ADDR5 0x21 + +/* Default IP address configuration. Used in ipconfigUSE_DHCP is set to 0, or + * ipconfigUSE_DHCP is set to 1 but a DNS server cannot be contacted. */ +#define configIP_ADDR0 192 +#define configIP_ADDR1 168 +#define configIP_ADDR2 0 +#define configIP_ADDR3 105 + +/* Default gateway IP address configuration. Used in ipconfigUSE_DHCP is set to + * 0, or ipconfigUSE_DHCP is set to 1 but a DNS server cannot be contacted. */ +#define configGATEWAY_ADDR0 192 +#define configGATEWAY_ADDR1 168 +#define configGATEWAY_ADDR2 0 +#define configGATEWAY_ADDR3 1 + +/* Default DNS server configuration. OpenDNS addresses are 208.67.222.222 and + * 208.67.220.220. Used in ipconfigUSE_DHCP is set to 0, or ipconfigUSE_DHCP is + * set to 1 but a DNS server cannot be contacted.*/ +#define configDNS_SERVER_ADDR0 208 +#define configDNS_SERVER_ADDR1 67 +#define configDNS_SERVER_ADDR2 222 +#define configDNS_SERVER_ADDR3 222 + +/* Default netmask configuration. Used in ipconfigUSE_DHCP is set to 0, or + * ipconfigUSE_DHCP is set to 1 but a DNS server cannot be contacted. */ +#define configNET_MASK0 255 +#define configNET_MASK1 255 +#define configNET_MASK2 255 +#define configNET_MASK3 0 + +#define democonfigNETWORK_TYPES ( AWSIOT_NETWORK_TYPE_ETH ) + +#endif /* FREERTOS_CONFIG_H */ diff --git a/applications/object_detection/configs/freertos_config/FreeRTOSIPConfig.h b/applications/object_detection/configs/freertos_config/FreeRTOSIPConfig.h new file mode 100644 index 00000000..4e1f9238 --- /dev/null +++ b/applications/object_detection/configs/freertos_config/FreeRTOSIPConfig.h @@ -0,0 +1,316 @@ +/* + * FreeRTOS V202212.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * https://www.FreeRTOS.org + * https://github.com/FreeRTOS + * + */ + + +/***************************************************************************** +* +* See the following URL for configuration information. +* https://www.FreeRTOS.org/FreeRTOS-Plus/FreeRTOS_Plus_TCP/TCP_IP_Configuration.html +* +*****************************************************************************/ + +#ifndef FREERTOS_IP_CONFIG_H +#define FREERTOS_IP_CONFIG_H + +/* Prototype for the function used to print out. In this case it prints to the + * console before the network is connected then a UDP port after the network has + * connected. */ +extern void vLoggingPrintf( const char * pcFormatString, + ... ); + +#ifdef HEAP3 + #define xPortGetMinimumEverFreeHeapSize( x ) 0 + #define xPortGetFreeHeapSize() 0 +#endif + +/* Set to 1 to print out debug messages. If ipconfigHAS_DEBUG_PRINTF is set to + * 1 then FreeRTOS_debug_printf should be defined to the function used to print + * out the debugging messages. */ +#define ipconfigHAS_DEBUG_PRINTF 0 +#if ( ipconfigHAS_DEBUG_PRINTF == 1 ) + #define FreeRTOS_debug_printf( X ) vLoggingPrintf X +#endif + +/* Set to 1 to print out non debugging messages, for example the output of the + * FreeRTOS_netstat() command, and ping replies. If ipconfigHAS_PRINTF is set to 1 + * then FreeRTOS_printf should be set to the function used to print out the + * messages. */ +#define ipconfigHAS_PRINTF 1 +#if ( ipconfigHAS_PRINTF == 1 ) + #define FreeRTOS_printf( X ) vLoggingPrintf X +#endif + +/* Define the byte order of the target MCU (the MCU FreeRTOS+TCP is executing + * on). Valid options are pdFREERTOS_BIG_ENDIAN and pdFREERTOS_LITTLE_ENDIAN. */ +#define ipconfigBYTE_ORDER pdFREERTOS_LITTLE_ENDIAN + +/* If the network card/driver includes checksum offloading (IP/TCP/UDP checksums) + * then set ipconfigDRIVER_INCLUDED_RX_IP_CHECKSUM to 1 to prevent the software + * stack repeating the checksum calculations. */ +#define ipconfigDRIVER_INCLUDED_RX_IP_CHECKSUM 1 + +/* Several API's will block until the result is known, or the action has been + * performed, for example FreeRTOS_send() and FreeRTOS_recv(). The timeouts can be + * set per socket, using setsockopt(). If not set, the times below will be + * used as defaults. */ +#define ipconfigSOCK_DEFAULT_RECEIVE_BLOCK_TIME ( 100 ) +#define ipconfigSOCK_DEFAULT_SEND_BLOCK_TIME ( 100 ) + +/* Include support for LLMNR: Link-local Multicast Name Resolution + * (non-Microsoft) */ +#define ipconfigUSE_LLMNR ( 0 ) + +/* Include support for NBNS: NetBIOS Name Service (Microsoft) */ +#define ipconfigUSE_NBNS ( 0 ) + +/* Include support for DNS caching. For TCP, having a small DNS cache is very + * useful. When a cache is present, ipconfigDNS_REQUEST_ATTEMPTS can be kept low + * and also DNS may use small timeouts. If a DNS reply comes in after the DNS + * socket has been destroyed, the result will be stored into the cache. The next + * call to FreeRTOS_gethostbyname() will return immediately, without even creating + * a socket. */ +#define ipconfigUSE_DNS_CACHE ( 1 ) +#define ipconfigDNS_CACHE_NAME_LENGTH ( 254 ) +#define ipconfigDNS_CACHE_ENTRIES ( 4 ) +#define ipconfigDNS_REQUEST_ATTEMPTS ( 2 ) + +/* The IP stack executes it its own task (although any application task can make + * use of its services through the published sockets API). ipconfigUDP_TASK_PRIORITY + * sets the priority of the task that executes the IP stack. The priority is a + * standard FreeRTOS task priority so can take any value from 0 (the lowest + * priority) to (configMAX_PRIORITIES - 1) (the highest priority). + * configMAX_PRIORITIES is a standard FreeRTOS configuration parameter defined in + * FreeRTOSConfig.h, not FreeRTOSIPConfig.h. Consideration needs to be given as to + * the priority assigned to the task executing the IP stack relative to the + * priority assigned to tasks that use the IP stack. */ +#define ipconfigIP_TASK_PRIORITY ( configMAX_PRIORITIES - 1 ) + +/* The size, in words (not bytes), of the stack allocated to the FreeRTOS+TCP + * task. This setting is less important when the FreeRTOS Win32 simulator is used + * as the Win32 simulator only stores a fixed amount of information on the task + * stack. FreeRTOS includes optional stack overflow detection, see: + * http://www.freertos.org/Stacks-and-stack-overflow-checking.html */ +#define ipconfigIP_TASK_STACK_SIZE_WORDS ( configMINIMAL_STACK_SIZE * 5 ) + +/* ipconfigRAND32() is called by the IP stack to generate random numbers for + * things such as a DHCP transaction number or initial sequence number. Random + * number generation is performed via this macro to allow applications to use their + * own random number generation method. For example, it might be possible to + * generate a random number by sampling noise on an analogue input. */ +/* #define ipconfigRAND32() uxRand() */ + +/* If ipconfigUSE_NETWORK_EVENT_HOOK is set to 1 then FreeRTOS+TCP will call the + * network event hook at the appropriate times. If ipconfigUSE_NETWORK_EVENT_HOOK + * is not set to 1 then the network event hook will never be called. See + * http://www.FreeRTOS.org/FreeRTOS-Plus/FreeRTOS_Plus_UDP/API/vApplicationIPNetworkEventHook.shtml + */ +#define ipconfigUSE_NETWORK_EVENT_HOOK 1 + +/* Sockets have a send block time attribute. If FreeRTOS_sendto() is called but + * a network buffer cannot be obtained then the calling task is held in the Blocked + * state (so other tasks can continue to executed) until either a network buffer + * becomes available or the send block time expires. If the send block time expires + * then the send operation is aborted. The maximum allowable send block time is + * capped to the value set by ipconfigMAX_SEND_BLOCK_TIME_TICKS. Capping the + * maximum allowable send block time prevents prevents a deadlock occurring when + * all the network buffers are in use and the tasks that process (and subsequently + * free) the network buffers are themselves blocked waiting for a network buffer. + * ipconfigMAX_SEND_BLOCK_TIME_TICKS is specified in RTOS ticks. A time in + * milliseconds can be converted to a time in ticks by dividing the time in + * milliseconds by portTICK_PERIOD_MS. */ +#define ipconfigUDP_MAX_SEND_BLOCK_TIME_TICKS ( 5000U ) + +/* If ipconfigUSE_DHCP is 1 then FreeRTOS+TCP will attempt to retrieve an IP + * address, netmask, DNS server address and gateway address from a DHCP server. If + * ipconfigUSE_DHCP is 0 then FreeRTOS+TCP will use a static IP address. The + * stack will revert to using the static IP address even when ipconfigUSE_DHCP is + * set to 1 if a valid configuration cannot be obtained from a DHCP server for any + * reason. The static configuration used is that passed into the stack by the + * FreeRTOS_IPInit() function call. */ +#define ipconfigUSE_DHCP 1 +#define ipconfigUSE_DHCP_HOOK 0 + +/* When ipconfigUSE_DHCP is set to 1, DHCP requests will be sent out at + * increasing time intervals until either a reply is received from a DHCP server + * and accepted, or the interval between transmissions reaches + * ipconfigMAXIMUM_DISCOVER_TX_PERIOD. The IP stack will revert to using the + * static IP address passed as a parameter to FreeRTOS_IPInit() if the + * re-transmission time interval reaches ipconfigMAXIMUM_DISCOVER_TX_PERIOD without + * a DHCP reply being received. */ +#define ipconfigMAXIMUM_DISCOVER_TX_PERIOD ( 120000U ) + +/* The ARP cache is a table that maps IP addresses to MAC addresses. The IP + * stack can only send a UDP message to a remove IP address if it knowns the MAC + * address associated with the IP address, or the MAC address of the router used to + * contact the remote IP address. When a UDP message is received from a remote IP + * address the MAC address and IP address are added to the ARP cache. When a UDP + * message is sent to a remote IP address that does not already appear in the ARP + * cache then the UDP message is replaced by a ARP message that solicits the + * required MAC address information. ipconfigARP_CACHE_ENTRIES defines the maximum + * number of entries that can exist in the ARP table at any one time. */ +#define ipconfigARP_CACHE_ENTRIES 6 + +/* ARP requests that do not result in an ARP response will be re-transmitted a + * maximum of ipconfigMAX_ARP_RETRANSMISSIONS times before the ARP request is + * aborted. */ +#define ipconfigMAX_ARP_RETRANSMISSIONS ( 5 ) + +/* ipconfigMAX_ARP_AGE defines the maximum time between an entry in the ARP + * table being created or refreshed and the entry being removed because it is stale. + * New ARP requests are sent for ARP cache entries that are nearing their maximum + * age. ipconfigMAX_ARP_AGE is specified in tens of seconds, so a value of 150 is + * equal to 1500 seconds (or 25 minutes). */ +#define ipconfigMAX_ARP_AGE 150 + +/* Implementing FreeRTOS_inet_addr() necessitates the use of string handling + * routines, which are relatively large. To save code space the full + * FreeRTOS_inet_addr() implementation is made optional, and a smaller and faster + * alternative called FreeRTOS_inet_addr_quick() is provided. FreeRTOS_inet_addr() + * takes an IP in decimal dot format (for example, "192.168.0.1") as its parameter. + * FreeRTOS_inet_addr_quick() takes an IP address as four separate numerical octets + * (for example, 192, 168, 0, 1) as its parameters. If + * ipconfigINCLUDE_FULL_INET_ADDR is set to 1 then both FreeRTOS_inet_addr() and + * FreeRTOS_indet_addr_quick() are available. If ipconfigINCLUDE_FULL_INET_ADDR is + * not set to 1 then only FreeRTOS_indet_addr_quick() is available. */ +#define ipconfigINCLUDE_FULL_INET_ADDR 1 + +/* ipconfigNUM_NETWORK_BUFFER_DESCRIPTORS defines the total number of network buffer that + * are available to the IP stack. The total number of network buffers is limited + * to ensure the total amount of RAM that can be consumed by the IP stack is capped + * to a pre-determinable value. */ +#define ipconfigNUM_NETWORK_BUFFER_DESCRIPTORS 60 + +/* A FreeRTOS queue is used to send events from application tasks to the IP + * stack. ipconfigEVENT_QUEUE_LENGTH sets the maximum number of events that can + * be queued for processing at any one time. The event queue must be a minimum of + * 5 greater than the total number of network buffers. */ +#define ipconfigEVENT_QUEUE_LENGTH ( ipconfigNUM_NETWORK_BUFFER_DESCRIPTORS + 5 ) + +/* The address of a socket is the combination of its IP address and its port + * number. FreeRTOS_bind() is used to manually allocate a port number to a socket + * (to 'bind' the socket to a port), but manual binding is not normally necessary + * for client sockets (those sockets that initiate outgoing connections rather than + * wait for incoming connections on a known port number). If + * ipconfigALLOW_SOCKET_SEND_WITHOUT_BIND is set to 1 then calling + * FreeRTOS_sendto() on a socket that has not yet been bound will result in the IP + * stack automatically binding the socket to a port number from the range + * socketAUTO_PORT_ALLOCATION_START_NUMBER to 0xffff. If + * ipconfigALLOW_SOCKET_SEND_WITHOUT_BIND is set to 0 then calling FreeRTOS_sendto() + * on a socket that has not yet been bound will result in the send operation being + * aborted. */ +#define ipconfigALLOW_SOCKET_SEND_WITHOUT_BIND 1 + +/* Defines the Time To Live (TTL) values used in outgoing UDP packets. */ +#define ipconfigUDP_TIME_TO_LIVE 128 +#define ipconfigTCP_TIME_TO_LIVE 128 /* also defined in FreeRTOSIPConfigDefaults.h */ + +/* USE_TCP: Use TCP and all its features */ +#define ipconfigUSE_TCP ( 1 ) + +/* USE_WIN: Let TCP use windowing mechanism. */ +#define ipconfigUSE_TCP_WIN ( 1 ) + +/* The MTU is the maximum number of bytes the payload of a network frame can + * contain. For normal Ethernet V2 frames the maximum MTU is 1500. Setting a + * lower value can save RAM, depending on the buffer management scheme used. If + * ipconfigCAN_FRAGMENT_OUTGOING_PACKETS is 1 then (ipconfigNETWORK_MTU - 28) must + * be divisible by 8. */ +#define ipconfigNETWORK_MTU 1500U + +/* Set ipconfigUSE_DNS to 1 to include a basic DNS client/resolver. DNS is used + * through the FreeRTOS_gethostbyname() API function. */ +#define ipconfigUSE_DNS 1 + +/* If ipconfigREPLY_TO_INCOMING_PINGS is set to 1 then the IP stack will + * generate replies to incoming ICMP echo (ping) requests. */ +#define ipconfigREPLY_TO_INCOMING_PINGS 1 + +/* If ipconfigSUPPORT_OUTGOING_PINGS is set to 1 then the + * FreeRTOS_SendPingRequest() API function is available. */ +#define ipconfigSUPPORT_OUTGOING_PINGS 0 + +/* If ipconfigSUPPORT_SELECT_FUNCTION is set to 1 then the FreeRTOS_select() + * (and associated) API function is available. */ +#define ipconfigSUPPORT_SELECT_FUNCTION 0 + +/* If ipconfigFILTER_OUT_NON_ETHERNET_II_FRAMES is set to 1 then Ethernet frames + * that are not in Ethernet II format will be dropped. This option is included for + * potential future IP stack developments. */ +#define ipconfigFILTER_OUT_NON_ETHERNET_II_FRAMES 1 + +/* If ipconfigETHERNET_DRIVER_FILTERS_FRAME_TYPES is set to 1 then it is the + * responsibility of the Ethernet interface to filter out packets that are of no + * interest. If the Ethernet interface does not implement this functionality, then + * set ipconfigETHERNET_DRIVER_FILTERS_FRAME_TYPES to 0 to have the IP stack + * perform the filtering instead (it is much less efficient for the stack to do it + * because the packet will already have been passed into the stack). If the + * Ethernet driver does all the necessary filtering in hardware then software + * filtering can be removed by using a value other than 1 or 0. */ +#define ipconfigETHERNET_DRIVER_FILTERS_FRAME_TYPES 1 + +/* Advanced only: in order to access 32-bit fields in the IP packets with + * 32-bit memory instructions, all packets will be stored 32-bit-aligned, plus 16-bits. + * This has to do with the contents of the IP-packets: all 32-bit fields are + * 32-bit-aligned, plus 16-bit(!) */ +#define ipconfigPACKET_FILLER_SIZE 2U + +/* Define the size of the pool of TCP window descriptors. On the average, each + * TCP socket will use up to 2 x 6 descriptors, meaning that it can have 2 x 6 + * outstanding packets (for Rx and Tx). When using up to 10 TP sockets + * simultaneously, one could define TCP_WIN_SEG_COUNT as 120. */ +#define ipconfigTCP_WIN_SEG_COUNT 240 + +/* Each TCP socket has a circular buffers for Rx and Tx, which have a fixed + * maximum size. Define the size of Rx buffer for TCP sockets. */ +#define ipconfigTCP_RX_BUFFER_LENGTH ( 10000 ) + +/* Define the size of Tx buffer for TCP sockets. */ +#define ipconfigTCP_TX_BUFFER_LENGTH ( 10000 ) + +/* When using call-back handlers, the driver may check if the handler points to + * real program memory (RAM or flash) or just has a random non-zero value. */ +#define ipconfigIS_VALID_PROG_ADDRESS( x ) ( ( x ) != NULL ) + +/* Include support for TCP hang protection. All sockets in a connecting or + * disconnecting stage will timeout after a period of non-activity. */ +#define ipconfigTCP_HANG_PROTECTION ( 1 ) +#define ipconfigTCP_HANG_PROTECTION_TIME ( 30 ) + +/* Include support for TCP keep-alive messages. */ +#define ipconfigTCP_KEEP_ALIVE ( 1 ) +#define ipconfigTCP_KEEP_ALIVE_INTERVAL ( 20 ) /* in seconds */ + +/* Include all API's and code that is needed for the IPv4 protocol. + * When defined as zero, the application should uses IPv6. */ +#define ipconfigUSE_IPv4 ( 1 ) + +/* + * If defined this macro enables the APIs that are backward compatible + * with single end point IPv4 version of the FreeRTOS+TCP library. + */ +#define ipconfigIPv4_BACKWARD_COMPATIBLE 0 + +#endif /* FREERTOS_IP_CONFIG_H */ diff --git a/applications/object_detection/configs/isp_config/CMakeLists.txt b/applications/object_detection/configs/isp_config/CMakeLists.txt new file mode 100644 index 00000000..f3e94e3e --- /dev/null +++ b/applications/object_detection/configs/isp_config/CMakeLists.txt @@ -0,0 +1,15 @@ +# Copyright 2023-2024 Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +add_library(isp_firmware_config INTERFACE) + +target_include_directories(isp_firmware_config + INTERFACE + . +) + +target_link_libraries(isp_firmware_config + INTERFACE + arm-corstone-platform-bsp +) diff --git a/applications/object_detection/configs/isp_config/acamera_firmware_config.h b/applications/object_detection/configs/isp_config/acamera_firmware_config.h new file mode 100644 index 00000000..db616f84 --- /dev/null +++ b/applications/object_detection/configs/isp_config/acamera_firmware_config.h @@ -0,0 +1,238 @@ +/* + * + * SPDX-License-Identifier: BSD-3-Clause + * + * Copyright (c) 2024, Arm Limited. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without modification, + * are permitted provided that the following conditions are met: + * - Redistributions of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * - Redistributions in binary form must reproduce the above copyright notice, this + * list of conditions and the following disclaimer in the documentation and/or + * other materials provided with the distribution. + * - Neither the name of ARM nor the names of its contributors may be used to + * endorse or promote products derived from this software without specific prior + * written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR + * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON + * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef __ACAMERA_FIRMWARE_CONFIG_H__ +#define __ACAMERA_FIRMWARE_CONFIG_H__ + +#include "platform_base_address.h" + +/* Autogenerated firmware settings. */ +#define ACAMERA_EVENT_QUEUE_SIZE 256 +#define ACAMERA_FW_SHP_DETECTED_MAXIMUM 1023 +#define ACAMERA_IRQ_AE_STATS 3 +#define ACAMERA_IRQ_AF2_STATS 1 +#define ACAMERA_IRQ_ANTIFOG_HIST 14 +#define ACAMERA_IRQ_AWB_STATS 4 +#define ACAMERA_IRQ_DFE_FRAME_END 17 +#define ACAMERA_IRQ_FRAME_END 0 +#define ACAMERA_IRQ_FRAME_START 7 +#define ACAMERA_IRQ_FRAME_WRITER_DS 6 +#define ACAMERA_IRQ_FRAME_WRITER_FR 5 +#define ACAMERA_ISP_PROFILING 0 +#define ACAMERA_ISP_PROFILING_INIT 0 +#define AE_CENTER_ZONES 65535 +#define AF_SPOT_COUNT_X 5 +#define AF_SPOT_COUNT_Y 5 +#define ANALOG_GAIN_ACCURACY ( 1 << ( LOG2_GAIN_SHIFT - 2 ) ) +#define AWB_BG_MAX_GAIN 0 +#define BLACK_LEVEL_SHIFT_DG 8 +#define BLACK_LEVEL_SHIFT_WB 8 +#define CALIBRATION_IRIDIX_AVG_COEF_INIT 0 +#define CM_SATURATION_TARGET 128 +#define CONNECTION_BUFFER_SIZE ( 26 * 1024 ) +#define CONNECTION_IN_THREAD 1 +#define DMA_WRITER_DEFAULT_BUFFER_NO 4 +#define EXPOSURE_DRIVES_LONG_INTEGRATION_TIME 0 +#define FILTER_LONG_INT_TIME 0 +#define FILTER_SHORT_INT_TIME 0 +#define FIRMWARE_CONTEXT_NUMBER 1 +#define FPGA_HAS_HDMI 0 +#define FSM_NAMES \ + { \ + "GENERIC", "SENSOR", "CMOS", "GENERAL", "AE_ACAMERA", "AWB_ACAMERA", "AF_ACAMERA", "COLOR_MATRIX", \ + "PURPLE_FRINGE", "IRIDIX_ACAMERA", "NOISE_REDUCTION", "CROP", "SHARPENING", "MATRIX_YUV", "GAMMA_ACAMERA", \ + "DMA_WRITER", "MONITOR" \ + } +#define FW_ALG_SUPPORT_SHARED_LIB 0 +#define FW_DO_INITIALIZATION 1 +#define FW_DS1_OUTPUT_FORMAT_PIPE PIPE_OUT_RGB +#define FW_EVT_QUEUE_TIMEOUT_MS 100 +#define FW_FR_OUTPUT_FORMAT_PIPE PIPE_OUT_RGB +#define FW_HAS_CONTROL_CHANNEL 0 +#define FW_HAS_CUSTOM_SETTINGS 1 +#define FW_INPUT_FORMAT DMA_FORMAT_RAW12 +#define FW_LOG_FROM_ISR 0 +#define FW_LOG_HAS_SRC 0 +#define FW_LOG_HAS_TIME 1 +#define FW_LOG_LEVEL LOG_NOTICE +#define FW_LOG_MASK 4294967295 +#define FW_LOG_REAL_TIME 0 +#define FW_OUTPUT_FORMAT DMA_FORMAT_A2R10G10B10 +#define FW_OUTPUT_FORMAT_SECONDARY DMA_FORMAT_DISABLE +#define FW_USE_SYSTEM_DMA 0 +#define FW_ZONE_AE 0 +#define HDMI_TDA19988 1 +#define IRIDIX_BYPASS_MAX_STR_CLIP 0 +#define IRIDIX_HAS_PRE_POST_GAMMA_LUT_LINEAR 0 +#define IRIDIX_STRENGTH_BY_EXP_RATIO 0 +#define IRIDIX_STRENGTH_TARGET 128 +#define ISP_BINARY_SEQUENCE 0 +#define ISP_CONTROLS_DMA_READER 1 +#define ISP_DEFAULT_AF_ZONES_HOR 15 +#define ISP_DEFAULT_AF_ZONES_VERT 15 +#define ISP_DISPLAY_MODE 1080 +#define ISP_DMA_RAW_CAPTURE 0 +#define ISP_FULL_HISTOGRAM_SIZE 1024 +#define ISP_FW_BUILD 1 +#define ISP_GAMMA_LUT_SIZE 129 +#define ISP_HAS_AE_ACAMERA_FSM 1 +#define ISP_HAS_AF 1 +#define ISP_HAS_AF_ACAMERA_FSM 1 +#define ISP_HAS_AWB_ACAMERA_FSM 1 +#define ISP_HAS_CMOS_FSM 1 +#define ISP_HAS_COLOR_MATRIX_FSM 1 +#define ISP_HAS_CONNECTION_BUFFER 0 +#define ISP_HAS_CONNECTION_CHARDEV 0 +#define ISP_HAS_CONNECTION_DEBUG 0 +#define ISP_HAS_CONNECTION_SOCKET 0 +#define ISP_HAS_CONNECTION_SOCKET_PORT 0 +#define ISP_HAS_CONNECTION_UART 1 +#define ISP_HAS_CROP_FSM 1 +#define ISP_HAS_DMA_WRITER_FSM 1 +#define ISP_HAS_DS1 1 +#define ISP_HAS_FPGA_WRAPPER 0 +#define ISP_HAS_GAMMA_ACAMERA_FSM 1 +#define ISP_HAS_GENERAL_FSM 1 +#define ISP_HAS_IRIDIX_ACAMERA_FSM 1 +#define ISP_HAS_MATRIX_YUV_FSM 1 +#define ISP_HAS_META_CB 0 +#define ISP_HAS_MONITOR_FSM 1 +#define ISP_HAS_NOISE_REDUCTION_FSM 1 +#define ISP_HAS_PURPLE_FRINGE_FSM 1 +#define ISP_HAS_RAW_CB 0 +#define ISP_HAS_SENSOR_FSM 1 +#define ISP_HAS_SHARPENING_FSM 1 +#define ISP_HAS_WDR_FRAME_BUFFER 0 +#define ISP_HW_IS_MALI_C32_C52 0 +#define ISP_INPUT_BITS 20 +#define ISP_IRQ_DISABLE_ALL_IRQ 4294967295 +#define ISP_IRQ_MASK_VECTOR 4289200114 +#define ISP_MAX_CALIBRATION_DATA_SIZE ( 128 * 1024 ) +#define ISP_MAX_SENSOR_MODES 16 +#define ISP_SENSOR_DRIVER_AD5821 0 +#define ISP_SENSOR_DRIVER_AN41908A 0 +#define ISP_SENSOR_DRIVER_BU64295 0 +#define ISP_SENSOR_DRIVER_BU64748 0 +#define ISP_SENSOR_DRIVER_DONGWOON 0 +#define ISP_SENSOR_DRIVER_DW9800 0 +#define ISP_SENSOR_DRIVER_FP5510A 0 +#define ISP_SENSOR_DRIVER_LC898201 0 +#define ISP_SENSOR_DRIVER_MODEL 0 +#define ISP_SENSOR_DRIVER_NULL 1 +#define ISP_SENSOR_DRIVER_ROHM 0 +#define ISP_SENSOR_DRIVER_V4L2 0 +#define ISP_SOC_DMA_BUS_OFFSET 0 +#define ISP_SOC_START_ADDR ISP_BASE_NS +#define ISP_VCLK_FREQ_MHZ 100 +#define ISP_WDR_DEFAULT_MODE WDR_MODE_LINEAR +#define ISP_WDR_SWITCH 1 +#define KERNEL_MODULE 0 +#define LOG2_GAIN_SHIFT 18 +#define LOG_MODULE_AE_ACAMERA 4 +#define LOG_MODULE_AE_ACAMERA_MASK 16 +#define LOG_MODULE_AF_ACAMERA 6 +#define LOG_MODULE_AF_ACAMERA_MASK 64 +#define LOG_MODULE_ALL 131071 +#define LOG_MODULE_AWB_ACAMERA 5 +#define LOG_MODULE_AWB_ACAMERA_MASK 32 +#define LOG_MODULE_CMOS 2 +#define LOG_MODULE_CMOS_MASK 4 +#define LOG_MODULE_COLOR_MATRIX 7 +#define LOG_MODULE_COLOR_MATRIX_MASK 128 +#define LOG_MODULE_CROP 11 +#define LOG_MODULE_CROP_MASK 2048 +#define LOG_MODULE_DMA_WRITER 15 +#define LOG_MODULE_DMA_WRITER_MASK 32768 +#define LOG_MODULE_GAMMA_ACAMERA 14 +#define LOG_MODULE_GAMMA_ACAMERA_MASK 16384 +#define LOG_MODULE_GENERAL 3 +#define LOG_MODULE_GENERAL_MASK 8 +#define LOG_MODULE_GENERIC 0 +#define LOG_MODULE_GENERIC_MASK 1 +#define LOG_MODULE_IRIDIX_ACAMERA 9 +#define LOG_MODULE_IRIDIX_ACAMERA_MASK 512 +#define LOG_MODULE_MATRIX_YUV 13 +#define LOG_MODULE_MATRIX_YUV_MASK 8192 +#define LOG_MODULE_MAX 17 +#define LOG_MODULE_MONITOR 16 +#define LOG_MODULE_MONITOR_MASK 65536 +#define LOG_MODULE_NOISE_REDUCTION 10 +#define LOG_MODULE_NOISE_REDUCTION_MASK 1024 +#define LOG_MODULE_PURPLE_FRINGE 8 +#define LOG_MODULE_PURPLE_FRINGE_MASK 256 +#define LOG_MODULE_SENSOR 1 +#define LOG_MODULE_SENSOR_MASK 2 +#define LOG_MODULE_SHARPENING 12 +#define LOG_MODULE_SHARPENING_MASK 4096 +#define MAX_REG_TRACK_ENTRIES 600 +#define OVEREXPOSE_TO_KEEP_ANTIFLICKER 0 +#define SENSOR_BINARY_SEQUENCE 0 +#define SENSOR_DEFAULT_EXP_NUM 1 +#define SENSOR_DEFAULT_PRESET_MODE 0 +#define SENSOR_HAS_FLASH 0 +#define SENSOR_HW_INTERFACE ACameraDefault +#define SENSOR_MASTER_CLOCK 27 +#define SENSOR_SUPPORT_OTP 0 +#define SYSTEM_AE_COMPENSATION_DEFAULT 128 +#define SYSTEM_ANTI_FLICKER_FREQUENCY_DEFAULT 50 +#define SYSTEM_EXPOSURE_PARTITION_VALUE_COUNT 10 +#define SYSTEM_EXPOSURE_RATIO_DEFAULT 8 +#define SYSTEM_MANUAL_EXPOSURE_RATIO_DEFAULT 0 +#define SYSTEM_MAXIMUM_IRIDIX_STRENGTH_DEFAULT 255 +#define SYSTEM_MINIMUM_IRIDIX_STRENGTH_DEFAULT 0 +#define USER_MODULE 0 +#define V4L2_INTERFACE_BUILD 0 +#define V4L2_SOC_SUBDEV_ENABLE 0 + +/* Autogenerated ISP core settings. DO NOT EDIT. */ +#define ISP_CORE_ID 0x58F8E91F +#define ISP_CONFIG_PING 1 +#define ISP_CONFIG_PONG 0 +#define ISP_CONFIG_LUT_OFFSET 0x054A8 +#define ISP_CONFIG_PING_OFFSET 0x18E88 +#define ISP_CONFIG_PING_SIZE 0x17FC0 +#define ISP_METERING_ZONES_MAX_V 15 +#define ISP_METERING_ZONES_MAX_H 15 +#define ISP_METERING_OFFSET_AWB 464 +#define ISP_METERING_OFFSET_AF 928 +#define ISP_INTERRUPT_EVENT_NONES_COUNT 16 +#define ISP_INTERRUPT_EVENT_ISP_START_FRAME_START 0 +#define ISP_INTERRUPT_EVENT_ISP_END_FRAME_END 1 +#define ISP_INTERRUPT_EVENT_METERING_AF 4 +#define ISP_INTERRUPT_EVENT_METERING_AEXP 5 +#define ISP_INTERRUPT_EVENT_METERING_AWB 6 +#define ISP_INTERRUPT_EVENT_ANTIFOG_HIST 8 +#define ISP_INTERRUPT_EVENT_BROKEN_FRAME 3 +#define ISP_INTERRUPT_EVENT_MULTICTX_ERROR 2 +#define ISP_INTERRUPT_EVENT_DMA_ERROR 22 +#define ISP_INTERRUPT_EVENT_WATCHDOG_EXP 19 +#define ISP_INTERRUPT_EVENT_FRAME_COLLISION 20 + +#endif /* __ACAMERA_FIRMWARE_CONFIG_H__*/ diff --git a/applications/object_detection/configs/isp_config/acamera_interface_config.h b/applications/object_detection/configs/isp_config/acamera_interface_config.h new file mode 100644 index 00000000..3934dea4 --- /dev/null +++ b/applications/object_detection/configs/isp_config/acamera_interface_config.h @@ -0,0 +1,49 @@ +/* + * + * SPDX-License-Identifier: BSD-3-Clause + * + * Copyright (c) 2024, Arm Limited. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without modification, + * are permitted provided that the following conditions are met: + * - Redistributions of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * - Redistributions in binary form must reproduce the above copyright notice, this + * list of conditions and the following disclaimer in the documentation and/or + * other materials provided with the distribution. + * - Neither the name of ARM nor the names of its contributors may be used to + * endorse or promote products derived from this software without specific prior + * written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR + * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON + * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + */ + +#ifndef __ACAMERA_INTERFACE_CONFIG_H__ +#define __ACAMERA_INTERFACE_CONFIG_H__ + +extern volatile uint32_t STREAM_ENABLED; + +#if 0 + #define FVP_SENSOR_WIDTH 1920 + #define FVP_SENSOR_HEIGHT 1080 +#else + #define FVP_SENSOR_WIDTH 576 + #define FVP_SENSOR_HEIGHT 576 +#endif +/* 0:RGGB 2:GBRG */ +#define FVP_SENSOR_RGGB_START 2 +#define FVP_SENSOR_DEPTH 12 +#define FVP_SENSOR_TWIDTH FVP_SENSOR_WIDTH +#define FVP_SENSOR_THEIGHT FVP_SENSOR_HEIGHT + +#endif /* ifndef __ACAMERA_INTERFACE_CONFIG_H__ */ diff --git a/applications/object_detection/configs/mbedtls_config/CMakeLists.txt b/applications/object_detection/configs/mbedtls_config/CMakeLists.txt new file mode 100644 index 00000000..a3d55dd6 --- /dev/null +++ b/applications/object_detection/configs/mbedtls_config/CMakeLists.txt @@ -0,0 +1,18 @@ +# Copyright 2023-2024, Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +target_include_directories(mbedtls-config + INTERFACE + . +) + +target_compile_definitions(mbedtls-config + INTERFACE + MBEDTLS_CONFIG_FILE="aws_mbedtls_config.h" +) + +target_link_libraries(mbedtls-config + INTERFACE + freertos_kernel +) diff --git a/applications/object_detection/configs/mbedtls_config/aws_mbedtls_config.h b/applications/object_detection/configs/mbedtls_config/aws_mbedtls_config.h new file mode 100644 index 00000000..d5be8045 --- /dev/null +++ b/applications/object_detection/configs/mbedtls_config/aws_mbedtls_config.h @@ -0,0 +1,3274 @@ +/** + * \file mbedtls_config.h + * + * \brief Configuration options (set of defines) + * + * This set of compile-time options may be used to enable + * or disable features selectively, and reduce the global + * memory footprint. + */ + +/* + * Copyright The Mbed TLS Contributors + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +/** + * This is an optional version symbol that enables compatibility handling of + * config files. + * + * It is equal to the #MBEDTLS_VERSION_NUMBER of the Mbed TLS version that + * introduced the config format we want to be compatible with. + */ +/*#define MBEDTLS_CONFIG_VERSION 0x03000000 */ + +/** + * \name SECTION: System support + * + * This section sets system specific settings. + * \{ + */ + +/** + * \def MBEDTLS_HAVE_ASM + * + * The compiler has support for asm(). + * + * Requires support for asm() in compiler. + * + * Used in: + * library/aria.c + * library/bn_mul.h + * + * Required by: + * MBEDTLS_AESNI_C + * MBEDTLS_PADLOCK_C + * + * Comment to disable the use of assembly code. + */ +#define MBEDTLS_HAVE_ASM + +/** + * \def MBEDTLS_NO_UDBL_DIVISION + * + * The platform lacks support for double-width integer division (64-bit + * division on a 32-bit platform, 128-bit division on a 64-bit platform). + * + * Used in: + * include/mbedtls/bignum.h + * library/bignum.c + * + * The bignum code uses double-width division to speed up some operations. + * Double-width division is often implemented in software that needs to + * be linked with the program. The presence of a double-width integer + * type is usually detected automatically through preprocessor macros, + * but the automatic detection cannot know whether the code needs to + * and can be linked with an implementation of division for that type. + * By default division is assumed to be usable if the type is present. + * Uncomment this option to prevent the use of double-width division. + * + * Note that division for the native integer type is always required. + * Furthermore, a 64-bit type is always required even on a 32-bit + * platform, but it need not support multiplication or division. In some + * cases it is also desirable to disable some double-width operations. For + * example, if double-width division is implemented in software, disabling + * it can reduce code size in some embedded targets. + */ +/*#define MBEDTLS_NO_UDBL_DIVISION */ + +/** + * \def MBEDTLS_NO_64BIT_MULTIPLICATION + * + * The platform lacks support for 32x32 -> 64-bit multiplication. + * + * Used in: + * library/poly1305.c + * + * Some parts of the library may use multiplication of two unsigned 32-bit + * operands with a 64-bit result in order to speed up computations. On some + * platforms, this is not available in hardware and has to be implemented in + * software, usually in a library provided by the toolchain. + * + * Sometimes it is not desirable to have to link to that library. This option + * removes the dependency of that library on platforms that lack a hardware + * 64-bit multiplier by embedding a software implementation in Mbed TLS. + * + * Note that depending on the compiler, this may decrease performance compared + * to using the library function provided by the toolchain. + */ +/*#define MBEDTLS_NO_64BIT_MULTIPLICATION */ + +/** + * \def MBEDTLS_HAVE_SSE2 + * + * CPU supports SSE2 instruction set. + * + * Uncomment if the CPU supports SSE2 (IA-32 specific). + */ +/*#define MBEDTLS_HAVE_SSE2 */ + +/** + * \def MBEDTLS_HAVE_TIME + * + * System has time.h and time(). + * The time does not need to be correct, only time differences are used, + * by contrast with MBEDTLS_HAVE_TIME_DATE + * + * Defining MBEDTLS_HAVE_TIME allows you to specify MBEDTLS_PLATFORM_TIME_ALT, + * MBEDTLS_PLATFORM_TIME_MACRO, MBEDTLS_PLATFORM_TIME_TYPE_MACRO and + * MBEDTLS_PLATFORM_STD_TIME. + * + * Comment if your system does not support time functions + */ +/*#define MBEDTLS_HAVE_TIME */ + +/** + * \def MBEDTLS_HAVE_TIME_DATE + * + * System has time.h, time(), and an implementation for + * mbedtls_platform_gmtime_r() (see below). + * The time needs to be correct (not necessarily very accurate, but at least + * the date should be correct). This is used to verify the validity period of + * X.509 certificates. + * + * Comment if your system does not have a correct clock. + * + * \note mbedtls_platform_gmtime_r() is an abstraction in platform_util.h that + * behaves similarly to the gmtime_r() function from the C standard. Refer to + * the documentation for mbedtls_platform_gmtime_r() for more information. + * + * \note It is possible to configure an implementation for + * mbedtls_platform_gmtime_r() at compile-time by using the macro + * MBEDTLS_PLATFORM_GMTIME_R_ALT. + */ +/*#define MBEDTLS_HAVE_TIME_DATE */ + +/** + * \def MBEDTLS_PLATFORM_MEMORY + * + * Enable the memory allocation layer. + * + * By default mbed TLS uses the system-provided calloc() and free(). + * This allows different allocators (self-implemented or provided) to be + * provided to the platform abstraction layer. + * + * Enabling MBEDTLS_PLATFORM_MEMORY without the + * MBEDTLS_PLATFORM_{FREE,CALLOC}_MACROs will provide + * "mbedtls_platform_set_calloc_free()" allowing you to set an alternative calloc() and + * free() function pointer at runtime. + * + * Enabling MBEDTLS_PLATFORM_MEMORY and specifying + * MBEDTLS_PLATFORM_{CALLOC,FREE}_MACROs will allow you to specify the + * alternate function at compile time. + * + * Requires: MBEDTLS_PLATFORM_C + * + * Enable this layer to allow use of alternative memory allocators. + */ +#define MBEDTLS_PLATFORM_MEMORY + +#include +void * mbedtls_platform_calloc( size_t nmemb, + size_t size ); +void mbedtls_platform_free( void * ptr ); + +/** + * \def MBEDTLS_PLATFORM_NO_STD_FUNCTIONS + * + * Do not assign standard functions in the platform layer (e.g. calloc() to + * MBEDTLS_PLATFORM_STD_CALLOC and printf() to MBEDTLS_PLATFORM_STD_PRINTF) + * + * This makes sure there are no linking errors on platforms that do not support + * these functions. You will HAVE to provide alternatives, either at runtime + * via the platform_set_xxx() functions or at compile time by setting + * the MBEDTLS_PLATFORM_STD_XXX defines, or enabling a + * MBEDTLS_PLATFORM_XXX_MACRO. + * + * Requires: MBEDTLS_PLATFORM_C + * + * Uncomment to prevent default assignment of standard functions in the + * platform layer. + */ +/*#define MBEDTLS_PLATFORM_NO_STD_FUNCTIONS */ + +/** + * \def MBEDTLS_PLATFORM_EXIT_ALT + * + * MBEDTLS_PLATFORM_XXX_ALT: Uncomment a macro to let mbed TLS support the + * function in the platform abstraction layer. + * + * Example: In case you uncomment MBEDTLS_PLATFORM_PRINTF_ALT, mbed TLS will + * provide a function "mbedtls_platform_set_printf()" that allows you to set an + * alternative printf function pointer. + * + * All these define require MBEDTLS_PLATFORM_C to be defined! + * + * \note MBEDTLS_PLATFORM_SNPRINTF_ALT is required on Windows; + * it will be enabled automatically by check_config.h + * + * \warning MBEDTLS_PLATFORM_XXX_ALT cannot be defined at the same time as + * MBEDTLS_PLATFORM_XXX_MACRO! + * + * Requires: MBEDTLS_PLATFORM_TIME_ALT requires MBEDTLS_HAVE_TIME + * + * Uncomment a macro to enable alternate implementation of specific base + * platform function + */ +/*#define MBEDTLS_PLATFORM_EXIT_ALT */ +/*#define MBEDTLS_PLATFORM_TIME_ALT */ +/*#define MBEDTLS_PLATFORM_FPRINTF_ALT */ +/*#define MBEDTLS_PLATFORM_PRINTF_ALT */ +/*#define MBEDTLS_PLATFORM_SNPRINTF_ALT */ +/*#define MBEDTLS_PLATFORM_VSNPRINTF_ALT */ +/*#define MBEDTLS_PLATFORM_NV_SEED_ALT */ +/*#define MBEDTLS_PLATFORM_SETUP_TEARDOWN_ALT */ + +/** + * \def MBEDTLS_DEPRECATED_WARNING + * + * Mark deprecated functions and features so that they generate a warning if + * used. Functionality deprecated in one version will usually be removed in the + * next version. You can enable this to help you prepare the transition to a + * new major version by making sure your code is not using this functionality. + * + * This only works with GCC and Clang. With other compilers, you may want to + * use MBEDTLS_DEPRECATED_REMOVED + * + * Uncomment to get warnings on using deprecated functions and features. + */ +#define MBEDTLS_DEPRECATED_WARNING + +/** + * \def MBEDTLS_DEPRECATED_REMOVED + * + * Remove deprecated functions and features so that they generate an error if + * used. Functionality deprecated in one version will usually be removed in the + * next version. You can enable this to help you prepare the transition to a + * new major version by making sure your code is not using this functionality. + * + * Uncomment to get errors on using deprecated functions and features. + */ +/*#define MBEDTLS_DEPRECATED_REMOVED */ + +/* \} name SECTION: System support */ + +/** + * \name SECTION: mbed TLS feature support + * + * This section sets support for features that are or are not needed + * within the modules that are enabled. + * \{ + */ + +/** + * \def MBEDTLS_TIMING_ALT + * + * Uncomment to provide your own alternate implementation for + * mbedtls_timing_get_timer(), mbedtls_set_alarm(), mbedtls_set/get_delay() + * + * Only works if you have MBEDTLS_TIMING_C enabled. + * + * You will need to provide a header "timing_alt.h" and an implementation at + * compile time. + */ +/*#define MBEDTLS_TIMING_ALT */ + +/** + * \def MBEDTLS_AES_ALT + * + * MBEDTLS__MODULE_NAME__ALT: Uncomment a macro to let mbed TLS use your + * alternate core implementation of a symmetric crypto, an arithmetic or hash + * module (e.g. platform specific assembly optimized implementations). Keep + * in mind that the function prototypes should remain the same. + * + * This replaces the whole module. If you only want to replace one of the + * functions, use one of the MBEDTLS__FUNCTION_NAME__ALT flags. + * + * Example: In case you uncomment MBEDTLS_AES_ALT, mbed TLS will no longer + * provide the "struct mbedtls_aes_context" definition and omit the base + * function declarations and implementations. "aes_alt.h" will be included from + * "aes.h" to include the new function definitions. + * + * Uncomment a macro to enable alternate implementation of the corresponding + * module. + * + * \warning MD5, DES and SHA-1 are considered weak and their + * use constitutes a security risk. If possible, we recommend + * avoiding dependencies on them, and considering stronger message + * digests and ciphers instead. + * + */ +/*#define MBEDTLS_AES_ALT */ +/*#define MBEDTLS_ARIA_ALT */ +/*#define MBEDTLS_CAMELLIA_ALT */ +/*#define MBEDTLS_CCM_ALT */ +/*#define MBEDTLS_CHACHA20_ALT */ +/*#define MBEDTLS_CHACHAPOLY_ALT */ +/*#define MBEDTLS_CMAC_ALT */ +/*#define MBEDTLS_DES_ALT */ +/*#define MBEDTLS_DHM_ALT */ +/*#define MBEDTLS_ECJPAKE_ALT */ +/*#define MBEDTLS_GCM_ALT */ +/*#define MBEDTLS_NIST_KW_ALT */ +/*#define MBEDTLS_MD5_ALT */ +/*#define MBEDTLS_POLY1305_ALT */ +/*#define MBEDTLS_RIPEMD160_ALT */ +/*#define MBEDTLS_RSA_ALT */ +/*#define MBEDTLS_SHA1_ALT */ +/*#define MBEDTLS_SHA256_ALT */ +/*#define MBEDTLS_SHA512_ALT */ + +/* + * When replacing the elliptic curve module, pleace consider, that it is + * implemented with two .c files: + * - ecp.c + * - ecp_curves.c + * You can replace them very much like all the other MBEDTLS__MODULE_NAME__ALT + * macros as described above. The only difference is that you have to make sure + * that you provide functionality for both .c files. + */ +/*#define MBEDTLS_ECP_ALT */ + +/** + * \def MBEDTLS_SHA256_PROCESS_ALT + * + * MBEDTLS__FUNCTION_NAME__ALT: Uncomment a macro to let mbed TLS use you + * alternate core implementation of symmetric crypto or hash function. Keep in + * mind that function prototypes should remain the same. + * + * This replaces only one function. The header file from mbed TLS is still + * used, in contrast to the MBEDTLS__MODULE_NAME__ALT flags. + * + * Example: In case you uncomment MBEDTLS_SHA256_PROCESS_ALT, mbed TLS will + * no longer provide the mbedtls_sha1_process() function, but it will still provide + * the other function (using your mbedtls_sha1_process() function) and the definition + * of mbedtls_sha1_context, so your implementation of mbedtls_sha1_process must be compatible + * with this definition. + * + * \note If you use the AES_xxx_ALT macros, then it is recommended to also set + * MBEDTLS_AES_ROM_TABLES in order to help the linker garbage-collect the AES + * tables. + * + * Uncomment a macro to enable alternate implementation of the corresponding + * function. + * + * \warning MD5, DES and SHA-1 are considered weak and their use + * constitutes a security risk. If possible, we recommend avoiding + * dependencies on them, and considering stronger message digests + * and ciphers instead. + * + * \warning If both MBEDTLS_ECDSA_SIGN_ALT and MBEDTLS_ECDSA_DETERMINISTIC are + * enabled, then the deterministic ECDH signature functions pass the + * the static HMAC-DRBG as RNG to mbedtls_ecdsa_sign(). Therefore + * alternative implementations should use the RNG only for generating + * the ephemeral key and nothing else. If this is not possible, then + * MBEDTLS_ECDSA_DETERMINISTIC should be disabled and an alternative + * implementation should be provided for mbedtls_ecdsa_sign_det_ext(). + * + */ +/*#define MBEDTLS_MD5_PROCESS_ALT */ +/*#define MBEDTLS_RIPEMD160_PROCESS_ALT */ +/*#define MBEDTLS_SHA1_PROCESS_ALT */ +/*#define MBEDTLS_SHA256_PROCESS_ALT */ +/*#define MBEDTLS_SHA512_PROCESS_ALT */ +/*#define MBEDTLS_DES_SETKEY_ALT */ +/*#define MBEDTLS_DES_CRYPT_ECB_ALT */ +/*#define MBEDTLS_DES3_CRYPT_ECB_ALT */ +/*#define MBEDTLS_AES_SETKEY_ENC_ALT */ +/*#define MBEDTLS_AES_SETKEY_DEC_ALT */ +/*#define MBEDTLS_AES_ENCRYPT_ALT */ +/*#define MBEDTLS_AES_DECRYPT_ALT */ +/*#define MBEDTLS_ECDH_GEN_PUBLIC_ALT */ +/*#define MBEDTLS_ECDH_COMPUTE_SHARED_ALT */ +/*#define MBEDTLS_ECDSA_VERIFY_ALT */ +/*#define MBEDTLS_ECDSA_SIGN_ALT */ +/*#define MBEDTLS_ECDSA_GENKEY_ALT */ + +/** + * \def MBEDTLS_ECP_INTERNAL_ALT + * + * Expose a part of the internal interface of the Elliptic Curve Point module. + * + * MBEDTLS_ECP__FUNCTION_NAME__ALT: Uncomment a macro to let mbed TLS use your + * alternative core implementation of elliptic curve arithmetic. Keep in mind + * that function prototypes should remain the same. + * + * This partially replaces one function. The header file from mbed TLS is still + * used, in contrast to the MBEDTLS_ECP_ALT flag. The original implementation + * is still present and it is used for group structures not supported by the + * alternative. + * + * The original implementation can in addition be removed by setting the + * MBEDTLS_ECP_NO_FALLBACK option, in which case any function for which the + * corresponding MBEDTLS_ECP__FUNCTION_NAME__ALT macro is defined will not be + * able to fallback to curves not supported by the alternative implementation. + * + * Any of these options become available by defining MBEDTLS_ECP_INTERNAL_ALT + * and implementing the following functions: + * unsigned char mbedtls_internal_ecp_grp_capable( + * const mbedtls_ecp_group *grp ) + * int mbedtls_internal_ecp_init( const mbedtls_ecp_group *grp ) + * void mbedtls_internal_ecp_free( const mbedtls_ecp_group *grp ) + * The mbedtls_internal_ecp_grp_capable function should return 1 if the + * replacement functions implement arithmetic for the given group and 0 + * otherwise. + * The functions mbedtls_internal_ecp_init and mbedtls_internal_ecp_free are + * called before and after each point operation and provide an opportunity to + * implement optimized set up and tear down instructions. + * + * Example: In case you set MBEDTLS_ECP_INTERNAL_ALT and + * MBEDTLS_ECP_DOUBLE_JAC_ALT, mbed TLS will still provide the ecp_double_jac() + * function, but will use your mbedtls_internal_ecp_double_jac() if the group + * for the operation is supported by your implementation (i.e. your + * mbedtls_internal_ecp_grp_capable() function returns 1 for this group). If the + * group is not supported by your implementation, then the original mbed TLS + * implementation of ecp_double_jac() is used instead, unless this fallback + * behaviour is disabled by setting MBEDTLS_ECP_NO_FALLBACK (in which case + * ecp_double_jac() will return MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE). + * + * The function prototypes and the definition of mbedtls_ecp_group and + * mbedtls_ecp_point will not change based on MBEDTLS_ECP_INTERNAL_ALT, so your + * implementation of mbedtls_internal_ecp__function_name__ must be compatible + * with their definitions. + * + * Uncomment a macro to enable alternate implementation of the corresponding + * function. + */ +/* Required for all the functions in this section */ +/*#define MBEDTLS_ECP_INTERNAL_ALT */ +/* Turn off software fallback for curves not supported in hardware */ +/*#define MBEDTLS_ECP_NO_FALLBACK */ +/* Support for Weierstrass curves with Jacobi representation */ +/*#define MBEDTLS_ECP_RANDOMIZE_JAC_ALT */ +/*#define MBEDTLS_ECP_ADD_MIXED_ALT */ +/*#define MBEDTLS_ECP_DOUBLE_JAC_ALT */ +/*#define MBEDTLS_ECP_NORMALIZE_JAC_MANY_ALT */ +/*#define MBEDTLS_ECP_NORMALIZE_JAC_ALT */ +/* Support for curves with Montgomery arithmetic */ +/*#define MBEDTLS_ECP_DOUBLE_ADD_MXZ_ALT */ +/*#define MBEDTLS_ECP_RANDOMIZE_MXZ_ALT */ +/*#define MBEDTLS_ECP_NORMALIZE_MXZ_ALT */ + +/** + * \def MBEDTLS_ENTROPY_HARDWARE_ALT + * + * Uncomment this macro to let mbed TLS use your own implementation of a + * hardware entropy collector. + * + * Your function must be called \c mbedtls_hardware_poll(), have the same + * prototype as declared in library/entropy_poll.h, and accept NULL as first + * argument. + * + * Uncomment to use your own hardware entropy collector. + */ +#define MBEDTLS_ENTROPY_HARDWARE_ALT + +/** + * \def MBEDTLS_AES_ROM_TABLES + * + * Use precomputed AES tables stored in ROM. + * + * Uncomment this macro to use precomputed AES tables stored in ROM. + * Comment this macro to generate AES tables in RAM at runtime. + * + * Tradeoff: Using precomputed ROM tables reduces RAM usage by ~8kb + * (or ~2kb if \c MBEDTLS_AES_FEWER_TABLES is used) and reduces the + * initialization time before the first AES operation can be performed. + * It comes at the cost of additional ~8kb ROM use (resp. ~2kb if \c + * MBEDTLS_AES_FEWER_TABLES below is used), and potentially degraded + * performance if ROM access is slower than RAM access. + * + * This option is independent of \c MBEDTLS_AES_FEWER_TABLES. + * + */ +#define MBEDTLS_AES_ROM_TABLES + +/** + * \def MBEDTLS_AES_FEWER_TABLES + * + * Use less ROM/RAM for AES tables. + * + * Uncommenting this macro omits 75% of the AES tables from + * ROM / RAM (depending on the value of \c MBEDTLS_AES_ROM_TABLES) + * by computing their values on the fly during operations + * (the tables are entry-wise rotations of one another). + * + * Tradeoff: Uncommenting this reduces the RAM / ROM footprint + * by ~6kb but at the cost of more arithmetic operations during + * runtime. Specifically, one has to compare 4 accesses within + * different tables to 4 accesses with additional arithmetic + * operations within the same table. The performance gain/loss + * depends on the system and memory details. + * + * This option is independent of \c MBEDTLS_AES_ROM_TABLES. + * + */ +#define MBEDTLS_AES_FEWER_TABLES + +/** + * \def MBEDTLS_CAMELLIA_SMALL_MEMORY + * + * Use less ROM for the Camellia implementation (saves about 768 bytes). + * + * Uncomment this macro to use less memory for Camellia. + */ +/*#define MBEDTLS_CAMELLIA_SMALL_MEMORY */ + +/** + * \def MBEDTLS_CIPHER_MODE_CBC + * + * Enable Cipher Block Chaining mode (CBC) for symmetric ciphers. + */ +#define MBEDTLS_CIPHER_MODE_CBC + +/** + * \def MBEDTLS_CIPHER_MODE_CFB + * + * Enable Cipher Feedback mode (CFB) for symmetric ciphers. + */ +#define MBEDTLS_CIPHER_MODE_CFB + +/** + * \def MBEDTLS_CIPHER_MODE_CTR + * + * Enable Counter Block Cipher mode (CTR) for symmetric ciphers. + */ +#define MBEDTLS_CIPHER_MODE_CTR + +/** + * \def MBEDTLS_CIPHER_MODE_OFB + * + * Enable Output Feedback mode (OFB) for symmetric ciphers. + */ +/*#define MBEDTLS_CIPHER_MODE_OFB */ + +/** + * \def MBEDTLS_CIPHER_MODE_XTS + * + * Enable Xor-encrypt-xor with ciphertext stealing mode (XTS) for AES. + */ +/*#define MBEDTLS_CIPHER_MODE_XTS */ + +/** + * \def MBEDTLS_CIPHER_NULL_CIPHER + * + * Enable NULL cipher. + * Warning: Only do so when you know what you are doing. This allows for + * encryption or channels without any security! + * + * To enable the following ciphersuites: + * MBEDTLS_TLS_ECDH_ECDSA_WITH_NULL_SHA + * MBEDTLS_TLS_ECDH_RSA_WITH_NULL_SHA + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_NULL_SHA + * MBEDTLS_TLS_ECDHE_RSA_WITH_NULL_SHA + * MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA256 + * MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA + * MBEDTLS_TLS_RSA_WITH_NULL_SHA256 + * MBEDTLS_TLS_RSA_WITH_NULL_SHA + * MBEDTLS_TLS_RSA_WITH_NULL_MD5 + * MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA + * MBEDTLS_TLS_PSK_WITH_NULL_SHA384 + * MBEDTLS_TLS_PSK_WITH_NULL_SHA256 + * MBEDTLS_TLS_PSK_WITH_NULL_SHA + * + * Uncomment this macro to enable the NULL cipher and ciphersuites + */ +#define MBEDTLS_CIPHER_NULL_CIPHER + +/** + * \def MBEDTLS_CIPHER_PADDING_PKCS7 + * + * MBEDTLS_CIPHER_PADDING_XXX: Uncomment or comment macros to add support for + * specific padding modes in the cipher layer with cipher modes that support + * padding (e.g. CBC) + * + * If you disable all padding modes, only full blocks can be used with CBC. + * + * Enable padding modes in the cipher layer. + */ +#define MBEDTLS_CIPHER_PADDING_PKCS7 +#define MBEDTLS_CIPHER_PADDING_ONE_AND_ZEROS +#define MBEDTLS_CIPHER_PADDING_ZEROS_AND_LEN +#define MBEDTLS_CIPHER_PADDING_ZEROS + +/** \def MBEDTLS_CTR_DRBG_USE_128_BIT_KEY + * + * Uncomment this macro to use a 128-bit key in the CTR_DRBG module. + * By default, CTR_DRBG uses a 256-bit key. + */ +/*#define MBEDTLS_CTR_DRBG_USE_128_BIT_KEY */ + +/** + * \def MBEDTLS_ECP_DP_SECP192R1_ENABLED + * + * MBEDTLS_ECP_XXXX_ENABLED: Enables specific curves within the Elliptic Curve + * module. By default all supported curves are enabled. + * + * Comment macros to disable the curve and functions for it + */ +/* Short Weierstrass curves (supporting ECP, ECDH, ECDSA) */ +#define MBEDTLS_ECP_DP_SECP192R1_ENABLED +#define MBEDTLS_ECP_DP_SECP224R1_ENABLED +#define MBEDTLS_ECP_DP_SECP256R1_ENABLED +#define MBEDTLS_ECP_DP_SECP384R1_ENABLED +#define MBEDTLS_ECP_DP_SECP521R1_ENABLED +#define MBEDTLS_ECP_DP_SECP192K1_ENABLED +#define MBEDTLS_ECP_DP_SECP224K1_ENABLED +#define MBEDTLS_ECP_DP_SECP256K1_ENABLED +#define MBEDTLS_ECP_DP_BP256R1_ENABLED +#define MBEDTLS_ECP_DP_BP384R1_ENABLED +#define MBEDTLS_ECP_DP_BP512R1_ENABLED +/* Montgomery curves (supporting ECP) */ +#define MBEDTLS_ECP_DP_CURVE25519_ENABLED +#define MBEDTLS_ECP_DP_CURVE448_ENABLED + +/** + * \def MBEDTLS_ECP_NIST_OPTIM + * + * Enable specific 'modulo p' routines for each NIST prime. + * Depending on the prime and architecture, makes operations 4 to 8 times + * faster on the corresponding curve. + * + * Comment this macro to disable NIST curves optimisation. + */ +#define MBEDTLS_ECP_NIST_OPTIM + +/** + * \def MBEDTLS_ECP_RESTARTABLE + * + * Enable "non-blocking" ECC operations that can return early and be resumed. + * + * This allows various functions to pause by returning + * #MBEDTLS_ERR_ECP_IN_PROGRESS (or, for functions in the SSL module, + * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS) and then be called later again in + * order to further progress and eventually complete their operation. This is + * controlled through mbedtls_ecp_set_max_ops() which limits the maximum + * number of ECC operations a function may perform before pausing; see + * mbedtls_ecp_set_max_ops() for more information. + * + * This is useful in non-threaded environments if you want to avoid blocking + * for too long on ECC (and, hence, X.509 or SSL/TLS) operations. + * + * Uncomment this macro to enable restartable ECC computations. + * + * \note This option only works with the default software implementation of + * elliptic curve functionality. It is incompatible with + * MBEDTLS_ECP_ALT, MBEDTLS_ECDH_XXX_ALT, MBEDTLS_ECDSA_XXX_ALT. + */ +/*#define MBEDTLS_ECP_RESTARTABLE */ + +/** + * \def MBEDTLS_ECDSA_DETERMINISTIC + * + * Enable deterministic ECDSA (RFC 6979). + * Standard ECDSA is "fragile" in the sense that lack of entropy when signing + * may result in a compromise of the long-term signing key. This is avoided by + * the deterministic variant. + * + * Requires: MBEDTLS_HMAC_DRBG_C, MBEDTLS_ECDSA_C + * + * Comment this macro to disable deterministic ECDSA. + */ +/*#define MBEDTLS_ECDSA_DETERMINISTIC */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_PSK_ENABLED + * + * Enable the PSK based ciphersuite modes in SSL / TLS. + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_PSK_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_PSK_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_CBC_SHA256 + */ +/*#define MBEDTLS_KEY_EXCHANGE_PSK_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED + * + * Enable the DHE-PSK based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_DHM_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 + * + * \warning Using DHE constitutes a security risk as it + * is not possible to validate custom DH parameters. + * If possible, it is recommended users should consider + * preferring other methods of key exchange. + * See dhm.h for more details. + * + */ +/*#define MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED + * + * Enable the ECDHE-PSK based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_ECDH_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 + */ +/*#define MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED + * + * Enable the RSA-PSK based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, + * MBEDTLS_X509_CRT_PARSE_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_CBC_SHA256 + */ +/*#define MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_RSA_ENABLED + * + * Enable the RSA-only based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, + * MBEDTLS_X509_CRT_PARSE_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA + * MBEDTLS_TLS_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA + */ +/*#define MBEDTLS_KEY_EXCHANGE_RSA_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED + * + * Enable the DHE-RSA based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_DHM_C, MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, + * MBEDTLS_X509_CRT_PARSE_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA + * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA + * + * \warning Using DHE constitutes a security risk as it + * is not possible to validate custom DH parameters. + * If possible, it is recommended users should consider + * preferring other methods of key exchange. + * See dhm.h for more details. + * + */ +/*#define MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED + * + * Enable the ECDHE-RSA based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_ECDH_C, MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, + * MBEDTLS_X509_CRT_PARSE_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 + */ +#define MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED + +/** + * \def MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED + * + * Enable the ECDHE-ECDSA based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_ECDH_C, MBEDTLS_ECDSA_C, MBEDTLS_X509_CRT_PARSE_C, + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 + */ +#define MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED + +/** + * \def MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED + * + * Enable the ECDH-ECDSA based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_ECDH_C, MBEDTLS_ECDSA_C, MBEDTLS_X509_CRT_PARSE_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 + */ +/*#define MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED + * + * Enable the ECDH-RSA based ciphersuite modes in SSL / TLS. + * + * Requires: MBEDTLS_ECDH_C, MBEDTLS_RSA_C, MBEDTLS_X509_CRT_PARSE_C + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_GCM_SHA384 + */ +/*#define MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED */ + +/** + * \def MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED + * + * Enable the ECJPAKE based ciphersuite modes in SSL / TLS. + * + * \warning This is currently experimental. EC J-PAKE support is based on the + * Thread v1.0.0 specification; incompatible changes to the specification + * might still happen. For this reason, this is disabled by default. + * + * Requires: MBEDTLS_ECJPAKE_C + * MBEDTLS_SHA256_C + * MBEDTLS_ECP_DP_SECP256R1_ENABLED + * + * This enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECJPAKE_WITH_AES_128_CCM_8 + */ +/*#define MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED */ + +/** + * \def MBEDTLS_PK_PARSE_EC_EXTENDED + * + * Enhance support for reading EC keys using variants of SEC1 not allowed by + * RFC 5915 and RFC 5480. + * + * Currently this means parsing the SpecifiedECDomain choice of EC + * parameters (only known groups are supported, not arbitrary domains, to + * avoid validation issues). + * + * Disable if you only need to support RFC 5915 + 5480 key formats. + */ +/*#define MBEDTLS_PK_PARSE_EC_EXTENDED */ + +/** + * \def MBEDTLS_ERROR_STRERROR_DUMMY + * + * Enable a dummy error function to make use of mbedtls_strerror() in + * third party libraries easier when MBEDTLS_ERROR_C is disabled + * (no effect when MBEDTLS_ERROR_C is enabled). + * + * You can safely disable this if MBEDTLS_ERROR_C is enabled, or if you're + * not using mbedtls_strerror() or error_strerror() in your application. + * + * Disable if you run into name conflicts and want to really remove the + * mbedtls_strerror() + */ +#define MBEDTLS_ERROR_STRERROR_DUMMY + +/** + * \def MBEDTLS_GENPRIME + * + * Enable the prime-number generation code. + * + * Requires: MBEDTLS_BIGNUM_C + */ +#define MBEDTLS_GENPRIME + +/** + * \def MBEDTLS_FS_IO + * + * Enable functions that use the filesystem. + */ +/*#define MBEDTLS_FS_IO */ + +/** + * \def MBEDTLS_NO_DEFAULT_ENTROPY_SOURCES + * + * Do not add default entropy sources in mbedtls_entropy_init(). + * + * This is useful to have more control over the added entropy sources in an + * application. + * + * Uncomment this macro to prevent loading of default entropy functions. + */ +/*#define MBEDTLS_NO_DEFAULT_ENTROPY_SOURCES */ + +/** + * \def MBEDTLS_NO_PLATFORM_ENTROPY + * + * Do not use built-in platform entropy functions. + * This is useful if your platform does not support + * standards like the /dev/urandom or Windows CryptoAPI. + * + * Uncomment this macro to disable the built-in platform entropy functions. + */ +#define MBEDTLS_NO_PLATFORM_ENTROPY + +/** + * \def MBEDTLS_ENTROPY_FORCE_SHA256 + * + * Force the entropy accumulator to use a SHA-256 accumulator instead of the + * default SHA-512 based one (if both are available). + * + * Requires: MBEDTLS_SHA256_C + * + * On 32-bit systems SHA-256 can be much faster than SHA-512. Use this option + * if you have performance concerns. + * + * This option is only useful if both MBEDTLS_SHA256_C and + * MBEDTLS_SHA512_C are defined. Otherwise the available hash module is used. + */ +#define MBEDTLS_ENTROPY_FORCE_SHA256 + +/** + * \def MBEDTLS_ENTROPY_NV_SEED + * + * Enable the non-volatile (NV) seed file-based entropy source. + * (Also enables the NV seed read/write functions in the platform layer) + * + * This is crucial (if not required) on systems that do not have a + * cryptographic entropy source (in hardware or kernel) available. + * + * Requires: MBEDTLS_ENTROPY_C, MBEDTLS_PLATFORM_C + * + * \note The read/write functions that are used by the entropy source are + * determined in the platform layer, and can be modified at runtime and/or + * compile-time depending on the flags (MBEDTLS_PLATFORM_NV_SEED_*) used. + * + * \note If you use the default implementation functions that read a seedfile + * with regular fopen(), please make sure you make a seedfile with the + * proper name (defined in MBEDTLS_PLATFORM_STD_NV_SEED_FILE) and at + * least MBEDTLS_ENTROPY_BLOCK_SIZE bytes in size that can be read from + * and written to or you will get an entropy source error! The default + * implementation will only use the first MBEDTLS_ENTROPY_BLOCK_SIZE + * bytes from the file. + * + * \note The entropy collector will write to the seed file before entropy is + * given to an external source, to update it. + */ +/*#define MBEDTLS_ENTROPY_NV_SEED */ + +/* MBEDTLS_PSA_CRYPTO_KEY_ID_ENCODES_OWNER + * + * Enable key identifiers that encode a key owner identifier. + * + * The owner of a key is identified by a value of type ::mbedtls_key_owner_id_t + * which is currently hard-coded to be int32_t. + * + * Note that this option is meant for internal use only and may be removed + * without notice. It is incompatible with MBEDTLS_USE_PSA_CRYPTO. + */ +/*#define MBEDTLS_PSA_CRYPTO_KEY_ID_ENCODES_OWNER */ + +/** + * \def MBEDTLS_MEMORY_DEBUG + * + * Enable debugging of buffer allocator memory issues. Automatically prints + * (to stderr) all (fatal) messages on memory allocation issues. Enables + * function for 'debug output' of allocated memory. + * + * Requires: MBEDTLS_MEMORY_BUFFER_ALLOC_C + * + * Uncomment this macro to let the buffer allocator print out error messages. + */ +/*#define MBEDTLS_MEMORY_DEBUG */ + +/** + * \def MBEDTLS_MEMORY_BACKTRACE + * + * Include backtrace information with each allocated block. + * + * Requires: MBEDTLS_MEMORY_BUFFER_ALLOC_C + * GLIBC-compatible backtrace() an backtrace_symbols() support + * + * Uncomment this macro to include backtrace information + */ +/*#define MBEDTLS_MEMORY_BACKTRACE */ + +/** + * \def MBEDTLS_PK_RSA_ALT_SUPPORT + * + * Support external private RSA keys (eg from a HSM) in the PK layer. + * + * Comment this macro to disable support for external private RSA keys. + */ +#define MBEDTLS_PK_RSA_ALT_SUPPORT + +/** + * \def MBEDTLS_PKCS1_V15 + * + * Enable support for PKCS#1 v1.5 encoding. + * + * Requires: MBEDTLS_RSA_C + * + * This enables support for PKCS#1 v1.5 operations. + */ +#define MBEDTLS_PKCS1_V15 + +/** + * \def MBEDTLS_PKCS1_V21 + * + * Enable support for PKCS#1 v2.1 encoding. + * + * Requires: MBEDTLS_MD_C, MBEDTLS_RSA_C + * + * This enables support for RSAES-OAEP and RSASSA-PSS operations. + */ +/*#define MBEDTLS_PKCS1_V21 */ + +/** \def MBEDTLS_PSA_CRYPTO_BUILTIN_KEYS + * + * Enable support for platform built-in keys. If you enable this feature, + * you must implement the function mbedtls_psa_platform_get_builtin_key(). + * See the documentation of that function for more information. + * + * Built-in keys are typically derived from a hardware unique key or + * stored in a secure element. + * + * Requires: MBEDTLS_PSA_CRYPTO_C. + * + * \warning This interface is experimental and may change or be removed + * without notice. + */ +/*#define MBEDTLS_PSA_CRYPTO_BUILTIN_KEYS */ + +/** \def MBEDTLS_PSA_CRYPTO_CLIENT + * + * Enable support for PSA crypto client. + * + * \note This option allows to include the code necessary for a PSA + * crypto client when the PSA crypto implementation is not included in + * the library (MBEDTLS_PSA_CRYPTO_C disabled). The code included is the + * code to set and get PSA key attributes. + * The development of PSA drivers partially relying on the library to + * fulfill the hardware gaps is another possible usage of this option. + * + * \warning This interface is experimental and may change or be removed + * without notice. + */ +/* #define MBEDTLS_PSA_CRYPTO_CLIENT */ + +/** \def MBEDTLS_PSA_CRYPTO_DRIVERS + * + * Enable support for the experimental PSA crypto driver interface. + * + * Requires: MBEDTLS_PSA_CRYPTO_C + * + * \warning This interface is experimental and may change or be removed + * without notice. + */ +/*#define MBEDTLS_PSA_CRYPTO_DRIVERS */ + +/** \def MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG + * + * Make the PSA Crypto module use an external random generator provided + * by a driver, instead of Mbed TLS's entropy and DRBG modules. + * + * \note This random generator must deliver random numbers with cryptographic + * quality and high performance. It must supply unpredictable numbers + * with a uniform distribution. The implementation of this function + * is responsible for ensuring that the random generator is seeded + * with sufficient entropy. If you have a hardware TRNG which is slow + * or delivers non-uniform output, declare it as an entropy source + * with mbedtls_entropy_add_source() instead of enabling this option. + * + * If you enable this option, you must configure the type + * ::mbedtls_psa_external_random_context_t in psa/crypto_platform.h + * and define a function called mbedtls_psa_external_get_random() + * with the following prototype: + * ``` + * psa_status_t mbedtls_psa_external_get_random( + * mbedtls_psa_external_random_context_t *context, + * uint8_t *output, size_t output_size, size_t *output_length); + * ); + * ``` + * The \c context value is initialized to 0 before the first call. + * The function must fill the \c output buffer with \p output_size bytes + * of random data and set \c *output_length to \p output_size. + * + * Requires: MBEDTLS_PSA_CRYPTO_C + * + * \warning If you enable this option, code that uses the PSA cryptography + * interface will not use any of the entropy sources set up for + * the entropy module, nor the NV seed that MBEDTLS_ENTROPY_NV_SEED + * enables. + * + * \note This option is experimental and may be removed without notice. + */ +/*#define MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG */ + +/** + * \def MBEDTLS_PSA_CRYPTO_SPM + * + * When MBEDTLS_PSA_CRYPTO_SPM is defined, the code is built for SPM (Secure + * Partition Manager) integration which separates the code into two parts: a + * NSPE (Non-Secure Process Environment) and an SPE (Secure Process + * Environment). + * + * Module: library/psa_crypto.c + * Requires: MBEDTLS_PSA_CRYPTO_C + * + */ +/*#define MBEDTLS_PSA_CRYPTO_SPM */ + +/** + * \def MBEDTLS_PSA_INJECT_ENTROPY + * + * Enable support for entropy injection at first boot. This feature is + * required on systems that do not have a built-in entropy source (TRNG). + * This feature is currently not supported on systems that have a built-in + * entropy source. + * + * Requires: MBEDTLS_PSA_CRYPTO_STORAGE_C, MBEDTLS_ENTROPY_NV_SEED + * + */ +/*#define MBEDTLS_PSA_INJECT_ENTROPY */ + +/** + * \def MBEDTLS_RSA_NO_CRT + * + * Do not use the Chinese Remainder Theorem + * for the RSA private operation. + * + * Uncomment this macro to disable the use of CRT in RSA. + * + */ +/*#define MBEDTLS_RSA_NO_CRT */ + +/** + * \def MBEDTLS_SELF_TEST + * + * Enable the checkup functions (*_self_test). + */ +/*#define MBEDTLS_SELF_TEST */ + +/** + * \def MBEDTLS_SHA256_SMALLER + * + * Enable an implementation of SHA-256 that has lower ROM footprint but also + * lower performance. + * + * The default implementation is meant to be a reasonable compromise between + * performance and size. This version optimizes more aggressively for size at + * the expense of performance. Eg on Cortex-M4 it reduces the size of + * mbedtls_sha256_process() from ~2KB to ~0.5KB for a performance hit of about + * 30%. + * + * Uncomment to enable the smaller implementation of SHA256. + */ +/*#define MBEDTLS_SHA256_SMALLER */ + +/** + * \def MBEDTLS_SHA512_SMALLER + * + * Enable an implementation of SHA-512 that has lower ROM footprint but also + * lower performance. + * + * Uncomment to enable the smaller implementation of SHA512. + */ +/*#define MBEDTLS_SHA512_SMALLER */ + +/** + * \def MBEDTLS_SSL_ALL_ALERT_MESSAGES + * + * Enable sending of alert messages in case of encountered errors as per RFC. + * If you choose not to send the alert messages, mbed TLS can still communicate + * with other servers, only debugging of failures is harder. + * + * The advantage of not sending alert messages, is that no information is given + * about reasons for failures thus preventing adversaries of gaining intel. + * + * Enable sending of all alert messages + */ +/*#define MBEDTLS_SSL_ALL_ALERT_MESSAGES */ + +/** + * \def MBEDTLS_SSL_DTLS_CONNECTION_ID + * + * Enable support for the DTLS Connection ID extension + * (version draft-ietf-tls-dtls-connection-id-05, + * https://tools.ietf.org/html/draft-ietf-tls-dtls-connection-id-05) + * which allows to identify DTLS connections across changes + * in the underlying transport. + * + * Setting this option enables the SSL APIs `mbedtls_ssl_set_cid()`, + * `mbedtls_ssl_get_peer_cid()` and `mbedtls_ssl_conf_cid()`. + * See the corresponding documentation for more information. + * + * \warning The Connection ID extension is still in draft state. + * We make no stability promises for the availability + * or the shape of the API controlled by this option. + * + * The maximum lengths of outgoing and incoming CIDs can be configured + * through the options + * - MBEDTLS_SSL_CID_OUT_LEN_MAX + * - MBEDTLS_SSL_CID_IN_LEN_MAX. + * + * Requires: MBEDTLS_SSL_PROTO_DTLS + * + * Uncomment to enable the Connection ID extension. + */ +/*#define MBEDTLS_SSL_DTLS_CONNECTION_ID */ + +/** + * \def MBEDTLS_SSL_ASYNC_PRIVATE + * + * Enable asynchronous external private key operations in SSL. This allows + * you to configure an SSL connection to call an external cryptographic + * module to perform private key operations instead of performing the + * operation inside the library. + * + */ +/*#define MBEDTLS_SSL_ASYNC_PRIVATE */ + +/** + * \def MBEDTLS_SSL_CONTEXT_SERIALIZATION + * + * Enable serialization of the TLS context structures, through use of the + * functions mbedtls_ssl_context_save() and mbedtls_ssl_context_load(). + * + * This pair of functions allows one side of a connection to serialize the + * context associated with the connection, then free or re-use that context + * while the serialized state is persisted elsewhere, and finally deserialize + * that state to a live context for resuming read/write operations on the + * connection. From a protocol perspective, the state of the connection is + * unaffected, in particular this is entirely transparent to the peer. + * + * Note: this is distinct from TLS session resumption, which is part of the + * protocol and fully visible by the peer. TLS session resumption enables + * establishing new connections associated to a saved session with shorter, + * lighter handshakes, while context serialization is a local optimization in + * handling a single, potentially long-lived connection. + * + * Enabling these APIs makes some SSL structures larger, as 64 extra bytes are + * saved after the handshake to allow for more efficient serialization, so if + * you don't need this feature you'll save RAM by disabling it. + * + * Comment to disable the context serialization APIs. + */ +#define MBEDTLS_SSL_CONTEXT_SERIALIZATION + +/** + * \def MBEDTLS_SSL_DEBUG_ALL + * + * Enable the debug messages in SSL module for all issues. + * Debug messages have been disabled in some places to prevent timing + * attacks due to (unbalanced) debugging function calls. + * + * If you need all error reporting you should enable this during debugging, + * but remove this for production servers that should log as well. + * + * Uncomment this macro to report all debug messages on errors introducing + * a timing side-channel. + * + */ +/*#define MBEDTLS_SSL_DEBUG_ALL */ + +/** \def MBEDTLS_SSL_ENCRYPT_THEN_MAC + * + * Enable support for Encrypt-then-MAC, RFC 7366. + * + * This allows peers that both support it to use a more robust protection for + * ciphersuites using CBC, providing deep resistance against timing attacks + * on the padding or underlying cipher. + * + * This only affects CBC ciphersuites, and is useless if none is defined. + * + * Requires: MBEDTLS_SSL_PROTO_TLS1_2 + * + * Comment this macro to disable support for Encrypt-then-MAC + */ +#define MBEDTLS_SSL_ENCRYPT_THEN_MAC + +/** \def MBEDTLS_SSL_EXTENDED_MASTER_SECRET + * + * Enable support for RFC 7627: Session Hash and Extended Master Secret + * Extension. + * + * This was introduced as "the proper fix" to the Triple Handshake family of + * attacks, but it is recommended to always use it (even if you disable + * renegotiation), since it actually fixes a more fundamental issue in the + * original SSL/TLS design, and has implications beyond Triple Handshake. + * + * Requires: MBEDTLS_SSL_PROTO_TLS1_2 + * + * Comment this macro to disable support for Extended Master Secret. + */ +#define MBEDTLS_SSL_EXTENDED_MASTER_SECRET + +/** + * \def MBEDTLS_SSL_KEEP_PEER_CERTIFICATE + * + * This option controls the availability of the API mbedtls_ssl_get_peer_cert() + * giving access to the peer's certificate after completion of the handshake. + * + * Unless you need mbedtls_ssl_peer_cert() in your application, it is + * recommended to disable this option for reduced RAM usage. + * + * \note If this option is disabled, mbedtls_ssl_get_peer_cert() is still + * defined, but always returns \c NULL. + * + * \note This option has no influence on the protection against the + * triple handshake attack. Even if it is disabled, Mbed TLS will + * still ensure that certificates do not change during renegotiation, + * for example by keeping a hash of the peer's certificate. + * + * Comment this macro to disable storing the peer's certificate + * after the handshake. + */ +/*#define MBEDTLS_SSL_KEEP_PEER_CERTIFICATE */ + +/** + * \def MBEDTLS_SSL_RENEGOTIATION + * + * Enable support for TLS renegotiation. + * + * The two main uses of renegotiation are (1) refresh keys on long-lived + * connections and (2) client authentication after the initial handshake. + * If you don't need renegotiation, it's probably better to disable it, since + * it has been associated with security issues in the past and is easy to + * misuse/misunderstand. + * + * Comment this to disable support for renegotiation. + * + * \note Even if this option is disabled, both client and server are aware + * of the Renegotiation Indication Extension (RFC 5746) used to + * prevent the SSL renegotiation attack (see RFC 5746 Sect. 1). + * (See \c mbedtls_ssl_conf_legacy_renegotiation for the + * configuration of this extension). + * + */ +/*#define MBEDTLS_SSL_RENEGOTIATION */ + +/** + * \def MBEDTLS_SSL_MAX_FRAGMENT_LENGTH + * + * Enable support for RFC 6066 max_fragment_length extension in SSL. + * + * Comment this macro to disable support for the max_fragment_length extension + */ +#define MBEDTLS_SSL_MAX_FRAGMENT_LENGTH + +/** + * \def MBEDTLS_SSL_PROTO_TLS1_2 + * + * Enable support for TLS 1.2 (and DTLS 1.2 if DTLS is enabled). + * + * Requires: MBEDTLS_SHA1_C or MBEDTLS_SHA256_C or MBEDTLS_SHA512_C + * (Depends on ciphersuites) + * + * Comment this macro to disable support for TLS 1.2 / DTLS 1.2 + */ +#define MBEDTLS_SSL_PROTO_TLS1_2 + +/** + * \def MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL + * + * This macro is used to selectively enable experimental parts + * of the code that contribute to the ongoing development of + * the prototype TLS 1.3 and DTLS 1.3 implementation, and provide + * no other purpose. + * + * \warning TLS 1.3 and DTLS 1.3 aren't yet supported in Mbed TLS, + * and no feature exposed through this macro is part of the + * public API. In particular, features under the control + * of this macro are experimental and don't come with any + * stability guarantees. + * + * Uncomment this macro to enable experimental and partial + * functionality specific to TLS 1.3. + */ +/*#define MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL */ + +/** + * \def MBEDTLS_SSL_PROTO_DTLS + * + * Enable support for DTLS (all available versions). + * + * Enable this and MBEDTLS_SSL_PROTO_TLS1_2 to enable DTLS 1.2. + * + * Requires: MBEDTLS_SSL_PROTO_TLS1_2 + * + * Comment this macro to disable support for DTLS + */ +/*#define MBEDTLS_SSL_PROTO_DTLS */ + +/** + * \def MBEDTLS_SSL_ALPN + * + * Enable support for RFC 7301 Application Layer Protocol Negotiation. + * + * Comment this macro to disable support for ALPN. + */ +#define MBEDTLS_SSL_ALPN + +/** + * \def MBEDTLS_SSL_DTLS_ANTI_REPLAY + * + * Enable support for the anti-replay mechanism in DTLS. + * + * Requires: MBEDTLS_SSL_TLS_C + * MBEDTLS_SSL_PROTO_DTLS + * + * \warning Disabling this is often a security risk! + * See mbedtls_ssl_conf_dtls_anti_replay() for details. + * + * Comment this to disable anti-replay in DTLS. + */ +/*#define MBEDTLS_SSL_DTLS_ANTI_REPLAY */ + +/** + * \def MBEDTLS_SSL_DTLS_HELLO_VERIFY + * + * Enable support for HelloVerifyRequest on DTLS servers. + * + * This feature is highly recommended to prevent DTLS servers being used as + * amplifiers in DoS attacks against other hosts. It should always be enabled + * unless you know for sure amplification cannot be a problem in the + * environment in which your server operates. + * + * \warning Disabling this can be a security risk! (see above) + * + * Requires: MBEDTLS_SSL_PROTO_DTLS + * + * Comment this to disable support for HelloVerifyRequest. + */ +/*#define MBEDTLS_SSL_DTLS_HELLO_VERIFY */ + +/** + * \def MBEDTLS_SSL_DTLS_SRTP + * + * Enable support for negotiation of DTLS-SRTP (RFC 5764) + * through the use_srtp extension. + * + * \note This feature provides the minimum functionality required + * to negotiate the use of DTLS-SRTP and to allow the derivation of + * the associated SRTP packet protection key material. + * In particular, the SRTP packet protection itself, as well as the + * demultiplexing of RTP and DTLS packets at the datagram layer + * (see Section 5 of RFC 5764), are not handled by this feature. + * Instead, after successful completion of a handshake negotiating + * the use of DTLS-SRTP, the extended key exporter API + * mbedtls_ssl_conf_export_keys_cb() should be used to implement + * the key exporter described in Section 4.2 of RFC 5764 and RFC 5705 + * (this is implemented in the SSL example programs). + * The resulting key should then be passed to an SRTP stack. + * + * Setting this option enables the runtime API + * mbedtls_ssl_conf_dtls_srtp_protection_profiles() + * through which the supported DTLS-SRTP protection + * profiles can be configured. You must call this API at + * runtime if you wish to negotiate the use of DTLS-SRTP. + * + * Requires: MBEDTLS_SSL_PROTO_DTLS + * + * Uncomment this to enable support for use_srtp extension. + */ +/*#define MBEDTLS_SSL_DTLS_SRTP */ + +/** + * \def MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE + * + * Enable server-side support for clients that reconnect from the same port. + * + * Some clients unexpectedly close the connection and try to reconnect using the + * same source port. This needs special support from the server to handle the + * new connection securely, as described in section 4.2.8 of RFC 6347. This + * flag enables that support. + * + * Requires: MBEDTLS_SSL_DTLS_HELLO_VERIFY + * + * Comment this to disable support for clients reusing the source port. + */ +/*#define MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE */ + +/** + * \def MBEDTLS_SSL_SESSION_TICKETS + * + * Enable support for RFC 5077 session tickets in SSL. + * Client-side, provides full support for session tickets (maintenance of a + * session store remains the responsibility of the application, though). + * Server-side, you also need to provide callbacks for writing and parsing + * tickets, including authenticated encryption and key management. Example + * callbacks are provided by MBEDTLS_SSL_TICKET_C. + * + * Comment this macro to disable support for SSL session tickets + */ +/*#define MBEDTLS_SSL_SESSION_TICKETS */ + +/** + * \def MBEDTLS_SSL_EXPORT_KEYS + * + * Enable support for exporting key block and master secret. + * This is required for certain users of TLS, e.g. EAP-TLS. + * + * Comment this macro to disable support for key export + */ +/*#define MBEDTLS_SSL_EXPORT_KEYS */ + +/** + * \def MBEDTLS_SSL_SERVER_NAME_INDICATION + * + * Enable support for RFC 6066 server name indication (SNI) in SSL. + * + * Requires: MBEDTLS_X509_CRT_PARSE_C + * + * Comment this macro to disable support for server name indication in SSL + */ +#define MBEDTLS_SSL_SERVER_NAME_INDICATION + +/** + * \def MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH + * + * When this option is enabled, the SSL buffer will be resized automatically + * based on the negotiated maximum fragment length in each direction. + * + * Requires: MBEDTLS_SSL_MAX_FRAGMENT_LENGTH + */ +/*#define MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH */ + +/** + * \def MBEDTLS_TEST_CONSTANT_FLOW_MEMSAN + * + * Enable testing of the constant-flow nature of some sensitive functions with + * clang's MemorySanitizer. This causes some existing tests to also test + * this non-functional property of the code under test. + * + * This setting requires compiling with clang -fsanitize=memory. The test + * suites can then be run normally. + * + * \warning This macro is only used for extended testing; it is not considered + * part of the library's API, so it may change or disappear at any time. + * + * Uncomment to enable testing of the constant-flow nature of selected code. + */ +/*#define MBEDTLS_TEST_CONSTANT_FLOW_MEMSAN */ + +/** + * \def MBEDTLS_TEST_CONSTANT_FLOW_VALGRIND + * + * Enable testing of the constant-flow nature of some sensitive functions with + * valgrind's memcheck tool. This causes some existing tests to also test + * this non-functional property of the code under test. + * + * This setting requires valgrind headers for building, and is only useful for + * testing if the tests suites are run with valgrind's memcheck. This can be + * done for an individual test suite with 'valgrind ./test_suite_xxx', or when + * using CMake, this can be done for all test suites with 'make memcheck'. + * + * \warning This macro is only used for extended testing; it is not considered + * part of the library's API, so it may change or disappear at any time. + * + * Uncomment to enable testing of the constant-flow nature of selected code. + */ +/*#define MBEDTLS_TEST_CONSTANT_FLOW_VALGRIND */ + +/** + * \def MBEDTLS_TEST_HOOKS + * + * Enable features for invasive testing such as introspection functions and + * hooks for fault injection. This enables additional unit tests. + * + * Merely enabling this feature should not change the behavior of the product. + * It only adds new code, and new branching points where the default behavior + * is the same as when this feature is disabled. + * However, this feature increases the attack surface: there is an added + * risk of vulnerabilities, and more gadgets that can make exploits easier. + * Therefore this feature must never be enabled in production. + * + * See `docs/architecture/testing/mbed-crypto-invasive-testing.md` for more + * information. + * + * Uncomment to enable invasive tests. + */ +/*#define MBEDTLS_TEST_HOOKS */ + +/** + * \def MBEDTLS_THREADING_ALT + * + * Provide your own alternate threading implementation. + * + * Requires: MBEDTLS_THREADING_C + * + * Uncomment this to allow your own alternate threading implementation. + */ +#define MBEDTLS_THREADING_ALT + +/** + * \def MBEDTLS_THREADING_PTHREAD + * + * Enable the pthread wrapper layer for the threading layer. + * + * Requires: MBEDTLS_THREADING_C + * + * Uncomment this to enable pthread mutexes. + */ +/*#define MBEDTLS_THREADING_PTHREAD */ + +/** + * \def MBEDTLS_USE_PSA_CRYPTO + * + * Make the X.509 and TLS library use PSA for cryptographic operations, and + * enable new APIs for using keys handled by PSA Crypto. + * + * \note Development of this option is currently in progress, and parts of Mbed + * TLS's X.509 and TLS modules are not ported to PSA yet. However, these parts + * will still continue to work as usual, so enabling this option should not + * break backwards compatibility. + * + * \warning The PSA Crypto API is in beta stage. While you're welcome to + * experiment using it, incompatible API changes are still possible, and some + * parts may not have reached the same quality as the rest of Mbed TLS yet. + * + * \warning This option enables new Mbed TLS APIs that are dependent on the + * PSA Crypto API, so can't come with the same stability guarantees as the + * rest of the Mbed TLS APIs. You're welcome to experiment with them, but for + * now, access to these APIs is opt-in (via enabling the present option), in + * order to clearly differentiate them from the stable Mbed TLS APIs. + * + * Requires: MBEDTLS_PSA_CRYPTO_C. + * + * Uncomment this to enable internal use of PSA Crypto and new associated APIs. + */ +/*#define MBEDTLS_USE_PSA_CRYPTO */ + +/** + * \def MBEDTLS_PSA_CRYPTO_CONFIG + * + * This setting allows support for cryptographic mechanisms through the PSA + * API to be configured separately from support through the mbedtls API. + * + * Uncomment this to enable use of PSA Crypto configuration settings which + * can be found in include/psa/crypto_config.h. + * + * This feature is still experimental and is not ready for production since + * it is not completed. + */ +/*#define MBEDTLS_PSA_CRYPTO_CONFIG */ + +/** + * \def MBEDTLS_VERSION_FEATURES + * + * Allow run-time checking of compile-time enabled features. Thus allowing users + * to check at run-time if the library is for instance compiled with threading + * support via mbedtls_version_check_feature(). + * + * Requires: MBEDTLS_VERSION_C + * + * Comment this to disable run-time checking and save ROM space + */ +/*#define MBEDTLS_VERSION_FEATURES */ + +/** + * \def MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK + * + * If set, this enables the X.509 API `mbedtls_x509_crt_verify_with_ca_cb()` + * and the SSL API `mbedtls_ssl_conf_ca_cb()` which allow users to configure + * the set of trusted certificates through a callback instead of a linked + * list. + * + * This is useful for example in environments where a large number of trusted + * certificates is present and storing them in a linked list isn't efficient + * enough, or when the set of trusted certificates changes frequently. + * + * See the documentation of `mbedtls_x509_crt_verify_with_ca_cb()` and + * `mbedtls_ssl_conf_ca_cb()` for more information. + * + * Uncomment to enable trusted certificate callbacks. + */ +#define MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK + +/** + * \def MBEDTLS_X509_REMOVE_INFO + * + * Disable mbedtls_x509_*_info() and related APIs. + * + * Uncomment to omit mbedtls_x509_*_info(), as well as mbedtls_debug_print_crt() + * and other functions/constants only used by these functions, thus reducing + * the code footprint by several KB. + */ +/*#define MBEDTLS_X509_REMOVE_INFO */ + +/** + * \def MBEDTLS_X509_RSASSA_PSS_SUPPORT + * + * Enable parsing and verification of X.509 certificates, CRLs and CSRS + * signed with RSASSA-PSS (aka PKCS#1 v2.1). + * + * Comment this macro to disallow using RSASSA-PSS in certificates. + */ +/*#define MBEDTLS_X509_RSASSA_PSS_SUPPORT */ +/* \} name SECTION: mbed TLS feature support */ + +/** + * \name SECTION: mbed TLS modules + * + * This section enables or disables entire modules in mbed TLS + * \{ + */ + +/** + * \def MBEDTLS_AESNI_C + * + * Enable AES-NI support on x86-64. + * + * Module: library/aesni.c + * Caller: library/aes.c + * + * Requires: MBEDTLS_HAVE_ASM + * + * This modules adds support for the AES-NI instructions on x86-64 + */ +/*#define MBEDTLS_AESNI_C */ + +/** + * \def MBEDTLS_AES_C + * + * Enable the AES block cipher. + * + * Module: library/aes.c + * Caller: library/cipher.c + * library/pem.c + * library/ctr_drbg.c + * + * This module enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_RSA_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_RSA_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA + * MBEDTLS_TLS_PSK_WITH_AES_256_GCM_SHA384 + * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA384 + * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA + * MBEDTLS_TLS_PSK_WITH_AES_128_GCM_SHA256 + * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA256 + * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA + * + * PEM_PARSE uses AES for decrypting encrypted keys. + */ +#define MBEDTLS_AES_C + +/** + * \def MBEDTLS_ASN1_PARSE_C + * + * Enable the generic ASN1 parser. + * + * Module: library/asn1.c + * Caller: library/x509.c + * library/dhm.c + * library/pkcs12.c + * library/pkcs5.c + * library/pkparse.c + */ +#define MBEDTLS_ASN1_PARSE_C + +/** + * \def MBEDTLS_ASN1_WRITE_C + * + * Enable the generic ASN1 writer. + * + * Module: library/asn1write.c + * Caller: library/ecdsa.c + * library/pkwrite.c + * library/x509_create.c + * library/x509write_crt.c + * library/x509write_csr.c + */ +#define MBEDTLS_ASN1_WRITE_C + +/** + * \def MBEDTLS_BASE64_C + * + * Enable the Base64 module. + * + * Module: library/base64.c + * Caller: library/pem.c + * + * This module is required for PEM support (required by X.509). + */ +#define MBEDTLS_BASE64_C + +/** + * \def MBEDTLS_BIGNUM_C + * + * Enable the multi-precision integer library. + * + * Module: library/bignum.c + * Caller: library/dhm.c + * library/ecp.c + * library/ecdsa.c + * library/rsa.c + * library/rsa_alt_helpers.c + * library/ssl_tls.c + * + * This module is required for RSA, DHM and ECC (ECDH, ECDSA) support. + */ +#define MBEDTLS_BIGNUM_C + +/** + * \def MBEDTLS_CAMELLIA_C + * + * Enable the Camellia block cipher. + * + * Module: library/camellia.c + * Caller: library/cipher.c + * + * This module enables the following ciphersuites (if other requisites are + * enabled as well): + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_CBC_SHA256 + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_GCM_SHA384 + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_CBC_SHA384 + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_GCM_SHA256 + * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_CBC_SHA256 + */ +/*#define MBEDTLS_CAMELLIA_C */ + +/** + * \def MBEDTLS_ARIA_C + * + * Enable the ARIA block cipher. + * + * Module: library/aria.c + * Caller: library/cipher.c + * + * This module enables the following ciphersuites (if other requisites are + * enabled as well): + * + * MBEDTLS_TLS_RSA_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_RSA_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_PSK_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_PSK_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_256_CBC_SHA384 + * MBEDTLS_TLS_PSK_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_PSK_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_128_GCM_SHA256 + * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_256_GCM_SHA384 + * MBEDTLS_TLS_ECDHE_PSK_WITH_ARIA_128_CBC_SHA256 + * MBEDTLS_TLS_ECDHE_PSK_WITH_ARIA_256_CBC_SHA384 + */ +/*#define MBEDTLS_ARIA_C */ + +/** + * \def MBEDTLS_CCM_C + * + * Enable the Counter with CBC-MAC (CCM) mode for 128-bit block cipher. + * + * Module: library/ccm.c + * + * Requires: MBEDTLS_AES_C or MBEDTLS_CAMELLIA_C + * + * This module enables the AES-CCM ciphersuites, if other requisites are + * enabled as well. + */ +/*#define MBEDTLS_CCM_C */ + +/** + * \def MBEDTLS_CHACHA20_C + * + * Enable the ChaCha20 stream cipher. + * + * Module: library/chacha20.c + */ +/*#define MBEDTLS_CHACHA20_C */ + +/** + * \def MBEDTLS_CHACHAPOLY_C + * + * Enable the ChaCha20-Poly1305 AEAD algorithm. + * + * Module: library/chachapoly.c + * + * This module requires: MBEDTLS_CHACHA20_C, MBEDTLS_POLY1305_C + */ +/*#define MBEDTLS_CHACHAPOLY_C */ + +/** + * \def MBEDTLS_CIPHER_C + * + * Enable the generic cipher layer. + * + * Module: library/cipher.c + * Caller: library/ssl_tls.c + * + * Uncomment to enable generic cipher wrappers. + */ +#define MBEDTLS_CIPHER_C + +/** + * \def MBEDTLS_CMAC_C + * + * Enable the CMAC (Cipher-based Message Authentication Code) mode for block + * ciphers. + * + * \note When #MBEDTLS_CMAC_ALT is active, meaning that the underlying + * implementation of the CMAC algorithm is provided by an alternate + * implementation, that alternate implementation may opt to not support + * AES-192 or 3DES as underlying block ciphers for the CMAC operation. + * + * Module: library/cmac.c + * + * Requires: MBEDTLS_AES_C or MBEDTLS_DES_C + * + */ +#define MBEDTLS_CMAC_C + +/** + * \def MBEDTLS_CTR_DRBG_C + * + * Enable the CTR_DRBG AES-based random generator. + * The CTR_DRBG generator uses AES-256 by default. + * To use AES-128 instead, enable \c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY above. + * + * \note To achieve a 256-bit security strength with CTR_DRBG, + * you must use AES-256 *and* use sufficient entropy. + * See ctr_drbg.h for more details. + * + * Module: library/ctr_drbg.c + * Caller: + * + * Requires: MBEDTLS_AES_C + * + * This module provides the CTR_DRBG AES random number generator. + */ +#define MBEDTLS_CTR_DRBG_C + +/** + * \def MBEDTLS_DEBUG_C + * + * Enable the debug functions. + * + * Module: library/debug.c + * Caller: library/ssl_cli.c + * library/ssl_srv.c + * library/ssl_tls.c + * + * This module provides debugging functions. + */ +/*#define MBEDTLS_DEBUG_C */ + +/** + * \def MBEDTLS_DES_C + * + * Enable the DES block cipher. + * + * Module: library/des.c + * Caller: library/pem.c + * library/cipher.c + * + * PEM_PARSE uses DES/3DES for decrypting encrypted keys. + * + * \warning DES is considered a weak cipher and its use constitutes a + * security risk. We recommend considering stronger ciphers instead. + */ +/*#define MBEDTLS_DES_C */ + +/** + * \def MBEDTLS_DHM_C + * + * Enable the Diffie-Hellman-Merkle module. + * + * Module: library/dhm.c + * Caller: library/ssl_cli.c + * library/ssl_srv.c + * + * This module is used by the following key exchanges: + * DHE-RSA, DHE-PSK + * + * \warning Using DHE constitutes a security risk as it + * is not possible to validate custom DH parameters. + * If possible, it is recommended users should consider + * preferring other methods of key exchange. + * See dhm.h for more details. + * + */ +/*#define MBEDTLS_DHM_C */ + +/** + * \def MBEDTLS_ECDH_C + * + * Enable the elliptic curve Diffie-Hellman library. + * + * Module: library/ecdh.c + * Caller: library/ssl_cli.c + * library/ssl_srv.c + * + * This module is used by the following key exchanges: + * ECDHE-ECDSA, ECDHE-RSA, DHE-PSK + * + * Requires: MBEDTLS_ECP_C + */ +#define MBEDTLS_ECDH_C + +/** + * \def MBEDTLS_ECDSA_C + * + * Enable the elliptic curve DSA library. + * + * Module: library/ecdsa.c + * Caller: + * + * This module is used by the following key exchanges: + * ECDHE-ECDSA + * + * Requires: MBEDTLS_ECP_C, MBEDTLS_ASN1_WRITE_C, MBEDTLS_ASN1_PARSE_C, + * and at least one MBEDTLS_ECP_DP_XXX_ENABLED for a + * short Weierstrass curve. + */ +#define MBEDTLS_ECDSA_C + +/** + * \def MBEDTLS_ECJPAKE_C + * + * Enable the elliptic curve J-PAKE library. + * + * \note EC J-PAKE support is based on the Thread v1.0.0 specification. + * It has not been reviewed for compliance with newer standards such as + * Thread v1.1 or RFC 8236. + * + * Module: library/ecjpake.c + * Caller: + * + * This module is used by the following key exchanges: + * ECJPAKE + * + * Requires: MBEDTLS_ECP_C, MBEDTLS_MD_C + */ +/*#define MBEDTLS_ECJPAKE_C */ + +/** + * \def MBEDTLS_ECP_C + * + * Enable the elliptic curve over GF(p) library. + * + * Module: library/ecp.c + * Caller: library/ecdh.c + * library/ecdsa.c + * library/ecjpake.c + * + * Requires: MBEDTLS_BIGNUM_C and at least one MBEDTLS_ECP_DP_XXX_ENABLED + */ +#define MBEDTLS_ECP_C + +/** + * \def MBEDTLS_ENTROPY_C + * + * Enable the platform-specific entropy code. + * + * Module: library/entropy.c + * Caller: + * + * Requires: MBEDTLS_SHA512_C or MBEDTLS_SHA256_C + * + * This module provides a generic entropy pool + */ +#define MBEDTLS_ENTROPY_C + +/** + * \def MBEDTLS_ERROR_C + * + * Enable error code to error string conversion. + * + * Module: library/error.c + * Caller: + * + * This module enables mbedtls_strerror(). + */ +/*#define MBEDTLS_ERROR_C */ + +/** + * \def MBEDTLS_GCM_C + * + * Enable the Galois/Counter Mode (GCM). + * + * Module: library/gcm.c + * + * Requires: MBEDTLS_AES_C or MBEDTLS_CAMELLIA_C or MBEDTLS_ARIA_C + * + * This module enables the AES-GCM and CAMELLIA-GCM ciphersuites, if other + * requisites are enabled as well. + */ +#define MBEDTLS_GCM_C + +/** + * \def MBEDTLS_HKDF_C + * + * Enable the HKDF algorithm (RFC 5869). + * + * Module: library/hkdf.c + * Caller: + * + * Requires: MBEDTLS_MD_C + * + * This module adds support for the Hashed Message Authentication Code + * (HMAC)-based key derivation function (HKDF). + */ +/*#define MBEDTLS_HKDF_C */ + +/** + * \def MBEDTLS_HMAC_DRBG_C + * + * Enable the HMAC_DRBG random generator. + * + * Module: library/hmac_drbg.c + * Caller: + * + * Requires: MBEDTLS_MD_C + * + * Uncomment to enable the HMAC_DRBG random number generator. + */ +/*#define MBEDTLS_HMAC_DRBG_C */ + +/** + * \def MBEDTLS_NIST_KW_C + * + * Enable the Key Wrapping mode for 128-bit block ciphers, + * as defined in NIST SP 800-38F. Only KW and KWP modes + * are supported. At the moment, only AES is approved by NIST. + * + * Module: library/nist_kw.c + * + * Requires: MBEDTLS_AES_C and MBEDTLS_CIPHER_C + */ +/*#define MBEDTLS_NIST_KW_C */ + +/** + * \def MBEDTLS_MD_C + * + * Enable the generic message digest layer. + * + * Module: library/md.c + * Caller: + * + * Uncomment to enable generic message digest wrappers. + */ +#define MBEDTLS_MD_C + +/** + * \def MBEDTLS_MD5_C + * + * Enable the MD5 hash algorithm. + * + * Module: library/md5.c + * Caller: library/md.c + * library/pem.c + * library/ssl_tls.c + * + * This module is required for TLS 1.2 depending on the handshake parameters. + * Further, it is used for checking MD5-signed certificates, and for PBKDF1 + * when decrypting PEM-encoded encrypted keys. + * + * \warning MD5 is considered a weak message digest and its use constitutes a + * security risk. If possible, we recommend avoiding dependencies on + * it, and considering stronger message digests instead. + * + */ +/*#define MBEDTLS_MD5_C */ + +/** + * \def MBEDTLS_MEMORY_BUFFER_ALLOC_C + * + * Enable the buffer allocator implementation that makes use of a (stack) + * based buffer to 'allocate' dynamic memory. (replaces calloc() and free() + * calls) + * + * Module: library/memory_buffer_alloc.c + * + * Requires: MBEDTLS_PLATFORM_C + * MBEDTLS_PLATFORM_MEMORY (to use it within mbed TLS) + * + * Enable this module to enable the buffer memory allocator. + */ +/*#define MBEDTLS_MEMORY_BUFFER_ALLOC_C */ + +/** + * \def MBEDTLS_NET_C + * + * Enable the TCP and UDP over IPv6/IPv4 networking routines. + * + * \note This module only works on POSIX/Unix (including Linux, BSD and OS X) + * and Windows. For other platforms, you'll want to disable it, and write your + * own networking callbacks to be passed to \c mbedtls_ssl_set_bio(). + * + * \note See also our Knowledge Base article about porting to a new + * environment: + * https://tls.mbed.org/kb/how-to/how-do-i-port-mbed-tls-to-a-new-environment-OS + * + * Module: library/net_sockets.c + * + * This module provides networking routines. + */ +/*#define MBEDTLS_NET_C */ + +/** + * \def MBEDTLS_OID_C + * + * Enable the OID database. + * + * Module: library/oid.c + * Caller: library/asn1write.c + * library/pkcs5.c + * library/pkparse.c + * library/pkwrite.c + * library/rsa.c + * library/x509.c + * library/x509_create.c + * library/x509_crl.c + * library/x509_crt.c + * library/x509_csr.c + * library/x509write_crt.c + * library/x509write_csr.c + * + * This modules translates between OIDs and internal values. + */ +#define MBEDTLS_OID_C + +/** + * \def MBEDTLS_PADLOCK_C + * + * Enable VIA Padlock support on x86. + * + * Module: library/padlock.c + * Caller: library/aes.c + * + * Requires: MBEDTLS_HAVE_ASM + * + * This modules adds support for the VIA PadLock on x86. + */ +/*#define MBEDTLS_PADLOCK_C */ + +/** + * \def MBEDTLS_PEM_PARSE_C + * + * Enable PEM decoding / parsing. + * + * Module: library/pem.c + * Caller: library/dhm.c + * library/pkparse.c + * library/x509_crl.c + * library/x509_crt.c + * library/x509_csr.c + * + * Requires: MBEDTLS_BASE64_C + * + * This modules adds support for decoding / parsing PEM files. + */ +#define MBEDTLS_PEM_PARSE_C + +/** + * \def MBEDTLS_PEM_WRITE_C + * + * Enable PEM encoding / writing. + * + * Module: library/pem.c + * Caller: library/pkwrite.c + * library/x509write_crt.c + * library/x509write_csr.c + * + * Requires: MBEDTLS_BASE64_C + * + * This modules adds support for encoding / writing PEM files. + */ +/*#define MBEDTLS_PEM_WRITE_C */ + +/** + * \def MBEDTLS_PK_C + * + * Enable the generic public (asymmetric) key layer. + * + * Module: library/pk.c + * Caller: library/ssl_tls.c + * library/ssl_cli.c + * library/ssl_srv.c + * + * Requires: MBEDTLS_RSA_C or MBEDTLS_ECP_C + * + * Uncomment to enable generic public key wrappers. + */ +#define MBEDTLS_PK_C + +/** + * \def MBEDTLS_PK_PARSE_C + * + * Enable the generic public (asymmetric) key parser. + * + * Module: library/pkparse.c + * Caller: library/x509_crt.c + * library/x509_csr.c + * + * Requires: MBEDTLS_PK_C + * + * Uncomment to enable generic public key parse functions. + */ +#define MBEDTLS_PK_PARSE_C + +/** + * \def MBEDTLS_PK_WRITE_C + * + * Enable the generic public (asymmetric) key writer. + * + * Module: library/pkwrite.c + * Caller: library/x509write.c + * + * Requires: MBEDTLS_PK_C + * + * Uncomment to enable generic public key write functions. + */ +#define MBEDTLS_PK_WRITE_C + +/** + * \def MBEDTLS_PKCS5_C + * + * Enable PKCS#5 functions. + * + * Module: library/pkcs5.c + * + * Requires: MBEDTLS_MD_C + * + * This module adds support for the PKCS#5 functions. + */ +/*#define MBEDTLS_PKCS5_C */ + +/** + * \def MBEDTLS_PKCS12_C + * + * Enable PKCS#12 PBE functions. + * Adds algorithms for parsing PKCS#8 encrypted private keys + * + * Module: library/pkcs12.c + * Caller: library/pkparse.c + * + * Requires: MBEDTLS_ASN1_PARSE_C, MBEDTLS_CIPHER_C, MBEDTLS_MD_C + * + * This module enables PKCS#12 functions. + */ +/*#define MBEDTLS_PKCS12_C */ + +/** + * \def MBEDTLS_PLATFORM_C + * + * Enable the platform abstraction layer that allows you to re-assign + * functions like calloc(), free(), snprintf(), printf(), fprintf(), exit(). + * + * Enabling MBEDTLS_PLATFORM_C enables to use of MBEDTLS_PLATFORM_XXX_ALT + * or MBEDTLS_PLATFORM_XXX_MACRO directives, allowing the functions mentioned + * above to be specified at runtime or compile time respectively. + * + * \note This abstraction layer must be enabled on Windows (including MSYS2) + * as other module rely on it for a fixed snprintf implementation. + * + * Module: library/platform.c + * Caller: Most other .c files + * + * This module enables abstraction of common (libc) functions. + */ +#define MBEDTLS_PLATFORM_C + +/** + * \def MBEDTLS_POLY1305_C + * + * Enable the Poly1305 MAC algorithm. + * + * Module: library/poly1305.c + * Caller: library/chachapoly.c + */ +/*#define MBEDTLS_POLY1305_C */ + +/** + * \def MBEDTLS_PSA_CRYPTO_C + * + * Enable the Platform Security Architecture cryptography API. + * + * \warning The PSA Crypto API is still beta status. While you're welcome to + * experiment using it, incompatible API changes are still possible, and some + * parts may not have reached the same quality as the rest of Mbed TLS yet. + * + * Module: library/psa_crypto.c + * + * Requires: either MBEDTLS_CTR_DRBG_C and MBEDTLS_ENTROPY_C, + * or MBEDTLS_HMAC_DRBG_C and MBEDTLS_ENTROPY_C, + * or MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG. + * + */ +/*#define MBEDTLS_PSA_CRYPTO_C */ + +/** + * \def MBEDTLS_PSA_CRYPTO_SE_C + * + * Enable secure element support in the Platform Security Architecture + * cryptography API. + * + * \warning This feature is not yet suitable for production. It is provided + * for API evaluation and testing purposes only. + * + * Module: library/psa_crypto_se.c + * + * Requires: MBEDTLS_PSA_CRYPTO_C, MBEDTLS_PSA_CRYPTO_STORAGE_C + * + */ +/*#define MBEDTLS_PSA_CRYPTO_SE_C */ + +/** + * \def MBEDTLS_PSA_CRYPTO_STORAGE_C + * + * Enable the Platform Security Architecture persistent key storage. + * + * Module: library/psa_crypto_storage.c + * + * Requires: MBEDTLS_PSA_CRYPTO_C, + * either MBEDTLS_PSA_ITS_FILE_C or a native implementation of + * the PSA ITS interface + */ +/*#define MBEDTLS_PSA_CRYPTO_STORAGE_C */ + +/** + * \def MBEDTLS_PSA_ITS_FILE_C + * + * Enable the emulation of the Platform Security Architecture + * Internal Trusted Storage (PSA ITS) over files. + * + * Module: library/psa_its_file.c + * + * Requires: MBEDTLS_FS_IO + */ +/*#define MBEDTLS_PSA_ITS_FILE_C */ + +/** + * \def MBEDTLS_RIPEMD160_C + * + * Enable the RIPEMD-160 hash algorithm. + * + * Module: library/ripemd160.c + * Caller: library/md.c + * + */ +/*#define MBEDTLS_RIPEMD160_C */ + +/** + * \def MBEDTLS_RSA_C + * + * Enable the RSA public-key cryptosystem. + * + * Module: library/rsa.c + * library/rsa_alt_helpers.c + * Caller: library/ssl_cli.c + * library/ssl_srv.c + * library/ssl_tls.c + * library/x509.c + * + * This module is used by the following key exchanges: + * RSA, DHE-RSA, ECDHE-RSA, RSA-PSK + * + * Requires: MBEDTLS_BIGNUM_C, MBEDTLS_OID_C + */ +#define MBEDTLS_RSA_C + +/** + * \def MBEDTLS_SHA1_C + * + * Enable the SHA1 cryptographic hash algorithm. + * + * Module: library/sha1.c + * Caller: library/md.c + * library/ssl_cli.c + * library/ssl_srv.c + * library/ssl_tls.c + * library/x509write_crt.c + * + * This module is required for TLS 1.2 depending on the handshake parameters, + * and for SHA1-signed certificates. + * + * \warning SHA-1 is considered a weak message digest and its use constitutes + * a security risk. If possible, we recommend avoiding dependencies + * on it, and considering stronger message digests instead. + * + */ +#define MBEDTLS_SHA1_C + +/** + * \def MBEDTLS_SHA224_C + * + * Enable the SHA-224 cryptographic hash algorithm. + * + * Requires: MBEDTLS_SHA256_C. The library does not currently support enabling + * SHA-224 without SHA-256. + * + * Module: library/sha256.c + * Caller: library/md.c + * library/ssl_cookie.c + * + * This module adds support for SHA-224. + */ +/*#define MBEDTLS_SHA224_C */ + +/** + * \def MBEDTLS_SHA256_C + * + * Enable the SHA-256 cryptographic hash algorithm. + * + * Requires: MBEDTLS_SHA224_C. The library does not currently support enabling + * SHA-256 without SHA-224. + * + * Module: library/sha256.c + * Caller: library/entropy.c + * library/md.c + * library/ssl_cli.c + * library/ssl_srv.c + * library/ssl_tls.c + * + * This module adds support for SHA-256. + * This module is required for the SSL/TLS 1.2 PRF function. + */ +#define MBEDTLS_SHA256_C + +/** + * \def MBEDTLS_SHA384_C + * + * Enable the SHA-384 cryptographic hash algorithm. + * + * Requires: MBEDTLS_SHA512_C + * + * Module: library/sha512.c + * Caller: library/md.c + * library/ssl_cli.c + * library/ssl_srv.c + * + * Comment to disable SHA-384 + */ +/*#define MBEDTLS_SHA384_C */ + +/** + * \def MBEDTLS_SHA512_C + * + * Enable SHA-512 cryptographic hash algorithms. + * + * Module: library/sha512.c + * Caller: library/entropy.c + * library/md.c + * library/ssl_tls.c + * library/ssl_cookie.c + * + * This module adds support for SHA-512. + */ +/*#define MBEDTLS_SHA512_C */ + +/** + * \def MBEDTLS_SSL_CACHE_C + * + * Enable simple SSL cache implementation. + * + * Module: library/ssl_cache.c + * Caller: + * + * Requires: MBEDTLS_SSL_CACHE_C + */ +/*#define MBEDTLS_SSL_CACHE_C */ + +/** + * \def MBEDTLS_SSL_COOKIE_C + * + * Enable basic implementation of DTLS cookies for hello verification. + * + * Module: library/ssl_cookie.c + * Caller: + */ +/*#define MBEDTLS_SSL_COOKIE_C */ + +/** + * \def MBEDTLS_SSL_TICKET_C + * + * Enable an implementation of TLS server-side callbacks for session tickets. + * + * Module: library/ssl_ticket.c + * Caller: + * + * Requires: MBEDTLS_CIPHER_C + */ +/*#define MBEDTLS_SSL_TICKET_C */ + +/** + * \def MBEDTLS_SSL_CLI_C + * + * Enable the SSL/TLS client code. + * + * Module: library/ssl_cli.c + * Caller: + * + * Requires: MBEDTLS_SSL_TLS_C + * + * This module is required for SSL/TLS client support. + */ +#define MBEDTLS_SSL_CLI_C + +/** + * \def MBEDTLS_SSL_SRV_C + * + * Enable the SSL/TLS server code. + * + * Module: library/ssl_srv.c + * Caller: + * + * Requires: MBEDTLS_SSL_TLS_C + * + * This module is required for SSL/TLS server support. + */ +/*#define MBEDTLS_SSL_SRV_C */ + +/** + * \def MBEDTLS_SSL_TLS_C + * + * Enable the generic SSL/TLS code. + * + * Module: library/ssl_tls.c + * Caller: library/ssl_cli.c + * library/ssl_srv.c + * + * Requires: MBEDTLS_CIPHER_C, MBEDTLS_MD_C + * and at least one of the MBEDTLS_SSL_PROTO_XXX defines + * + * This module is required for SSL/TLS. + */ +#define MBEDTLS_SSL_TLS_C + +/** + * \def MBEDTLS_THREADING_C + * + * Enable the threading abstraction layer. + * By default mbed TLS assumes it is used in a non-threaded environment or that + * contexts are not shared between threads. If you do intend to use contexts + * between threads, you will need to enable this layer to prevent race + * conditions. See also our Knowledge Base article about threading: + * https://tls.mbed.org/kb/development/thread-safety-and-multi-threading + * + * Module: library/threading.c + * + * This allows different threading implementations (self-implemented or + * provided). + * + * You will have to enable either MBEDTLS_THREADING_ALT or + * MBEDTLS_THREADING_PTHREAD. + * + * Enable this layer to allow use of mutexes within mbed TLS + */ +#define MBEDTLS_THREADING_C + +/** + * \def MBEDTLS_TIMING_C + * + * Enable the semi-portable timing interface. + * + * \note The provided implementation only works on POSIX/Unix (including Linux, + * BSD and OS X) and Windows. On other platforms, you can either disable that + * module and provide your own implementations of the callbacks needed by + * \c mbedtls_ssl_set_timer_cb() for DTLS, or leave it enabled and provide + * your own implementation of the whole module by setting + * \c MBEDTLS_TIMING_ALT in the current file. + * + * \note See also our Knowledge Base article about porting to a new + * environment: + * https://tls.mbed.org/kb/how-to/how-do-i-port-mbed-tls-to-a-new-environment-OS + * + * Module: library/timing.c + */ +/*#define MBEDTLS_TIMING_C */ + +/** + * \def MBEDTLS_VERSION_C + * + * Enable run-time version information. + * + * Module: library/version.c + * + * This module provides run-time version information. + */ +/*#define MBEDTLS_VERSION_C */ + +/** + * \def MBEDTLS_X509_USE_C + * + * Enable X.509 core for using certificates. + * + * Module: library/x509.c + * Caller: library/x509_crl.c + * library/x509_crt.c + * library/x509_csr.c + * + * Requires: MBEDTLS_ASN1_PARSE_C, MBEDTLS_BIGNUM_C, MBEDTLS_OID_C, + * MBEDTLS_PK_PARSE_C + * + * This module is required for the X.509 parsing modules. + */ +#define MBEDTLS_X509_USE_C + +/** + * \def MBEDTLS_X509_CRT_PARSE_C + * + * Enable X.509 certificate parsing. + * + * Module: library/x509_crt.c + * Caller: library/ssl_cli.c + * library/ssl_srv.c + * library/ssl_tls.c + * + * Requires: MBEDTLS_X509_USE_C + * + * This module is required for X.509 certificate parsing. + */ +#define MBEDTLS_X509_CRT_PARSE_C + +/** + * \def MBEDTLS_X509_CRL_PARSE_C + * + * Enable X.509 CRL parsing. + * + * Module: library/x509_crl.c + * Caller: library/x509_crt.c + * + * Requires: MBEDTLS_X509_USE_C + * + * This module is required for X.509 CRL parsing. + */ +/*#define MBEDTLS_X509_CRL_PARSE_C */ + +/** + * \def MBEDTLS_X509_CSR_PARSE_C + * + * Enable X.509 Certificate Signing Request (CSR) parsing. + * + * Module: library/x509_csr.c + * Caller: library/x509_crt_write.c + * + * Requires: MBEDTLS_X509_USE_C + * + * This module is used for reading X.509 certificate request. + */ +/*#define MBEDTLS_X509_CSR_PARSE_C */ + +/** + * \def MBEDTLS_X509_CREATE_C + * + * Enable X.509 core for creating certificates. + * + * Module: library/x509_create.c + * + * Requires: MBEDTLS_BIGNUM_C, MBEDTLS_OID_C, MBEDTLS_PK_WRITE_C + * + * This module is the basis for creating X.509 certificates and CSRs. + */ +/*#define MBEDTLS_X509_CREATE_C */ + +/** + * \def MBEDTLS_X509_CRT_WRITE_C + * + * Enable creating X.509 certificates. + * + * Module: library/x509_crt_write.c + * + * Requires: MBEDTLS_X509_CREATE_C + * + * This module is required for X.509 certificate creation. + */ +/*#define MBEDTLS_X509_CRT_WRITE_C */ + +/** + * \def MBEDTLS_X509_CSR_WRITE_C + * + * Enable creating X.509 Certificate Signing Requests (CSR). + * + * Module: library/x509_csr_write.c + * + * Requires: MBEDTLS_X509_CREATE_C + * + * This module is required for X.509 certificate request writing. + */ +/*#define MBEDTLS_X509_CSR_WRITE_C */ + +/* \} name SECTION: mbed TLS modules */ + +/** + * \name SECTION: Module configuration options + * + * This section allows for the setting of module specific sizes and + * configuration options. The default values are already present in the + * relevant header files and should suffice for the regular use cases. + * + * Our advice is to enable options and change their values here + * only if you have a good reason and know the consequences. + * + * Please check the respective header file for documentation on these + * parameters (to prevent duplicate documentation). + * \{ + */ + +/* MPI / BIGNUM options */ +/*#define MBEDTLS_MPI_WINDOW_SIZE 6 / **< Maximum window size used. * / */ +/*#define MBEDTLS_MPI_MAX_SIZE 1024 / **< Maximum number of bytes for usable MPIs. * / */ + +/* CTR_DRBG options */ +/*#define MBEDTLS_CTR_DRBG_ENTROPY_LEN 48 / **< Amount of entropy used per seed by default (48 with SHA-512, 32 with SHA-256) * / */ +/*#define MBEDTLS_CTR_DRBG_RESEED_INTERVAL 10000 / **< Interval before reseed is performed by default * / */ +/*#define MBEDTLS_CTR_DRBG_MAX_INPUT 256 / **< Maximum number of additional input bytes * / */ +/*#define MBEDTLS_CTR_DRBG_MAX_REQUEST 1024 / **< Maximum number of requested bytes per call * / */ +/*#define MBEDTLS_CTR_DRBG_MAX_SEED_INPUT 384 / **< Maximum size of (re)seed buffer * / */ + +/* HMAC_DRBG options */ +/*#define MBEDTLS_HMAC_DRBG_RESEED_INTERVAL 10000 / **< Interval before reseed is performed by default * / */ +/*#define MBEDTLS_HMAC_DRBG_MAX_INPUT 256 / **< Maximum number of additional input bytes * / */ +/*#define MBEDTLS_HMAC_DRBG_MAX_REQUEST 1024 / **< Maximum number of requested bytes per call * / */ +/*#define MBEDTLS_HMAC_DRBG_MAX_SEED_INPUT 384 / **< Maximum size of (re)seed buffer * / */ + +/* ECP options */ +/*#define MBEDTLS_ECP_WINDOW_SIZE 4 / **< Maximum window size used * / */ +/*#define MBEDTLS_ECP_FIXED_POINT_OPTIM 1 / **< Enable fixed-point speed-up * / */ + +/* Entropy options */ +/*#define MBEDTLS_ENTROPY_MAX_SOURCES 20 / **< Maximum number of sources supported * / */ +/*#define MBEDTLS_ENTROPY_MAX_GATHER 128 / **< Maximum amount requested from entropy sources * / */ +/*#define MBEDTLS_ENTROPY_MIN_HARDWARE 32 / **< Default minimum number of bytes required for the hardware entropy source mbedtls_hardware_poll() before entropy is released * / */ + +/* Memory buffer allocator options */ +/*#define MBEDTLS_MEMORY_ALIGN_MULTIPLE 4 / **< Align on multiples of this value * / */ + +/* Platform options */ +/*#define MBEDTLS_PLATFORM_STD_MEM_HDR / **< Header to include if MBEDTLS_PLATFORM_NO_STD_FUNCTIONS is defined. Don't define if no header is needed. * / */ +/*#define MBEDTLS_PLATFORM_STD_CALLOC calloc / **< Default allocator to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_FREE free / **< Default free to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_EXIT exit / **< Default exit to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_TIME time / **< Default time to use, can be undefined. MBEDTLS_HAVE_TIME must be enabled * / */ +/*#define MBEDTLS_PLATFORM_STD_FPRINTF fprintf / **< Default fprintf to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_PRINTF printf / **< Default printf to use, can be undefined * / */ +/* Note: your snprintf must correctly zero-terminate the buffer! */ +/*#define MBEDTLS_PLATFORM_STD_SNPRINTF snprintf / **< Default snprintf to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_EXIT_SUCCESS 0 / **< Default exit value to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_EXIT_FAILURE 1 / **< Default exit value to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_NV_SEED_READ mbedtls_platform_std_nv_seed_read / **< Default nv_seed_read function to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_NV_SEED_WRITE mbedtls_platform_std_nv_seed_write / **< Default nv_seed_write function to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_STD_NV_SEED_FILE "seedfile" / **< Seed file to read/write with default implementation * / */ + +/* To Use Function Macros MBEDTLS_PLATFORM_C must be enabled */ +/* MBEDTLS_PLATFORM_XXX_MACRO and MBEDTLS_PLATFORM_XXX_ALT cannot both be defined */ +/*#define MBEDTLS_PLATFORM_CALLOC_MACRO mbedtls_platform_calloc / **< Default allocator macro to use, can be undefined */ +/*#define MBEDTLS_PLATFORM_FREE_MACRO mbedtls_platform_free / **< Default free macro to use, can be undefined */ +/*#define MBEDTLS_PLATFORM_EXIT_MACRO exit / **< Default exit macro to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_TIME_MACRO time / **< Default time macro to use, can be undefined. MBEDTLS_HAVE_TIME must be enabled * / */ +/*#define MBEDTLS_PLATFORM_TIME_TYPE_MACRO time_t / **< Default time macro to use, can be undefined. MBEDTLS_HAVE_TIME must be enabled * / */ +/*#define MBEDTLS_PLATFORM_FPRINTF_MACRO fprintf / **< Default fprintf macro to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_PRINTF_MACRO printf / **< Default printf macro to use, can be undefined * / */ +/* Note: your snprintf must correctly zero-terminate the buffer! */ +/*#define MBEDTLS_PLATFORM_SNPRINTF_MACRO snprintf / **< Default snprintf macro to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_VSNPRINTF_MACRO vsnprintf / **< Default vsnprintf macro to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_NV_SEED_READ_MACRO mbedtls_platform_std_nv_seed_read / **< Default nv_seed_read function to use, can be undefined * / */ +/*#define MBEDTLS_PLATFORM_NV_SEED_WRITE_MACRO mbedtls_platform_std_nv_seed_write / **< Default nv_seed_write function to use, can be undefined * / */ + +/* PSA options */ + +/** + * Use HMAC_DRBG with the specified hash algorithm for HMAC_DRBG for the + * PSA crypto subsystem. + * + * If this option is unset: + * - If CTR_DRBG is available, the PSA subsystem uses it rather than HMAC_DRBG. + * - Otherwise, the PSA subsystem uses HMAC_DRBG with either + * #MBEDTLS_MD_SHA512 or #MBEDTLS_MD_SHA256 based on availability and + * on unspecified heuristics. + */ +/*#define MBEDTLS_PSA_HMAC_DRBG_MD_TYPE MBEDTLS_MD_SHA256 */ + +/** \def MBEDTLS_PSA_KEY_SLOT_COUNT + * Restrict the PSA library to supporting a maximum amount of simultaneously + * loaded keys. A loaded key is a key stored by the PSA Crypto core as a + * volatile key, or a persistent key which is loaded temporarily by the + * library as part of a crypto operation in flight. + * + * If this option is unset, the library will fall back to a default value of + * 32 keys. + */ +/*#define MBEDTLS_PSA_KEY_SLOT_COUNT 16 */ + +/* SSL Cache options */ +/*#define MBEDTLS_SSL_CACHE_DEFAULT_TIMEOUT 86400 / **< 1 day * / */ +/*#define MBEDTLS_SSL_CACHE_DEFAULT_MAX_ENTRIES 50 / **< Maximum entries in cache * / */ + +/* SSL options */ + +/** \def MBEDTLS_SSL_IN_CONTENT_LEN + * + * Maximum length (in bytes) of incoming plaintext fragments. + * + * This determines the size of the incoming TLS I/O buffer in such a way + * that it is capable of holding the specified amount of plaintext data, + * regardless of the protection mechanism used. + * + * \note When using a value less than the default of 16KB on the client, it is + * recommended to use the Maximum Fragment Length (MFL) extension to + * inform the server about this limitation. On the server, there + * is no supported, standardized way of informing the client about + * restriction on the maximum size of incoming messages, and unless + * the limitation has been communicated by other means, it is recommended + * to only change the outgoing buffer size #MBEDTLS_SSL_OUT_CONTENT_LEN + * while keeping the default value of 16KB for the incoming buffer. + * + * Uncomment to set the maximum plaintext size of the incoming I/O buffer. + */ +/*#define MBEDTLS_SSL_IN_CONTENT_LEN 16384 */ + +/** \def MBEDTLS_SSL_CID_IN_LEN_MAX + * + * The maximum length of CIDs used for incoming DTLS messages. + * + */ +/*#define MBEDTLS_SSL_CID_IN_LEN_MAX 32 */ + +/** \def MBEDTLS_SSL_CID_OUT_LEN_MAX + * + * The maximum length of CIDs used for outgoing DTLS messages. + * + */ +/*#define MBEDTLS_SSL_CID_OUT_LEN_MAX 32 */ + +/** \def MBEDTLS_SSL_CID_TLS1_3_PADDING_GRANULARITY + * + * This option controls the use of record plaintext padding + * in TLS 1.3 and when using the Connection ID extension in DTLS 1.2. + * + * The padding will always be chosen so that the length of the + * padded plaintext is a multiple of the value of this option. + * + * Note: A value of \c 1 means that no padding will be used + * for outgoing records. + * + * Note: On systems lacking division instructions, + * a power of two should be preferred. + */ +/*#define MBEDTLS_SSL_CID_TLS1_3_PADDING_GRANULARITY 16 */ + +/** \def MBEDTLS_SSL_OUT_CONTENT_LEN + * + * Maximum length (in bytes) of outgoing plaintext fragments. + * + * This determines the size of the outgoing TLS I/O buffer in such a way + * that it is capable of holding the specified amount of plaintext data, + * regardless of the protection mechanism used. + * + * It is possible to save RAM by setting a smaller outward buffer, while keeping + * the default inward 16384 byte buffer to conform to the TLS specification. + * + * The minimum required outward buffer size is determined by the handshake + * protocol's usage. Handshaking will fail if the outward buffer is too small. + * The specific size requirement depends on the configured ciphers and any + * certificate data which is sent during the handshake. + * + * Uncomment to set the maximum plaintext size of the outgoing I/O buffer. + */ +/*#define MBEDTLS_SSL_OUT_CONTENT_LEN 16384 */ + +/** \def MBEDTLS_SSL_DTLS_MAX_BUFFERING + * + * Maximum number of heap-allocated bytes for the purpose of + * DTLS handshake message reassembly and future message buffering. + * + * This should be at least 9/8 * MBEDTLSSL_IN_CONTENT_LEN + * to account for a reassembled handshake message of maximum size, + * together with its reassembly bitmap. + * + * A value of 2 * MBEDTLS_SSL_IN_CONTENT_LEN (32768 by default) + * should be sufficient for all practical situations as it allows + * to reassembly a large handshake message (such as a certificate) + * while buffering multiple smaller handshake messages. + * + */ +/*#define MBEDTLS_SSL_DTLS_MAX_BUFFERING 32768 */ + +/*#define MBEDTLS_PSK_MAX_LEN 32 / **< Max size of TLS pre-shared keys, in bytes (default 256 bits) * / */ +/*#define MBEDTLS_SSL_COOKIE_TIMEOUT 60 / **< Default expiration delay of DTLS cookies, in seconds if HAVE_TIME, or in number of cookies issued * / */ + +/** + * Complete list of ciphersuites to use, in order of preference. + * + * \warning No dependency checking is done on that field! This option can only + * be used to restrict the set of available ciphersuites. It is your + * responsibility to make sure the needed modules are active. + * + * Use this to save a few hundred bytes of ROM (default ordering of all + * available ciphersuites) and a few to a few hundred bytes of RAM. + * + * The value below is only an example, not the default. + */ +/*#define MBEDTLS_SSL_CIPHERSUITES MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384,MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 */ + +/* X509 options */ +/*#define MBEDTLS_X509_MAX_INTERMEDIATE_CA 8 / **< Maximum number of intermediate CAs in a verification chain. * / */ +/*#define MBEDTLS_X509_MAX_FILE_PATH_LEN 512 / **< Maximum length of a path/filename string in bytes including the null terminator character ('\0'). * / */ + +/** + * Uncomment the macro to let mbed TLS use your alternate implementation of + * mbedtls_platform_zeroize(). This replaces the default implementation in + * platform_util.c. + * + * mbedtls_platform_zeroize() is a widely used function across the library to + * zero a block of memory. The implementation is expected to be secure in the + * sense that it has been written to prevent the compiler from removing calls + * to mbedtls_platform_zeroize() as part of redundant code elimination + * optimizations. However, it is difficult to guarantee that calls to + * mbedtls_platform_zeroize() will not be optimized by the compiler as older + * versions of the C language standards do not provide a secure implementation + * of memset(). Therefore, MBEDTLS_PLATFORM_ZEROIZE_ALT enables users to + * configure their own implementation of mbedtls_platform_zeroize(), for + * example by using directives specific to their compiler, features from newer + * C standards (e.g using memset_s() in C11) or calling a secure memset() from + * their system (e.g explicit_bzero() in BSD). + */ +/*#define MBEDTLS_PLATFORM_ZEROIZE_ALT */ + +/** + * Uncomment the macro to let Mbed TLS use your alternate implementation of + * mbedtls_platform_gmtime_r(). This replaces the default implementation in + * platform_util.c. + * + * gmtime() is not a thread-safe function as defined in the C standard. The + * library will try to use safer implementations of this function, such as + * gmtime_r() when available. However, if Mbed TLS cannot identify the target + * system, the implementation of mbedtls_platform_gmtime_r() will default to + * using the standard gmtime(). In this case, calls from the library to + * gmtime() will be guarded by the global mutex mbedtls_threading_gmtime_mutex + * if MBEDTLS_THREADING_C is enabled. We recommend that calls from outside the + * library are also guarded with this mutex to avoid race conditions. However, + * if the macro MBEDTLS_PLATFORM_GMTIME_R_ALT is defined, Mbed TLS will + * unconditionally use the implementation for mbedtls_platform_gmtime_r() + * supplied at compile time. + */ +/*#define MBEDTLS_PLATFORM_GMTIME_R_ALT */ + +/** + * Enable the verified implementations of ECDH primitives from Project Everest + * (currently only Curve25519). This feature changes the layout of ECDH + * contexts and therefore is a compatibility break for applications that access + * fields of a mbedtls_ecdh_context structure directly. See also + * MBEDTLS_ECDH_LEGACY_CONTEXT in include/mbedtls/ecdh.h. + */ +/*#define MBEDTLS_ECDH_VARIANT_EVEREST_ENABLED */ + +/* \} name SECTION: Customisation configuration options */ + +/*#define MBEDTLS_PSA_CRYPTO_C */ + +/*#undef MBEDTLS_PSA_CRYPTO_C */ diff --git a/applications/object_detection/configs/mbedtls_config/threading_alt.h b/applications/object_detection/configs/mbedtls_config/threading_alt.h new file mode 100644 index 00000000..1b177933 --- /dev/null +++ b/applications/object_detection/configs/mbedtls_config/threading_alt.h @@ -0,0 +1,61 @@ +/* + * FreeRTOS V202112.00 + * Copyright (C) 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy of + * this software and associated documentation files (the "Software"), to deal in + * the Software without restriction, including without limitation the rights to + * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + * the Software, and to permit persons to whom the Software is furnished to do so, + * subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in all + * copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + * + * https://www.FreeRTOS.org + * https://github.com/FreeRTOS + * + */ + +/** + * @file threading_alt.h + * @brief mbed TLS threading functions implemented for FreeRTOS. + */ + + +#ifndef MBEDTLS_THREADING_ALT_H_ +#define MBEDTLS_THREADING_ALT_H_ + +/* FreeRTOS includes. */ +#include "FreeRTOS.h" +#include "semphr.h" + +/** + * @brief mbed TLS mutex type. + * + * mbed TLS requires platform specific definition for the mutext type. Defining the type for + * FreeRTOS with FreeRTOS semaphore + * handle and semaphore storage as members. + */ +typedef struct mbedtls_threading_mutex +{ + SemaphoreHandle_t mutexHandle; + StaticSemaphore_t mutexStorage; +} mbedtls_threading_mutex_t; + +/* mbed TLS mutex functions. */ +void mbedtls_platform_mutex_init( mbedtls_threading_mutex_t * pMutex ); +void mbedtls_platform_mutex_free( mbedtls_threading_mutex_t * pMutex ); +int mbedtls_platform_mutex_lock( mbedtls_threading_mutex_t * pMutex ); +int mbedtls_platform_mutex_unlock( mbedtls_threading_mutex_t * pMutex ); + + + +#endif /* ifndef MBEDTLS_THREADING_ALT_H_ */ diff --git a/applications/object_detection/configs/tfm_config/project_config.h b/applications/object_detection/configs/tfm_config/project_config.h new file mode 100644 index 00000000..86414f75 --- /dev/null +++ b/applications/object_detection/configs/tfm_config/project_config.h @@ -0,0 +1,86 @@ +/* Copyright 2023-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: BSD-3-Clause + */ + +#define PLATFORM_SERVICE_INPUT_BUFFER_SIZE 64 + +#define PLATFORM_SERVICE_OUTPUT_BUFFER_SIZE 64 + +#define PLATFORM_SP_STACK_SIZE 0x500 + +#define PLATFORM_NV_COUNTER_MODULE_DISABLED 0 + +#define CRYPTO_ENGINE_BUF_SIZE 0x8000 + +#define CRYPTO_CONC_OPER_NUM 8 + +#define CRYPTO_RNG_MODULE_ENABLED 1 + +#define CRYPTO_KEY_MODULE_ENABLED 1 + +#define CRYPTO_AEAD_MODULE_ENABLED 1 + +#define CRYPTO_MAC_MODULE_ENABLED 1 + +#define CRYPTO_HASH_MODULE_ENABLED 1 + +#define CRYPTO_CIPHER_MODULE_ENABLED 1 + +#define CRYPTO_ASYM_SIGN_MODULE_ENABLED 1 + +#define CRYPTO_ASYM_ENCRYPT_MODULE_ENABLED 1 + +#define CRYPTO_KEY_DERIVATION_MODULE_ENABLED 1 + +#define CRYPTO_IOVEC_BUFFER_SIZE 5120 + +#define CRYPTO_NV_SEED 1 + +#define CRYPTO_SINGLE_PART_FUNCS_DISABLED 0 + +#define CRYPTO_STACK_SIZE 0x1B00 + +#define TFM_FWU_BUF_SIZE PSA_FWU_MAX_WRITE_SIZE + +#define FWU_STACK_SIZE 0x600 + +#define ATTEST_INCLUDE_OPTIONAL_CLAIMS 1 + +#define ATTEST_INCLUDE_COSE_KEY_ID 0 + +#define ATTEST_STACK_SIZE 0x700 + +#define ATTEST_TOKEN_PROFILE_PSA_IOT_1 1 + +#define ITS_CREATE_FLASH_LAYOUT 1 + +#define ITS_RAM_FS 0 + +#define ITS_VALIDATE_METADATA_FROM_FLASH 1 + +#define ITS_MAX_ASSET_SIZE 1300 + +#define ITS_BUF_SIZE ITS_MAX_ASSET_SIZE + +#define ITS_NUM_ASSETS 10 + +#define ITS_STACK_SIZE 0x720 + +#define PS_CREATE_FLASH_LAYOUT 1 + +#define PS_RAM_FS 0 + +#define PS_ROLLBACK_PROTECTION 1 + +#define PS_VALIDATE_METADATA_FROM_FLASH 1 + +#define PS_MAX_ASSET_SIZE 2048 + +#define PS_NUM_ASSETS 10 + +#define PS_STACK_SIZE 0x700 + +#define CONFIG_TFM_CONN_HANDLE_MAX_NUM 8 + +#define CONFIG_TFM_DOORBELL_API 0 diff --git a/applications/object_detection/isp/CMakeLists.txt b/applications/object_detection/isp/CMakeLists.txt new file mode 100644 index 00000000..4188c5f3 --- /dev/null +++ b/applications/object_detection/isp/CMakeLists.txt @@ -0,0 +1,23 @@ +# Copyright 2023-2024 Arm Limited and/or its affiliates +# +# SPDX-License-Identifier: MIT + +add_library(isp-config INTERFACE) + +target_include_directories(isp-config + INTERFACE + . +) + +target_sources(isp-config + INTERFACE + acamera_callbacks.c + isp_app.c + isp_config.c +) + +target_link_libraries(isp-config + INTERFACE + isp_control + helpers-logging +) diff --git a/applications/object_detection/isp/acamera_callbacks.c b/applications/object_detection/isp/acamera_callbacks.c new file mode 100644 index 00000000..fd7aa5ec --- /dev/null +++ b/applications/object_detection/isp/acamera_callbacks.c @@ -0,0 +1,213 @@ +/* Copyright 2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include "acamera_logger.h" +#include "acamera_types.h" +#include "application_command_api.h" + +#include "isp_config.h" + +pvFrameReadyHandler_t pvDownscaledFrameReady = NULL; +pvFrameReadyHandler_t pvFullResolutionFrameReady = NULL; + +static uint32_t ulCoherentDmaAllocatedSize = 0; +typedef uint8_t ucFrameBuffer_t[ isp_configMAX_OUTPUT_FRAME_SIZE ]; +static ucFrameBuffer_t * xFullResolutionBuffer = ( ucFrameBuffer_t * ) isp_configFULL_RESOLUTION_BUFFER_BASE; +static ucFrameBuffer_t * xDownscaledBuffer = ( ucFrameBuffer_t * ) isp_configDOWNSCALED_BUFFER_BASE; + +/* This is for FVP stream optimisation */ +#include "FreeRTOS.h" +#include "semphr.h" +extern SemaphoreHandle_t xStreamSemaphore; + +/* Only 1 allocation is supported, it is for the Temper Frame */ +void * pvCallbackDmaAllocCoherent( uint32_t ulContextId, + uint64_t ullSize, + uint64_t * ullDmaAddress ) +{ + uint32_t ulAddress; + + if( ulCoherentDmaAllocatedSize + ullSize > isp_configCOHERENT_DMA_MEMORY_SIZE ) + { + LOG( LOG_ERR, "Not enough memory" ); + *ullDmaAddress = 0; + return NULL; + } + + ulAddress = isp_configCOHERENT_DMA_MEMORY_BASE + ulCoherentDmaAllocatedSize; + ulCoherentDmaAllocatedSize += ullSize; + + *ullDmaAddress = ulAddress; + + /* compute bus address */ + *ullDmaAddress -= ISP_SOC_DMA_BUS_OFFSET; + + LOG( LOG_DEBUG, + "DMA alloc: 0x%x, Memory left: 0x%x", + ( uint32_t ) ullSize, + isp_configCOHERENT_DMA_MEMORY_SIZE - ulCoherentDmaAllocatedSize ); + return ( void * ) ulAddress; +} + +/* Only 1 allocation is supported, it is for the Temper Frame */ +void vCallbackDmaFreeCoherent( uint32_t ulContextId, + uint64_t ullSize, + void * pvVirtualAddress, + uint64_t ullDmaAddress ) +{ + if( ( ( uint32_t ) pvVirtualAddress ) != isp_configCOHERENT_DMA_MEMORY_SIZE ) + { + LOG( LOG_ERR, "DMA free: Trying to fr ee unknown memory: 0x%x", ( uint32_t ) pvVirtualAddress ); + } + + ulCoherentDmaAllocatedSize = 0; +} + +int32_t lCallbackStreamGetFrame( uint32_t ulContextId, + acamera_stream_type_t xType, + aframe_t * pxAFrames, + uint64_t ullNumPlanes ) +{ + static uint32_t ulDownscaledFrameCounter = 0; + uint32_t ulAddress = 0; + + while( ullNumPlanes > 1 ) + { + ullNumPlanes--; + pxAFrames[ ullNumPlanes ].address = 0; + pxAFrames[ ullNumPlanes ].status = dma_buf_purge; + } + + if( xType == ACAMERA_STREAM_FR ) + { + static uint32_t ulFullResolutionFrameCounter = 0; + ulAddress = ( uint32_t ) xFullResolutionBuffer[ ulFullResolutionFrameCounter % isp_configMAX_BUFFERED_FRAMES ]; + ulFullResolutionFrameCounter++; + } + else if( xType == ACAMERA_STREAM_DS1 ) + { + ulAddress = ( uint32_t ) xDownscaledBuffer[ ulDownscaledFrameCounter % isp_configMAX_BUFFERED_FRAMES ]; + ulDownscaledFrameCounter++; + } + + pxAFrames[ 0 ].address = ulAddress; + + if( ulAddress == 0 ) + { + LOG( LOG_ERR, "No buffer available!" ); + pxAFrames[ 0 ].status = dma_buf_purge; + return -1; + } + + pxAFrames[ 0 ].status = dma_buf_empty; + return 0; +} + +/* Return pixel width in bits for format. */ +static inline uint32_t prvGetPixelWidth( uint32_t ulFormat ) +{ + uint32_t ulResult = 32; + + switch( ulFormat ) + { + case DMA_FORMAT_RGB24: + ulResult = 24; + break; + + case DMA_FORMAT_RGB565: + case DMA_FORMAT_RAW16: + case DMA_FORMAT_YUY2: + case DMA_FORMAT_UYVY: + ulResult = 16; + break; + + case DMA_FORMAT_RAW12: + ulResult = 12; + break; + + case DMA_FORMAT_NV12_Y: + case DMA_FORMAT_NV12_UV: + case DMA_FORMAT_NV12_VU: + case DMA_FORMAT_YV12_Y: + case DMA_FORMAT_YV12_U: + case DMA_FORMAT_YV12_V: + ulResult = 8; + break; + } + + return ulResult; +} + +int32_t lCallbackStreamPutFrame( uint32_t ulContextId, + acamera_stream_type_t xType, + aframe_t * pxAFrames, + uint64_t ullNumPlanes ) +{ + uint32_t ulAddress; + uint32_t ulWidth; + uint32_t ulHeight; + uint32_t ulBitsPerPixel; + uint32_t ulFormat; + uint32_t ulBitsPerPixelFormat; + + /* Note: We only care about the first plane */ + LOG( LOG_CRIT, "\033[1;33m-- aframes->frame_id = %d --\033[1;0m", pxAFrames->frame_id ); + + while( ullNumPlanes > 0 ) + { + ullNumPlanes--; + pxAFrames[ ullNumPlanes ].status = dma_buf_purge; + } + + if( pxAFrames->address ) + { + ulAddress = pxAFrames->address; + ulWidth = pxAFrames->width; + ulHeight = pxAFrames->height; + ulBitsPerPixel = pxAFrames->size / ulWidth / ulHeight; + + LOG( LOG_CRIT, "\033[1;33m-- %u X %u @ %u bytes per pixel --\033[1;0m", ulWidth, ulHeight, ulBitsPerPixel ); + } + + if( xType == ACAMERA_STREAM_DS1 ) + { + application_command( TIMAGE, DS1_FORMAT_BASE_PLANE_ID, 0, COMMAND_GET, &ulFormat ); + ulBitsPerPixelFormat = _get_pixel_width( ulFormat ) / 8; + + if( ulBitsPerPixel != ulBitsPerPixelFormat ) + { + /* Current frame and current format mismatch */ + /* Default to 2 bytes = RGB565, otherwise RGB32 */ + ulFormat = ( ulBitsPerPixel == 2 ) ? DMA_FORMAT_RGB565 : DMA_FORMAT_RGB32; + LOG( LOG_NOTICE, "Bits per pixel mismatch! Expected: %u. Using format: 0x%x", ulBitsPerPixelFormat * 8, ulFormat ); + } + + if( pvDownscaledFrameReady != NULL ) + { + pvDownscaledFrameReady( ulAddress, ulWidth, ulHeight, ulFormat, pxAFrames->frame_id ); + } + + /* Let stream thread start the next frame */ + xSemaphoreGive( xStreamSemaphore ); + } + else if( xType == ACAMERA_STREAM_FR ) + { + application_command( TIMAGE, FR_FORMAT_BASE_PLANE_ID, 0, COMMAND_GET, &ulFormat ); + ulBitsPerPixelFormat = _get_pixel_width( ulFormat ) / 8; + + if( ulBitsPerPixel != ulBitsPerPixelFormat ) + { + ulFormat = ( ulBitsPerPixel == 2 ) ? DMA_FORMAT_RGB565 : DMA_FORMAT_RGB32; + LOG( LOG_NOTICE, "Bits per pixel mismatch! Expected: %u. Using format: 0x%x", ulBitsPerPixelFormat * 8, ulFormat ); + } + + if( pvFullResolutionFrameReady != NULL ) + { + pvFullResolutionFrameReady( ulAddress, ulWidth, ulHeight, ulFormat, pxAFrames->frame_id ); + } + } + + return 0; +} diff --git a/applications/object_detection/isp/isp_app.c b/applications/object_detection/isp/isp_app.c new file mode 100644 index 00000000..2bb71f74 --- /dev/null +++ b/applications/object_detection/isp/isp_app.c @@ -0,0 +1,318 @@ +/* Copyright 2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include "acamera_command_api.h" +#include "acamera_control_config.h" +#include "acamera_firmware_api.h" +#include "acamera_firmware_config.h" +#include "acamera_interface_config.h" +#include "acamera_logger.h" + +#include "application_command_api.h" + +#include "system_cdma_platform.h" +#include "system_interrupts.h" + +#if ISP_HAS_STREAM_CONNECTION + #include "acamera_connection.h" +#endif + +#include "FreeRTOS.h" +#include "semphr.h" +#include "task.h" +#include "events.h" + +#include "isp_config.h" + +#define ENABLE_TPG_AT_START 0 + +static acamera_settings xACameraSettings[ FIRMWARE_CONTEXT_NUMBER ] = +{ + { + .sensor_init = fvp_sensor_init, + .sensor_deinit = fvp_sensor_deinit, + .get_calibrations = get_calibrations_dummy, + .lens_init = NULL, + .lens_deinit = NULL, + .isp_base = 0x0, + .hw_isp_addr = ISP_SOC_START_ADDR, + .callback_dma_alloc_coherent = pvCallbackDmaAllocCoherent, + .callback_dma_free_coherent = vCallbackDmaFreeCoherent, + .callback_stream_get_frame = lCallbackStreamGetFrame, + .callback_stream_put_frame = lCallbackStreamPutFrame, + }, +}; + +/* This example will run the infinite loop to process the firmware events. */ +/* This variable also can be changed outside to stop the processing. */ +volatile int32_t lACameraMainLoopActive = 1; + +/* Stream semaphore is used to block stream thread until previous frame is displayed */ +SemaphoreHandle_t xStreamSemaphore; +StaticSemaphore_t xStreamSemaphoreBuffer; + +#if ISP_HAS_STREAM_CONNECTION && CONNECTION_IN_THREAD + static void connection_thread( void * pvParameters ); +#endif +static void prvStreamControlThread( void * pvParameters ); +static void prvACameraThread( void * pvParameters ); + +/* this is a main application IRQ handler to drive the firmware */ +/* The main purpose is to redirect ISP irq event to the firmware core */ +/* Please see the ACamera Porting Guide for details. */ +static void prvInterruptHandler( void * pvPtr, + uint32_t ulMask ) +{ + /* the lower bits are for ISP interrupts on ACamera FPGA reference platform */ + uint32_t ulIspMask = ulMask & 0x0000FFFF; + + /* tell the firmware that isp interrupts happened */ + if( ulIspMask ) + { + /* the first irq pins are connected to ISP */ + acamera_interrupt_handler(); + } +} + +static void prvCreateTasks() +{ + /* ACamera provides a simple protocol to communicate with firmware */ + /* outside application. Several channels are supported depend on the */ + /* system requirements. */ + /* To start using ACamera Control Tool the connection must be initialised */ + /* before by calling acamera_connection_init */ + /* The connection module parses input commands from ACT and call the required */ + /* api command like acamera_command or acamera_calibrations. */ + /* Please see acamera_command_api.h for details. */ + #if ISP_HAS_STREAM_CONNECTION + #if !CONNECTION_IN_THREAD + acamera_connection_init(); + #else + + /*xTaskCreate(connection_thread, + * "connection", + * configMINIMAL_STACK_SIZE * 2, + * NULL, + * (configMAX_PRIORITIES - 1) | portPRIVILEGE_BIT, + * NULL);*/ + #endif + #endif + + xTaskCreate( prvACameraThread, + "acamera", + configMINIMAL_STACK_SIZE * 4, + NULL, + ( configMAX_PRIORITIES - 2 ) | portPRIVILEGE_BIT, + NULL ); + + xStreamSemaphore = xSemaphoreCreateBinaryStatic( &xStreamSemaphoreBuffer ); + xTaskCreate( prvStreamControlThread, + "stream", + configMINIMAL_STACK_SIZE, + NULL, + ( configMAX_PRIORITIES - 3 ) | portPRIVILEGE_BIT, + NULL ); +} + +static int prvIspInitialConfig() +{ + int32_t lResult; + uint32_t ulReturnCode; + uint32_t ulContextNumber; + uint32_t ulPreviousContextNumber = 0; + + /* The firmware supports multicontext. */ + /* It means that the customer can use the same firmware for controlling */ + /* several instances of different sensors/isp. To initialise a context */ + /* the structure acamera_settings must be filled properly. */ + /* the total number of initialized context must not exceed FIRMWARE_CONTEXT_NUMBER */ + /* all contexts are numerated from 0 till ctx_number - 1 */ + lResult = acamera_init( xACameraSettings, FIRMWARE_CONTEXT_NUMBER ); + + if( lResult != 0 ) + { + LOG( LOG_ERR, "Failed to start firmware processing thread. (0x%x)", lResult ); + return lResult; + } + + application_command( TGENERAL, ACTIVE_CONTEXT, 0, COMMAND_GET, &ulPreviousContextNumber ); + + system_interrupt_set_handler( prvInterruptHandler, NULL ); + + for( ulContextNumber = 0; ulContextNumber < FIRMWARE_CONTEXT_NUMBER; ulContextNumber++ ) + { + application_command( TGENERAL, ACTIVE_CONTEXT, ulContextNumber, COMMAND_SET, &ulReturnCode ); + + /* Disable most ISP calibrations. FVP sensor streams processed RGB images which does not need calibration */ + application_command( TALGORITHMS, AE_MODE_ID, AE_FULL_MANUAL, COMMAND_SET, &ulReturnCode ); + application_command( TALGORITHMS, AWB_MODE_ID, AWB_MANUAL, COMMAND_SET, &ulReturnCode ); + + application_command( TISP_MODULES, ISP_MODULES_MANUAL_TEMPER, ON, COMMAND_SET, &ulReturnCode ); + application_command( TISP_MODULES, ISP_MODULES_MANUAL_IRIDIX, ON, COMMAND_SET, &ulReturnCode ); + + application_command( TSYSTEM, SYSTEM_MANUAL_EXPOSURE, ON, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_MANUAL_EXPOSURE_RATIO, ON, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_MANUAL_ISP_DIGITAL_GAIN, ON, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_MANUAL_AWB, ON, COMMAND_SET, &ulReturnCode ); + +#define AWB_GAIN 9 + application_command( TSYSTEM, SYSTEM_AWB_BLUE_GAIN, AWB_GAIN, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_AWB_GREEN_EVEN_GAIN, AWB_GAIN, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_AWB_GREEN_ODD_GAIN, AWB_GAIN, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_AWB_RED_GAIN, AWB_GAIN, COMMAND_SET, &ulReturnCode ); +#define CCM_GAIN 4095 + application_command( TSYSTEM, SYSTEM_MANUAL_CCM, ON, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_RR, CCM_GAIN, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_GG, CCM_GAIN, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_BB, CCM_GAIN, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_RG, 0, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_RB, 0, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_GR, 0, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_GB, 0, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_BR, 0, COMMAND_SET, &ulReturnCode ); + application_command( TSYSTEM, SYSTEM_CCM_MATRIX_BG, 0, COMMAND_SET, &ulReturnCode ); + + application_command( TSYSTEM, TEST_PATTERN_MODE_ID, 5, COMMAND_SET, &ulReturnCode ); + + #if ENABLE_TPG_AT_START + application_command( TSYSTEM, TEST_PATTERN_ENABLE_ID, ON, COMMAND_SET, &ulReturnCode ); + #endif + + application_command( TSCENE_MODES, SHARPENING_STRENGTH_ID, 0, COMMAND_SET, &ulReturnCode ); + + application_command( TIMAGE, FR_FORMAT_BASE_PLANE_ID, DMA_FORMAT_RGB565, COMMAND_SET, &ulReturnCode ); + application_command( TIMAGE, DS1_FORMAT_BASE_PLANE_ID, DMA_FORMAT_RGB565, COMMAND_SET, &ulReturnCode ); + } + + application_command( TGENERAL, ACTIVE_CONTEXT, ulPreviousContextNumber, COMMAND_SET, &ulReturnCode ); + + return 0; +} + +int32_t lIspInit() +{ + system_cdma_setup(); + prvCreateTasks(); + return 0; +} + +/* On systems which support pthreads it is more efficient to run */ +/* control channel in a separate thread to let the firmware to communicate */ +/* with ACamera Control Tool (ACT) */ +/* ACT allows to change the firmware behaviour by calling API functions, change ISP registers and */ +/* update calibration LUTs. */ +/* Please read the ACamera Control Tool User Guide for details */ +#if ISP_HAS_STREAM_CONNECTION && CONNECTION_IN_THREAD + static void connection_thread( void * pvParameters ) + { + /* acamera_connection_init is used to initialize the */ + /* communication channel between the firmware application */ + /* and the firmware. It is used only together with ACT tool */ + /* and may be omitted on the customer discretion */ + /* if ACT is not required */ + acamera_connection_init(); + + for( ; ; ) + { + /* the function checks the incoming requests from */ + /* ACT tool and call the corresponding API command. */ + /* Please note that acamera_connection_process may be omitted */ + /* if ACT tool is not used for the firmware API. */ + acamera_connection_process(); + } + } +#endif /* if ISP_HAS_STREAM_CONNECTION && CONNECTION_IN_THREAD */ + +extern volatile uint32_t UART_PLEASE_HOLD_STREAM; +static void prvStreamControlThread( void * pvParameters ) +{ + uint32_t ulFrameCount = 0; + + while( 1 ) + { + EventBits_t xFlags = xEventGroupWaitBits( xSystemEvents, ( EventBits_t ) EVENT_MASK_ML_STOP, pdTRUE, pdFAIL, 10 ); + + if( xFlags & EVENT_MASK_ML_STOP ) + { + /* jump out to outer loop */ + LogInfo( ( "Stopping processing\r\n" ) ); + break; + } + + if( STREAM_ENABLED ) + { + if( !ulFrameCount || !UART_PLEASE_HOLD_STREAM ) + { + LOG( LOG_CRIT, "\033[1;34m-- CAMERA STREAM TRIGGER #%d --\033[1;0m", ulFrameCount ); + ulFrameCount++; + *( ( uint8_t * ) ( ISP_VIRTUAL_CAMERA_BASE_NS ) ) = 0x1; /* camera enable */ + } + + /* Displaying a frame gives the semaphore. If previous frame is not yet + * displayed, or no frames have been displayed, task waits 100 ticks. */ + xSemaphoreTake( xStreamSemaphore, 100 ); + } + + vTaskDelay( 5 ); + } +} + +static void prvACameraThread( void * pvParameters ) +{ + static uint32_t ulDoInitialSetup = 1; + uint32_t ulReturnCode = 0; + + if( prvIspInitialConfig() ) + { + lACameraMainLoopActive = 0; + } + + /* acamera_process function must be called on every incoming interrupt */ + /* to give the firmware the possibility to apply */ + /* all internal algorithms and change the ISP state. */ + /* The external application can be run in the same loop on bare metal systems. */ + while( lACameraMainLoopActive ) + { + /* acamera_process must be called for each initialised context */ + acamera_process(); + #if ISP_HAS_STREAM_CONNECTION && !CONNECTION_IN_THREAD + /* acamera_connection_process is used for communication between */ + /* firmware and ACT through different possible channels like */ + /* cmd_queue memory in ISP, socket, UART, chardev etc. */ + /* Different channels can be supported depending on the target */ + /* platform. The common case when cmd_queue buffer is used */ + /* ( see acamera_isp_config.h ) */ + /* The channels supported by this routine can be used not only on */ + /* NIOS2 platform but on the customer system as well. */ + acamera_connection_process(); + #endif + + /* This needs to be after first acamera_process(); because for the first time, it resets the FSMs and */ + /* Crop FSM resets the scaler width/height values */ + if( ulDoInitialSetup ) + { + ulDoInitialSetup = 0; + application_command( TIMAGE, IMAGE_RESIZE_TYPE_ID, SCALER_DS, COMMAND_SET, &ulReturnCode ); + application_command( TIMAGE, IMAGE_RESIZE_WIDTH_ID, 192, COMMAND_SET, &ulReturnCode ); + application_command( TIMAGE, IMAGE_RESIZE_HEIGHT_ID, 192, COMMAND_SET, &ulReturnCode ); + application_command( TIMAGE, IMAGE_RESIZE_ENABLE_ID, RUN, COMMAND_SET, &ulReturnCode ); + + application_command( TSENSOR, SENSOR_STREAMING, ON, COMMAND_SET, &ulReturnCode ); + } + + taskYIELD(); + } + + /* this api function will free */ + /* all resources allocated by the firmware */ + acamera_terminate(); + + LOG( LOG_CRIT, "Acamera terminated" ); + + for( ; ; ) + { + /* Task shouldn't exit */ + } +} diff --git a/applications/object_detection/isp/isp_config.c b/applications/object_detection/isp/isp_config.c new file mode 100644 index 00000000..8cf89e81 --- /dev/null +++ b/applications/object_detection/isp/isp_config.c @@ -0,0 +1,459 @@ +/* Copyright 2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include CMSIS_device_header + +#include "device_cfg.h" +#include "device_definition.h" +#include "hdlcd_drv.h" +#include "hdlcd_helper.h" +#include "stdint.h" +#include +#include + +#include "arm_2d.h" + +#include "isp_config.h" + +#include "ml_interface.h" + +/* + * Semihosting is a mechanism that enables code running on an ARM target + * to communicate and use the Input/Output facilities of a host computer + * that is running a debugger. + * There is an issue where if you use armclang at -O0 optimisation with + * no parameters specified in the main function, the initialisation code + * contains a breakpoint for semihosting by default. This will stop the + * code from running before main is reached. + * Semihosting can be disabled by defining __ARM_use_no_argv symbol + * (or using higher optimization level). + */ +#if defined( __ARMCC_VERSION ) && ( __ARMCC_VERSION >= 6010050 ) + __asm( " .global __ARM_use_no_argv\n" ); +#endif + +#define isp_configHIGHLIGHTED_FRAME_WIDTH 2 + +#define isp_configMAX_BORDER_WIDTH 320 +#define isp_configMAX_BORDER_HEIGHT 240 +#define isp_configMAX_BORDER_BYTES_PER_PIXEL 4 +#define isp_configBORDER_BUFFER_SIZE ( isp_configMAX_BORDER_WIDTH * isp_configMAX_BORDER_HEIGHT * isp_configMAX_BORDER_BYTES_PER_PIXEL ) + +#define isp_configMAX_DETECT_RESULTS 10 + +/* This buffer is used by FP unit, align it to prevent unaligned FP faults */ +static uint8_t ucBorderTileBuffer[ isp_configBORDER_BUFFER_SIZE ] __ALIGNED( 4 ); +static volatile uint8_t ucFlag = 0; + +static struct LastFrame xLastFrame = +{ + .ulDisplayed = 0, + .ulInferred = 0, + .xFullResolution = + { + .ulMode = HDLCD_PIXEL_FORMAT_NOT_SUPPORTED, + .ulWidth = 0, + .ulHeight = 0 + } +}; + +struct arm_2d_tile_t xRootTile = +{ + .tInfo = + { + .bIsRoot = true, + .bHasEnforcedColour = true, + }, +}; + +#define isp_configMAX_INFER_FRAME_WIDTH 192 +#define isp_configMAX_INFER_FRAME_HEIGHT 192 +#define isp_configMAX_INFER_FRAME_SIZE ( isp_configMAX_INFER_FRAME_WIDTH * isp_configMAX_INFER_FRAME_HEIGHT ) +uint8_t ucGrayBuffer[ isp_configMAX_INFER_FRAME_SIZE ] __attribute__( ( aligned( 32 ) ) ); +static void prvHdlcdShow( uint32_t address, + uint32_t width, + uint32_t height, + uint32_t mode ); +static void prvHandleFullResolutionFrame( uint32_t ulAddress, + uint32_t ulWidth, + uint32_t ulHeight, + uint32_t ulMode, + uint32_t ulFrameId ); +static void prvHandleDownscaledFrame( uint32_t ulAddress, + uint32_t ulWidth, + uint32_t ulHeight, + uint32_t ulMode, + uint32_t ulFrameId ); +static void prvDrawFrameOnTile( struct arm_2d_tile_t * pxDest, + struct arm_2d_region_t * pxFrame, + enum hdlcd_pixel_format eFormat ); +void vHDLCDHandler( void ); +void vEnableHdlcdIrq( void ); +void vDisableHdlcdIrq( void ); + +extern int32_t lIspInit(); +extern void arm_2d_init( void ); + +void vStartISPDemo() +{ + enum hdlcd_error_t eHdlcdErr; + + NVIC_SetPriority( ISP_IRQn, 7 ); + NVIC_SetPriority( HDLCD_IRQn, 7 ); + NVIC_SetPriority( NPU0_IRQn, 5 ); + NVIC_SetPriority( UARTTX0_IRQn, 4 ); + + LogInfo( ( "ISP init\r\n" ) ); + + arm_2d_init(); + + LogInfo( ( "Starting HDLCD config!\r\n" ) ); + + eHdlcdErr = hdlcd_init( &HDLCD_DEV ); + + if( eHdlcdErr != HDLCD_ERR_NONE ) + { + LogInfo( ( "HDLCD init error! \r\n" ) ); + + while( 1 ) + { + } + } + + eHdlcdErr = hdlcd_static_config( &HDLCD_DEV ); + + if( eHdlcdErr != HDLCD_ERR_NONE ) + { + LogInfo( ( "HDLCD static config error! \r\n" ) ); + + while( 1 ) + { + } + } + + LogInfo( ( "Starting ISP init!\r\n" ) ); + + lIspInit(); + + pvDownscaledFrameReady = prvHandleDownscaledFrame; + pvFullResolutionFrameReady = prvHandleFullResolutionFrame; +} + +static uint32_t ulHdlcdToArm2dColorScheme( enum hdlcd_pixel_format eFormat ) +{ + switch( eFormat ) + { + case HDLCD_PIXEL_FORMAT_RGB565: + return ARM_2D_COLOUR_RGB565; + + case HDLCD_PIXEL_FORMAT_RGB32: + return ARM_2D_COLOUR_RGB32; + + case HDLCD_PIXEL_FORMAT_A2R10G10B10: + return ARM_2D_COLOUR_32BIT; + + default: + return 0; + } +} + +void vRgbToGrayscale( const uint8_t * pucSrcImage, + uint8_t * pucDstImage, + const size_t xDstImageSize, + enum hdlcd_pixel_format eFormat ) +{ + const struct hdlcd_pixel_cfg_t * const pxPixelConfig = HDLCD_MODES[ eFormat ].pixel_cfg; + const uint32_t ulRedOffset = pxPixelConfig->red.offset; + const uint32_t ulGreenOffset = pxPixelConfig->green.offset; + const uint32_t ulBlueOffset = pxPixelConfig->blue.offset; + const uint32_t ulRedMask = ( 0x1UL << pxPixelConfig->red.bit_size ) - 1; + const uint32_t ulGreenMask = ( 0x1UL << pxPixelConfig->green.bit_size ) - 1; + const uint32_t ulBlueMask = ( 0x1UL << pxPixelConfig->blue.bit_size ) - 1; + const float xRed = 0.299 * ( 0x1UL << ( 8 - pxPixelConfig->red.bit_size ) ); + const float xGreen = 0.587 * ( 0x1UL << ( 8 - pxPixelConfig->green.bit_size ) ); + const float xBlue = 0.114 * ( 0x1UL << ( 8 - pxPixelConfig->blue.bit_size ) ); + + for( size_t i = 0; i < xDstImageSize; ++i, pucSrcImage += HDLCD_MODES[ eFormat ].bytes_per_pixel ) + { + uint32_t ulGrayIntensity = xRed * ( ( ( *( uint32_t * ) pucSrcImage ) >> ulRedOffset ) & ulRedMask ) + xGreen * ( ( ( *( uint32_t * ) pucSrcImage ) >> ulGreenOffset ) & ulGreenMask ) + + xBlue * ( ( ( *( uint32_t * ) pucSrcImage ) >> ulBlueOffset ) & ulBlueMask ); + *pucDstImage++ = ulGrayIntensity <= 0xff ? ulGrayIntensity : 0xff; + } +} + +void vGrayscaleToRgb( const uint8_t * pucSrcImage, + uint8_t * pucDstImage, + const size_t xSrcImageSize, + enum hdlcd_pixel_format eFormat ) +{ + const struct hdlcd_pixel_cfg_t * const pxPixelConfig = HDLCD_MODES[ eFormat ].pixel_cfg; + const uint32_t ulRedOffset = pxPixelConfig->red.offset; + const uint32_t ulGreenOffset = pxPixelConfig->green.offset; + const uint32_t ulBlueOffset = pxPixelConfig->blue.offset; + const uint32_t ulRedMask = ( 0x1UL << pxPixelConfig->red.bit_size ) - 1; + const uint32_t ulGreenMask = ( 0x1UL << pxPixelConfig->green.bit_size ) - 1; + const uint32_t ulBlueMask = ( 0x1UL << pxPixelConfig->blue.bit_size ) - 1; + uint32_t ulColor; + uint32_t ulGrayMinimum = 0xff; + uint32_t ulGrayMaximum = 0x00; + + for( size_t i = 0; i < xSrcImageSize; ++i, pucDstImage += HDLCD_MODES[ eFormat ].bytes_per_pixel ) + { + ulColor = ( ( ( ( ( uint32_t ) *pucSrcImage ) >> ( 8 - pxPixelConfig->red.bit_size ) ) & ulRedMask ) << ulRedOffset ) + | ( ( ( ( ( uint32_t ) *pucSrcImage ) >> ( 8 - pxPixelConfig->green.bit_size ) ) & ulGreenMask ) << ulGreenOffset ) + | ( ( ( ( ( uint32_t ) *pucSrcImage ) >> ( 8 - pxPixelConfig->blue.bit_size ) ) & ulBlueMask ) << ulBlueOffset ); + + if( ( uint32_t ) *pucSrcImage > ulGrayMaximum ) + { + ulGrayMaximum = ( uint32_t ) *pucSrcImage; + } + + if( ( uint32_t ) *pucSrcImage < ulGrayMinimum ) + { + ulGrayMinimum = ( uint32_t ) *pucSrcImage; + } + + for( size_t j = 0; j < HDLCD_MODES[ eFormat ].bytes_per_pixel; j++ ) + { + pucDstImage[ j ] = ( ( uint8_t * ) &ulColor )[ j ]; + } + + pucSrcImage++; + } + + LogInfo( ( "srcGray: %02u-%02u\r\n", ulGrayMinimum, ulGrayMaximum ) ); +} + +static void prvHandleFullResolutionFrame( uint32_t ulAddress, + uint32_t ulWidth, + uint32_t ulHeight, + uint32_t ulMode, + uint32_t ulFrameId ) +{ + /* If the previous full frame wasn't inferred yet, skip this frame */ + if( xLastFrame.ulDisplayed == xLastFrame.ulInferred ) + { + xLastFrame.ulDisplayed = ulFrameId; + xLastFrame.xFullResolution.ulMode = ulMode; + xLastFrame.xFullResolution.ulWidth = ulWidth; + xLastFrame.xFullResolution.ulHeight = ulHeight; + prvHdlcdShow( ulAddress, ulWidth, ulHeight, ulMode ); + } +} + +static void prvHandleDownscaledFrame( uint32_t ulAddress, + uint32_t ulWidth, + uint32_t ulHeight, + uint32_t ulMode, + uint32_t ulFrameId ) +{ + xLastFrame.ulInferred = ulFrameId; + uint32_t i = 0UL; + uint32_t ulResultsCount = 0UL; + struct DetectRegion_t xResults[ isp_configMAX_DETECT_RESULTS ]; + float xUpscaleWidth = xLastFrame.xFullResolution.ulWidth / ulWidth; + float xUpscaleHeight = xLastFrame.xFullResolution.ulHeight / ulHeight; + + static struct arm_2d_region_t xFrameRegion = { 0 }; + enum hdlcd_pixel_format ePixelFormat = HDLCD_PIXEL_FORMAT_RGB565; + + if( ePixelFormat == HDLCD_PIXEL_FORMAT_NOT_SUPPORTED ) + { + LogError( ( "Unsupported pixel format: 0x%x\r\n", xLastFrame.xFullResolution.ulMode ) ); + return; + } + + if( ulWidth * ulHeight > isp_configMAX_INFER_FRAME_SIZE ) + { + LogError( ( "Input frame too big for inference!\r\n" ) ); + return; + } + + /* Note: mode Vs xLastFrame.xFullResolution.ulMode */ + LogInfo( ( "Converting to Gray: 0x%x -> 0x%x\r\n", ulAddress, ( uint32_t ) ucGrayBuffer ) ); + vRgbToGrayscale( ( uint8_t * ) ulAddress, ucGrayBuffer, ulWidth * ulHeight, HDLCD_PIXEL_FORMAT_RGB565 ); + + /*taskENTER_CRITICAL(); */ + ulResultsCount = isp_configMAX_DETECT_RESULTS; + lMLRunInference( ucGrayBuffer, xResults, &ulResultsCount ); + /*taskEXIT_CRITICAL(); */ + + hdlcd_disable( &HDLCD_DEV ); + + /* Draw frame according to selected region in the secondary buffer */ + for( i = 0; i < ulResultsCount; i++ ) + { + xFrameRegion.tSize.iWidth = ( int ) xResults[ i ].ulW * xUpscaleWidth; + xFrameRegion.tSize.iHeight = ( int ) xResults[ i ].ulH * xUpscaleHeight; + xFrameRegion.tLocation.iX = ( int ) xResults[ i ].ulX * xUpscaleWidth; + xFrameRegion.tLocation.iY = ( int ) xResults[ i ].ulY * xUpscaleHeight; + + if( xFrameRegion.tSize.iWidth < isp_configHIGHLIGHTED_FRAME_WIDTH * 3 ) + { + xFrameRegion.tSize.iWidth = isp_configHIGHLIGHTED_FRAME_WIDTH * 3; + } + + if( xFrameRegion.tSize.iHeight < isp_configHIGHLIGHTED_FRAME_WIDTH * 3 ) + { + xFrameRegion.tSize.iHeight = isp_configHIGHLIGHTED_FRAME_WIDTH * 3; + } + + prvDrawFrameOnTile( &xRootTile, &xFrameRegion, ePixelFormat ); + } + + hdlcd_enable( &HDLCD_DEV ); + /* Wait until HDLCD is displayed at least once */ + ucFlag = 0; + vEnableHdlcdIrq(); + + while( ucFlag < 5 ) + { + } + + vDisableHdlcdIrq(); +} + +static void prvHdlcdShow( uint32_t ulAddress, + uint32_t ulWidth, + uint32_t ulHeight, + uint32_t ulMode ) +{ + struct hdlcd_resolution_cfg_t xCustomResolutionConfig = { 0 }; + + enum hdlcd_error_t eHdlcdErr = 0; + enum hdlcd_pixel_format ePixelFormat = HDLCD_PIXEL_FORMAT_RGB565; + + LogInfo( ( "Displaying image...\r\n" ) ); + + if( ePixelFormat == HDLCD_PIXEL_FORMAT_NOT_SUPPORTED ) + { + LogError( ( "Unsupported pixel format: 0x%x\r\n", ulMode ) ); + return; + } + + hdlcd_disable( &HDLCD_DEV ); + + /* Note: This only works, because FVP ignores all timing values */ + xCustomResolutionConfig.v_data = ulHeight; + xCustomResolutionConfig.h_data = ulWidth; + hdlcd_set_custom_resolution( &HDLCD_DEV, &xCustomResolutionConfig ); + + struct hdlcd_buffer_cfg_t xHdlcdBuffer = + { + .base_address = ulAddress, + .line_length = ulWidth * HDLCD_MODES[ ePixelFormat ].bytes_per_pixel, + .line_count = ulHeight - 1, + .line_pitch = ulWidth * HDLCD_MODES[ ePixelFormat ].bytes_per_pixel, + .pixel_format = HDLCD_MODES[ ePixelFormat ].pixel_format + }; + + eHdlcdErr = hdlcd_buffer_config( &HDLCD_DEV, &xHdlcdBuffer ); + + if( eHdlcdErr != HDLCD_ERR_NONE ) + { + LogError( ( "HDLCD buffer config error! \r\n" ) ); + + while( 1 ) + { + } + } + + eHdlcdErr = hdlcd_pixel_config( &HDLCD_DEV, HDLCD_MODES[ ePixelFormat ].pixel_cfg ); + + if( eHdlcdErr != HDLCD_ERR_NONE ) + { + LogError( ( "HDLCD pixel config error! \r\n" ) ); + + while( 1 ) + { + } + } + + xRootTile.pwBuffer = ( uint32_t * ) ulAddress; + xRootTile.tRegion.tSize.iWidth = ulWidth; + xRootTile.tRegion.tSize.iHeight = ulHeight; + xRootTile.tInfo.tColourInfo.chScheme = ulHdlcdToArm2dColorScheme( ePixelFormat ); + + hdlcd_enable( &HDLCD_DEV ); + + LogInfo( ( "Image displayed\r\n" ) ); +} + +static void prvDrawFrameOnTile( struct arm_2d_tile_t * pxDestination, + struct arm_2d_region_t * pxFrame, + enum hdlcd_pixel_format eFormat ) +{ + static struct arm_2d_region_t xBlendRegion = { 0 }; + static struct arm_2d_tile_t xBorderTile = { 0 }; + uint_fast8_t xOpacity = 0xFF; + + xBlendRegion.tSize.iWidth = ( pxFrame->tSize.iWidth - 2 * isp_configHIGHLIGHTED_FRAME_WIDTH ); + xBlendRegion.tSize.iHeight = ( pxFrame->tSize.iHeight - 2 * isp_configHIGHLIGHTED_FRAME_WIDTH ); + xBlendRegion.tLocation.iX = isp_configHIGHLIGHTED_FRAME_WIDTH; + xBlendRegion.tLocation.iY = isp_configHIGHLIGHTED_FRAME_WIDTH; + + xBorderTile.tRegion.tSize.iWidth = pxFrame->tSize.iWidth; + xBorderTile.tRegion.tSize.iHeight = pxFrame->tSize.iHeight; + xBorderTile.tInfo.bIsRoot = true; + xBorderTile.tInfo.bHasEnforcedColour = true; + xBorderTile.tInfo.tColourInfo.chScheme = ulHdlcdToArm2dColorScheme( eFormat ); + xBorderTile.pwBuffer = ( uint32_t * ) ucBorderTileBuffer; + + if( eFormat == HDLCD_PIXEL_FORMAT_RGB565 ) + { + arm_2dp_rgb16_fill_colour( + NULL, &xBorderTile, &xBorderTile.tRegion, ( uint16_t ) HDLCD_MODES[ eFormat ].default_highlight_color ); + arm_2dp_rgb16_fill_colour( NULL, &xBorderTile, &xBlendRegion, ( uint16_t ) HDLCD_MODES[ eFormat ].default_mask_color ); + arm_2dp_rgb565_tile_copy_with_colour_keying_and_opacity( + NULL, + &xBorderTile, + pxDestination, + pxFrame, + xOpacity, + ( arm_2d_color_rgb565_t ) { .tValue = ( uint16_t ) HDLCD_MODES[ eFormat ].default_mask_color } ); + } + else + { + arm_2dp_rgb32_fill_colour( + NULL, &xBorderTile, &xBorderTile.tRegion, HDLCD_MODES[ eFormat ].default_highlight_color ); + arm_2dp_rgb32_fill_colour( NULL, &xBorderTile, &xBlendRegion, HDLCD_MODES[ eFormat ].default_mask_color ); + arm_2dp_cccn888_tile_copy_with_colour_keying_and_opacity( + NULL, + &xBorderTile, + pxDestination, + pxFrame, + xOpacity, + ( arm_2d_color_cccn888_t ) { .tValue = HDLCD_MODES[ eFormat ].default_mask_color } ); + } +} + +void vHDLCDHandler( void ) +{ + /* Clear IRQ */ + uint32_t ulIrqState = hdlcd_get_irq_state( &HDLCD_DEV ); + + hdlcd_clear_irq( &HDLCD_DEV, ulIrqState ); + + if( ulIrqState & INT_DMA_END_Msk ) + { + ucFlag++; + } + + __NVIC_ClearPendingIRQ( HDLCD_IRQn ); +} + +void vEnableHdlcdIrq( void ) +{ + NVIC_ClearPendingIRQ( HDLCD_IRQn ); + NVIC_SetVector( HDLCD_IRQn, ( uint32_t ) vHDLCDHandler ); + NVIC_EnableIRQ( HDLCD_IRQn ); + hdlcd_enable_irq( &HDLCD_DEV, INT_DMA_END_Msk ); +} + +void vDisableHdlcdIrq( void ) +{ + NVIC_DisableIRQ( HDLCD_IRQn ); + hdlcd_disable_irq( &HDLCD_DEV, INT_DMA_END_Msk ); +} diff --git a/applications/object_detection/isp/isp_config.h b/applications/object_detection/isp/isp_config.h new file mode 100644 index 00000000..689f0ccf --- /dev/null +++ b/applications/object_detection/isp/isp_config.h @@ -0,0 +1,93 @@ +/* Copyright 2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include "acamera_firmware_api.h" +#include "acamera_sensor_api.h" +#include "acamera_types.h" + +#include "platform_base_address.h" + +#include "logging_levels.h" + +/* Logging configuration for the MQTT library. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "ISP" +#endif + +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif + +#include "logging_stack.h" + +#define isp_configCOHERENT_DMA_MEMORY_BASE DDR4_BLK2_BASE_NS + +#define isp_configMAX_FRAME_WIDTH 2048 +#define isp_configMAX_FRAME_HEIGHT 1080 +#define isp_configMAX_BITS_PER_PIXEL 20 +#define isp_configMAX_BYTES_PER_PIXEL 4 +#define isp_configMAX_INPUT_FRAME_SIZE ( isp_configMAX_FRAME_WIDTH * isp_configMAX_FRAME_HEIGHT * isp_configMAX_BITS_PER_PIXEL / 8 ) +#define isp_configMAX_OUTPUT_FRAME_SIZE ( isp_configMAX_FRAME_WIDTH * isp_configMAX_FRAME_HEIGHT * isp_configMAX_BYTES_PER_PIXEL ) + +/* 5400KB */ +#define isp_configCOHERENT_DMA_MEMORY_SIZE isp_configMAX_INPUT_FRAME_SIZE + +#define isp_configMAX_BUFFERED_FRAMES 4 +#define isp_configFULL_RESOLUTION_BUFFER_BASE ( isp_configCOHERENT_DMA_MEMORY_BASE + isp_configCOHERENT_DMA_MEMORY_SIZE ) +#define isp_configDOWNSCALED_BUFFER_BASE ( isp_configFULL_RESOLUTION_BUFFER_BASE + ( isp_configMAX_OUTPUT_FRAME_SIZE * isp_configMAX_BUFFERED_FRAMES ) ) + +struct FullResolutionImage +{ + uint32_t ulMode; + uint32_t ulWidth; + uint32_t ulHeight; +}; + +struct LastFrame +{ + uint32_t ulDisplayed; + uint32_t ulInferred; + struct FullResolutionImage xFullResolution; +}; + +typedef void (* pvFrameReadyHandler_t)( uint32_t ulAddress, + uint32_t ulWidth, + uint32_t ulHeight, + uint32_t ulMode, + uint32_t ulFrameId ); + +extern pvFrameReadyHandler_t pvDownscaledFrameReady; +extern pvFrameReadyHandler_t pvFullResolutionFrameReady; + +extern void fvp_sensor_init( void ** ppvContext, + sensor_control_t * ); +extern void fvp_sensor_deinit( void * pvContext ); +extern uint32_t get_calibrations_dummy( uint32_t ulContextNumber, + void * pvSensorArg, + ACameraCalibrations * ); + +extern void * pvCallbackDmaAllocCoherent( uint32_t ulContextId, + uint64_t ullSize, + uint64_t * pullDmaAddress ); +extern void vCallbackDmaFreeCoherent( uint32_t ulContextId, + uint64_t ullSize, + void * pvVirtualAddress, + uint64_t ullDmaAddress ); + +/* The ISP pipeline can have several outputs such as Full Resolution, DownScaler1, DownScaler2 etc */ +/* It is possible to set up the firmware to return the metadata for each output frame from */ +/* the specific channel. This callbacks must be set in acamera_settings structure and passed to the firmware in */ +/* acamera_init api function */ +/* The context id can be used to differentiate contexts */ +extern int32_t lCallbackStreamGetFrame( uint32_t ulContextId, + acamera_stream_type_t xType, + aframe_t * pxAFrames, + uint64_t ullNumPlanes ); +extern int32_t lCallbackStreamPutFrame( uint32_t ulContextId, + acamera_stream_type_t xType, + aframe_t * pxAFrames, + uint64_t ullNumPlanes ); + +void vStartISPDemo(); diff --git a/applications/object_detection/main.c b/applications/object_detection/main.c new file mode 100644 index 00000000..128db86a --- /dev/null +++ b/applications/object_detection/main.c @@ -0,0 +1,324 @@ +/* Copyright 2023-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include + +#include "FreeRTOS.h" +#include "task.h" + +#include "app_config.h" +#include "aws_clientcredential.h" +#include "blink_task.h" +#include "dev_mode_key_provisioning.h" +#include "events.h" +#include "mbedtls/platform.h" +#include "mbedtls/threading.h" +#include "ml_interface.h" +#include "mqtt_agent_task.h" +#include "tfm_ns_interface.h" +#include "bsp_serial.h" +#include "aws_mbedtls_config.h" +#include "threading_alt.h" +#include "psa/crypto.h" +#include "tfm_ns_interface.h" + +#include "FreeRTOSConfig.h" + +#ifdef AUDIO_VSI + #include "Driver_SAI.h" +#endif + +/* + * Semihosting is a mechanism that enables code running on an ARM target + * to communicate and use the Input/Output facilities of a host computer + * that is running a debugger. + * There is an issue where if you use armclang at -O0 optimisation with + * no parameters specified in the main function, the initialisation code + * contains a breakpoint for semihosting by default. This will stop the + * code from running before main is reached. + * Semihosting can be disabled by defining __ARM_use_no_argv symbol + * (or using higher optimization level). + */ +#if defined( __ARMCC_VERSION ) && ( __ARMCC_VERSION >= 6010050 ) + __asm( " .global __ARM_use_no_argv\n" ); +#endif + +extern void vStartOtaTask( void ); +extern int32_t network_startup( void ); + +#ifdef AUDIO_VSI + void vVsiCallbackTask( void * arg ); + static void prvStartVsiCallbackTask( void ); + extern uint32_t ulVsiEvent; + extern void (* pxOnVsiEvent)( void * ); + extern void * pvVsiContext; + TaskHandle_t xVsiTaskHandle = NULL; +#endif + +psa_key_handle_t xOTACodeVerifyKeyHandle = 0; +QueueHandle_t xMlMqttQueue = NULL; + +static bool prvAreAwsCredentialsValid( void ) +{ + if( ( strcmp( clientcredentialMQTT_BROKER_ENDPOINT, "dummy.endpointid.amazonaws.com" ) == 0 ) || + ( strcmp( clientcredentialIOT_THING_NAME, "dummy_thingname" ) == 0 ) ) + { + printf( "[ERR] INVALID BROKER ENDPOINT AND/OR THING NAME.\r\n" ); + printf( "[ERR] Set the right credentials in aws_clientcredential.h\r\n" ); + return false; + } + + return true; +} + +void vOtaActiveHook( void ) +{ + vMlTaskInferenceStop(); +} + +void vOtaNotActiveHook( void ) +{ + vMlTaskInferenceStart(); +} + +void vAssertCalled( const char * pcFile, + unsigned long ulLine ) +{ + printf( "ASSERT failed! file %s:%lu, \n", pcFile, ulLine ); + + taskENTER_CRITICAL(); + { + /* Use the debugger to set ul to a non-zero value in order to step out + * of this function to determine why it was called. */ + volatile uint32_t ulLooping = 0; + + while( ulLooping == 0LU ) + { + portNOP(); + } + } + taskEXIT_CRITICAL(); +} + +BaseType_t xApplicationGetRandomNumber( uint32_t * pulNumber ) +{ + return ( BaseType_t ) ( psa_generate_random( ( uint8_t * ) ( pulNumber ), sizeof( uint32_t ) ) == PSA_SUCCESS ); +} + +uint32_t ulApplicationGetNextSequenceNumber( uint32_t ulSourceAddress, + uint16_t usSourcePort, + uint32_t ulDestinationAddress, + uint16_t usDestinationPort ) +{ + ( void ) ulSourceAddress; + ( void ) usSourcePort; + ( void ) ulDestinationAddress; + ( void ) usDestinationPort; + + uint32_t uxRandomValue = 0U; + psa_status_t xPsaStatus = psa_generate_random( ( uint8_t * ) ( &uxRandomValue ), sizeof( uint32_t ) ); + + if( xPsaStatus != PSA_SUCCESS ) + { + LogError( ( "psa_generate_random failed with %d.", xPsaStatus ) ); + configASSERT( 0 ); + } + + return uxRandomValue; +} + +int main( void ) +{ + bsp_serial_init(); + + xLoggingTaskInitialize( appCONFIG_LOGGING_TASK_STACK_SIZE, + appCONFIG_LOGGING_TASK_PRIORITY, + appCONFIG_LOGGING_MESSAGE_QUEUE_LENGTH ); + + UBaseType_t uxStatus = tfm_ns_interface_init(); + + if( uxStatus != 0 ) + { + LogError( ( "TF-M non-secure interface init failed with [%d]. Exiting...\n", uxStatus ) ); + return EXIT_FAILURE; + } + + LogInfo( ( "PSA Framework version is: %d\n", psa_framework_version() ) ); + + if( xEventHelperInit() != 0 ) + { + LogError( ( "System events group was not initialised successfully" ) ); + return EXIT_FAILURE; + } + + ( void ) mbedtls_platform_set_calloc_free( mbedtls_platform_calloc, mbedtls_platform_free ); + + mbedtls_threading_set_alt( mbedtls_platform_mutex_init, + mbedtls_platform_mutex_free, + mbedtls_platform_mutex_lock, + mbedtls_platform_mutex_unlock ); + + UBaseType_t xReturnValue = vDevModeKeyProvisioning(); + + if( xReturnValue != CKR_OK ) + { + LogError( ( "Device key provisioning failed [%d]\n", xReturnValue ) ); + LogError( ( "Device cannot connect to IoT Core. Exiting...\n" ) ); + return EXIT_FAILURE; + } + + LogInfo( ( "Device key provisioning succeeded \n" ) ); + + /* FIXME: Magic value */ + uxStatus = xOtaProvisionCodeSigningKey( &xOTACodeVerifyKeyHandle, 3072 ); + + if( uxStatus != PSA_SUCCESS ) + { + LogError( ( "OTA signing key provision failed [%d]\n", uxStatus ) ); + } + + LogInfo( ( "OTA signing key provisioning succeeded \n" ) ); + + /* The next initializations are done as a part of the main */ + /* function as these resources are shared between tasks */ + /* and it is not guranteed that the task which initialise */ + /* these resources will start first before the tasks using them. */ + xMlMqttQueue = xQueueCreate( 20, sizeof( MLMqttMsg_t ) ); + + if( xMlMqttQueue == NULL ) + { + LogError( ( "Failed to create xMlMqttQueue\r\n" ) ); + return EXIT_FAILURE; + } + + vMlTaskInferenceStop(); + + if( prvAreAwsCredentialsValid() == true ) + { + if( network_startup() != 0 ) + { + return EXIT_FAILURE; + } + + vStartMqttAgentTask(); + + vStartOtaTask(); + + vStartMlMqttTask(); + } + else + { + vMlTaskInferenceStart(); + } + + vStartBlinkTask(); + + vStartMlTask( NULL ); + + vTaskStartScheduler(); + + /* If all is well, the scheduler will now be running, and the following + * line will never be reached. If the following line does execute, then + * there was insufficient FreeRTOS heap memory available for the idle and/or + * timer tasks to be created. See the memory management section on the + * FreeRTOS web site for more details. NOTE: This demo uses static allocation + * for the idle and timer tasks so this line should never execute. */ + while( 1 ) + { + } + + return EXIT_FAILURE; +} + +/** + * Dummy implementation of the callback function vApplicationStackOverflowHook(). + */ +#if ( configCHECK_FOR_STACK_OVERFLOW > 0 ) + __WEAK void vApplicationStackOverflowHook( TaskHandle_t xTask, + char * pcTaskName ) + { + ( void ) xTask; + ( void ) pcTaskName; + + /* Assert when stack overflow is enabled but no application defined function exists */ + configASSERT( 0 ); + } +#endif + +/*---------------------------------------------------------------------------*/ +#if ( configSUPPORT_STATIC_ALLOCATION == 1 ) + +/* + * vApplicationGetIdleTaskMemory gets called when configSUPPORT_STATIC_ALLOCATION + * equals to 1 and is required for static memory allocation support. + */ + + __WEAK void vApplicationGetIdleTaskMemory( StaticTask_t ** ppxIdleTaskTCBBuffer, + StackType_t ** ppxIdleTaskStackBuffer, + uint32_t * pulIdleTaskStackSize ) + { + /* Idle task control block and stack */ + static StaticTask_t xIdleTCB = { 0 }; + static StackType_t xIdleStack[ configMINIMAL_STACK_SIZE ] = { 0 }; + + *ppxIdleTaskTCBBuffer = &xIdleTCB; + *ppxIdleTaskStackBuffer = &xIdleStack[ 0 ]; + *pulIdleTaskStackSize = ( uint32_t ) configMINIMAL_STACK_SIZE; + } + +/* + * vApplicationGetTimerTaskMemory gets called when configSUPPORT_STATIC_ALLOCATION + * equals to 1 and is required for static memory allocation support. + */ + __WEAK void vApplicationGetTimerTaskMemory( StaticTask_t ** ppxTimerTaskTCBBuffer, + StackType_t ** ppxTimerTaskStackBuffer, + uint32_t * pulTimerTaskStackSize ) + { + /* Timer task control block and stack */ + static StaticTask_t xTimerTCB = { 0 }; + static StackType_t xTimerStack[ configTIMER_TASK_STACK_DEPTH ] = { 0 }; + + *ppxTimerTaskTCBBuffer = &xTimerTCB; + *ppxTimerTaskStackBuffer = &xTimerStack[ 0 ]; + *pulTimerTaskStackSize = ( uint32_t ) configTIMER_TASK_STACK_DEPTH; + } +#endif /* if ( configSUPPORT_STATIC_ALLOCATION == 1 ) */ + +#ifdef AUDIO_VSI + void vVsiCallbackTask( void * arg ) + { + ( void ) arg; + + while( 1 ) + { + ( void ) ulTaskNotifyTake( pdTRUE, portMAX_DELAY ); + + if( ( ulVsiEvent & ARM_SAI_EVENT_RECEIVE_COMPLETE ) && ( pxOnVsiEvent ) ) + { + pxOnVsiEvent( pvVsiContext ); + } + + if( ulVsiEvent & ARM_SAI_EVENT_RX_OVERFLOW ) + { + LogError( ( "VSI Receive Overflow Error \r\n" ) ); + } + } + } + + static void prvStartVsiCallbackTask( void ) + { + if( xTaskCreate( vVsiCallbackTask, + "VSI_CALLBACK_TASK", + appCONFIG_VSI_CALLBACK_TASK_STACK_SIZE, + NULL, + appCONFIG_VSI_CALLBACK_TASK_PRIORITY, + &xVsiTaskHandle ) != pdPASS ) + { + LogError( ( "Failed to create Vsi Callback Task\r\n" ) ); + } + } +#endif /* ifdef AUDIO_VSI */ diff --git a/applications/object_detection/ml_interface.cc b/applications/object_detection/ml_interface.cc new file mode 100755 index 00000000..c162300b --- /dev/null +++ b/applications/object_detection/ml_interface.cc @@ -0,0 +1,512 @@ +/* Copyright 2021-2024 Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#include "FreeRTOS.h" +#include "queue.h" +#include "task.h" + +#include "ml_interface.h" +#include "AppContext.hpp" +#include "BufAttributes.hpp" +#include "demo_config.h" +extern "C" { +#include "events.h" +#ifdef USE_ETHOS +#include "ethosu_driver.h" +#include "ethosu_npu_init.h" +#endif +} +#include "DetectorPostProcessing.hpp" +#include "DetectorPreProcessing.hpp" +#include "mqtt_agent_task.h" +#include "TensorFlowLiteMicro.hpp" +#include "YoloFastestModel.hpp" +#include CMSIS_device_header +#include "log_macros.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +extern "C" { +/* Include header that defines log levels. */ +#include "logging_levels.h" +/* Configure name and log level. */ +#ifndef LIBRARY_LOG_NAME + #define LIBRARY_LOG_NAME "ML_IFC" +#endif +#ifndef LIBRARY_LOG_LEVEL + #define LIBRARY_LOG_LEVEL LOG_INFO +#endif +#include "logging_stack.h" + +#include "isp_config.h" +} + +#include "app_config.h" + +/** + * @brief The topic to subscribe and publish to in the example. + * + * The topic name starts with the client identifier to ensure that each demo + * interacts with a unique topic name. + */ +#define mqttexampleTOPIC democonfigCLIENT_IDENTIFIER "/ml/inference" + +/** + * @brief The MQTT agent manages the MQTT contexts. This set the handle to the + * context used by this demo. + */ +extern MQTTAgentContext_t xGlobalMqttAgentContext; + +/** + * @brief The maximum time for which application waits for an MQTT operation to be complete. + * This involves receiving an acknowledgment for broker for SUBSCRIBE, UNSUBSCRIBE and non + * QOS0 publishes. + */ +#define appMQTT_TIMEOUT_MS ( 5000U ) + +/** + * @brief Used to clear bits in a task's notification value. + */ +#define appMAX_UINT32 ( 0xffffffff ) + +extern EventGroupHandle_t xSystemEvents; +extern QueueHandle_t xMlMqttQueue; + +// Define tensor arena and declare functions required to access the model +namespace arm { + namespace app { + uint8_t ucTensorArena[ACTIVATION_BUF_SZ] ACTIVATION_BUF_ATTRIBUTE; + namespace object_detection { + extern uint8_t *GetModelPointer(); + extern size_t GetModelLen(); + } /* namespace object_detection */ + } /* namespace app */ +} /* namespace arm */ + +namespace { + +// Import +using namespace arm::app; + +// Model +arm::app::ApplicationContext xCaseContext; + +static int prvProcessImage(ApplicationContext &xApplicationContext, const uint8_t *pucImage, struct DetectRegion_t *pxCResults, uint32_t *pulResultsNum); + +extern "C" { + +static void prvAppPublishCommandCallback( MQTTAgentCommandContext_t * pxCommandContext, + MQTTAgentReturnInfo_t * pxReturnInfo ) +{ + pxCommandContext->xReturnStatus = pxReturnInfo->returnCode; + + if( pxCommandContext->xTaskToNotify != NULL ) + { + xTaskNotify( pxCommandContext->xTaskToNotify, ( uint32_t ) ( pxReturnInfo->returnCode ), eSetValueWithOverwrite ); + } +} + +static void prvMqttSendMessage( const char * pcMessage ) +{ + static MQTTPublishInfo_t xPublishInfo = { (MQTTQoS_t)0 }; + static MQTTAgentCommandInfo_t xCommandParams = { 0 }; + static MQTTAgentCommandContext_t xCommandContext = { (MQTTStatus_t)0 }; + MQTTStatus_t xMqttStatus = MQTTBadParameter; + + xPublishInfo.pTopicName = mqttexampleTOPIC; + xPublishInfo.topicNameLength = ( uint16_t ) strlen( mqttexampleTOPIC ); + xPublishInfo.qos = MQTTQoS1; + xPublishInfo.pPayload = pcMessage; + xPublishInfo.payloadLength = strlen( pcMessage ); + + xCommandContext.xTaskToNotify = xTaskGetCurrentTaskHandle(); + xTaskNotifyStateClear( NULL ); + + xCommandParams.blockTimeMs = appMQTT_TIMEOUT_MS; + xCommandParams.cmdCompleteCallback = prvAppPublishCommandCallback; + xCommandParams.pCmdCompleteCallbackContext = ( MQTTAgentCommandContext_t * ) &xCommandContext; + + LogInfo( ( "Attempting to publish (%s) to the MQTT topic %s.\r\n", pcMessage, mqttexampleTOPIC ) ); + xMqttStatus = MQTTAgent_Publish( &xGlobalMqttAgentContext, + &xPublishInfo, + &xCommandParams ); + + /* Wait for command to complete so MQTTSubscribeInfo_t remains in scope for the + * duration of the command. */ + if( xMqttStatus == MQTTSuccess ) + { + BaseType_t xResult = xTaskNotifyWait( 0, appMAX_UINT32, NULL, pdMS_TO_TICKS( appMQTT_TIMEOUT_MS ) ); + + if( xResult != pdTRUE ) + { + xMqttStatus = MQTTSendFailed; + } + else + { + xMqttStatus = xCommandContext.xReturnStatus; + } + } + + if( xMqttStatus != MQTTSuccess ) + { + LogError( ( "Failed to publish result over MQTT" ) ); + } + else + { + LogInfo( ( "Sent PUBLISH packet to broker %.*s to broker.\n", + strlen( mqttexampleTOPIC ), + mqttexampleTOPIC ) ); + } +} + +void vMlTaskInferenceStart(void) +{ + if(xSystemEvents == NULL) + { + LogError( ( "xSystemEvents is not initialised\r\n" ) ); + return; + } + + LogInfo( ( "Signal task inference start\r\n" ) ); + ( void )xEventGroupClearBits( xSystemEvents, (EventBits_t)EVENT_MASK_ML_STOP ); + ( void )xEventGroupSetBits( xSystemEvents, (EventBits_t)EVENT_MASK_ML_START ); + +} + +void vMlTaskInferenceStop(void) +{ + if(xSystemEvents == NULL) + { + LogError( ( "xSystemEvents is not initialised\r\n" ) ); + return; + } + + LogInfo( ( "Signal task inference stop\r\n" ) ); + ( void )xEventGroupClearBits( xSystemEvents, (EventBits_t)EVENT_MASK_ML_START ); + ( void )xEventGroupSetBits( xSystemEvents, (EventBits_t)EVENT_MASK_ML_STOP ); +} + +void vStartMlTask( void *pvParameters ) +{ + if ( + xTaskCreate( + vMlTask, + "ML_TASK", + appCONFIG_ML_TASK_STACK_SIZE, + pvParameters, + appCONFIG_ML_TASK_PRIORITY, + NULL + ) != pdPASS + ) { + LogError( ( "Failed to create ML Task\r\n" ) ); + } +} + +void vStartMlMqttTask( void ) +{ + if ( + xTaskCreate( + vMlMqttTask, + "ML_MQTT", + appCONFIG_ML_MQTT_TASK_STACK_SIZE, + NULL, + appCONFIG_ML_MQTT_TASK_PRIORITY, + NULL + ) != pdPASS + ) { + LogError( ( "Failed to create ML Mqtt Task\r\n" ) ); + } +} + +int32_t lMLRunInference(const uint8_t *pucImg, struct DetectRegion_t *pxResults, uint32_t *pulResultsNum) +{ + prvProcessImage(xCaseContext, pucImg, pxResults, pulResultsNum); + return 0; +} +} // extern "C" { + +static void prvSetMlProcessingstate(const char *pcInferenceResult) +{ + size_t xMsgLen = strlen(pcInferenceResult) + 1; + char *pcMsgResult = reinterpret_cast(malloc(xMsgLen)); + if (pcMsgResult) { + if(xMlMqttQueue == NULL) + { + LogError( ( "xMlMqttQueue is not initialised\r\n" ) ); + free(reinterpret_cast(pcMsgResult)); + return; + } + + memcpy(pcMsgResult, pcInferenceResult, xMsgLen); + const MLMqttMsg_t msg = {pcMsgResult}; + if (xQueueSendToBack(xMlMqttQueue, (void *)&msg, (TickType_t)0) != pdTRUE) { + LogError( ( "Failed to send message to xMlMqttQueue\r\n" ) ); + free(reinterpret_cast(pcMsgResult)); + } + } else { + LogWarn( ( "Failed to send ml processing inference_result (alloc failure)" ) ); + } +} + +/** + * @brief Presents inference results using the data presentation + * object. + * @param[in] platform Reference to the hal platform object. + * @param[in] results Vector of classification results to be displayed. + * @return true if successful, false otherwise. + **/ +static bool prvPresentInferenceResult(const std::vector &xResults); + +static int prvProcessImage(ApplicationContext &xApplicationContext, const uint8_t *pucImage, struct DetectRegion_t *pxCResults, uint32_t *pulResultsNum) +{ + // Get the global model + auto &xModel = xApplicationContext.Get("model"); + + if (!xModel.IsInited()) { + LogError( ("Model is not initialised! Terminating processing.\n") ); + return -1; + } + + TfLiteTensor *xInputTensor = xModel.GetInputTensor(0); + TfLiteTensor *xOutputTensor0 = xModel.GetOutputTensor(0); + TfLiteTensor *xOutputTensor1 = xModel.GetOutputTensor(1); + + if (!xInputTensor->dims) { + LogError( ("Invalid input tensor dims\n") ); + return -1; + } else if (xInputTensor->dims->size < 3) { + LogError( ("Input tensor dimension should be >= 3\n") ); + return -1; + } + + TfLiteIntArray *pxInputShape = xModel.GetInputShape(0); + + const int lInputImgCols = pxInputShape->data[YoloFastestModel::ms_inputColsIdx]; + const int lInputImgRows = pxInputShape->data[YoloFastestModel::ms_inputRowsIdx]; + + /* Set up pre and post-processing. */ + /* RGB to grayscale skipped, already done outside */ + DetectorPreProcess xPreProcess = DetectorPreProcess(xInputTensor, false, xModel.IsDataSigned()); + + std::vector xResults; + const object_detection::PostProcessParams xPostProcessParams{lInputImgRows, + lInputImgCols, + object_detection::originalImageSize, + object_detection::anchor1, + object_detection::anchor2}; + DetectorPostProcess xPostProcess = DetectorPostProcess(xOutputTensor0, xOutputTensor1, xResults, xPostProcessParams); + /* Ensure there are no results leftover from previous inference when running all. */ + xResults.clear(); + + /* Run the pre-processing, inference and post-processing. */ + if (!xPreProcess.DoPreProcess(pucImage, xInputTensor->bytes)) { + LogError( ("Pre-processing failed.") ); + return -1; + } + + /* Run inference over this image. */ + info("Running inference on image at addr 0x%x\n", (uint32_t)pucImage); + + if (!xModel.RunInference()) { + LogError( ("Inference failed.") ); + return -1; + } + + if (!xPostProcess.DoPostProcess()) { + LogError( ("Post-processing failed.") ); + return -1; + } + + for (uint32_t i = 0; i < xResults.size() && i < *pulResultsNum; ++i) { + pxCResults[i].ulX = xResults[i].m_x0; + pxCResults[i].ulY = xResults[i].m_y0; + pxCResults[i].ulW = xResults[i].m_w; + pxCResults[i].ulH = xResults[i].m_h; + } + if (!prvPresentInferenceResult(xResults)) { + return -1; + } + if (*pulResultsNum > xResults.size()) { + *pulResultsNum = xResults.size(); + } + + return 0; +} + +static bool prvPresentInferenceResult(const std::vector &xResults) +{ + /* If profiling is enabled, and the time is valid. */ + LogInfo( ("Final results:\n") ); + LogInfo( ("Total number of inferences: 1\n") ); + LogInfo( ("Detected faces: %d\n", xResults.size()) ); + + for (uint32_t i = 0; i < xResults.size(); ++i) { + LogInfo( ("%" PRIu32 ") (%f) -> %s {x=%d,y=%d,w=%d,h=%d}\n", + i, + xResults[i].m_normalisedVal, + "Detection box:", + xResults[i].m_x0, + xResults[i].m_y0, + xResults[i].m_w, + xResults[i].m_h) ); + } + + std::string xFinalResultStr = "Detected faces: "; + xFinalResultStr += std::to_string( xResults.size() ); + + LogInfo( ( "Complete recognition: %s\n", xFinalResultStr.c_str() ) ); + + // Send the inference result + prvSetMlProcessingstate(xFinalResultStr.c_str()); + + return true; +} + +} // anonymous namespace + +#ifdef USE_ETHOS +extern struct ethosu_driver ethosu_drv; /* Default Ethos-U55 device driver */ + +/** + * @brief Initialises the Arm Ethos-U55 NPU + * @return 0 if successful, error code otherwise + **/ +static int prvArmNpuInit(void); + +static int prvArmNpuInit(void) +{ + int lErr = 0; + + SCB_EnableICache(); + SCB_EnableDCache(); + +#if defined(ETHOS_U_NPU_TIMING_ADAPTER_ENABLED) + /* If the platform has timing adapter blocks along with Ethos-U core + * block, initialise them here. */ + if (0 != (err = arm_ethosu_timing_adapter_init())) { + LogError( ("Failed to init timing adapter\n") ); + return lErr; + } +#endif /* ETHOS_U_NPU_TIMING_ADAPTER_ENABLED */ + + // Initialize the ethos NPU + if (0 != (lErr = arm_ethosu_npu_init())) { + LogError( ("Failed to init arm npu\n") ); + return lErr; + } + + LogInfo( ( "Ethos-U55 device initialised\n" ) ); + + /* Get Ethos-U55 version */ + struct ethosu_driver_version xDriverVersion; + struct ethosu_hw_info xHwInfo; + + ethosu_get_driver_version(&xDriverVersion); + ethosu_get_hw_info(ðosu_drv, &xHwInfo); + + LogInfo( ( "Ethos-U version info:\n" ) ); + LogInfo( ( "\tArch: v%" PRIu32 ".%" PRIu32 ".%" PRIu32 "\n", + xHwInfo.version.arch_major_rev, + xHwInfo.version.arch_minor_rev, + xHwInfo.version.arch_patch_rev ) ); + LogInfo( ( "\tDriver: v%" PRIu8 ".%" PRIu8 ".%" PRIu8 "\n", + xDriverVersion.major, + xDriverVersion.minor, + xDriverVersion.patch ) ); + LogInfo( ( "\tMACs/cc: %" PRIu32 "\n", (uint32_t)(1 << xHwInfo.cfg.macs_per_cc) ) ); + LogInfo( ( "\tCmd stream: v%" PRIu32 "\n", xHwInfo.cfg.cmd_stream_version ) ); + + return 0; +} +#endif /* USE_ETHOS */ + +extern "C" { + +static int prvMlInterfaceInit(void) +{ + static arm::app::YoloFastestModel xModel; /* Model wrapper object. */ + +#ifdef USE_ETHOS + // Initialize the ethos U55 + if (prvArmNpuInit() != 0) { + LogError( ( "Failed to arm npu\n" ) ); + return -1; + } +#endif /* USE_ETHOS */ + + /* Load the model. */ + if (!xModel.Init(arm::app::ucTensorArena, + sizeof(arm::app::ucTensorArena), + arm::app::object_detection::GetModelPointer(), + arm::app::object_detection::GetModelLen())) { + LogError( ( "Failed to initialise model\n" ) ); + return -1; + } + + xModel.ShowModelInfoHandler(); + + /* Instantiate application context. */ + xCaseContext.Set("model", xModel); + + PrintTensorFlowVersion(); + LogInfo( ( "*** ML interface initialised\r\n" ) ); + return 0; +} + +void vMlTask(void *pvParameters) +{ + LogInfo( ( "ML Task start\r\n" ) ); + + EventBits_t xFlags = xEventGroupWaitBits( + xSystemEvents, (EventBits_t)EVENT_MASK_ML_START, pdTRUE, pdFAIL, portMAX_DELAY + ); + + if (xFlags & EVENT_MASK_ML_START) { + LogInfo( ( "Initial start of image processing\r\n" ) ); + } + + if (prvMlInterfaceInit() < 0) { + LogError( ( "prvMlInterfaceInit failed\r\n" ) ); + return; + } + + vStartISPDemo(); + + while (1) { + xFlags = xEventGroupWaitBits( + xSystemEvents, (EventBits_t)EVENT_MASK_ML_STOP, pdTRUE, pdFAIL, 300 + ); + + if (xFlags & EVENT_MASK_ML_STOP) { + LogInfo( ( "Stopping image processing\r\n" ) ); + break; + } + } +} + +void vMlMqttTask(void *pvParameters) +{ + (void)pvParameters; + + while (1) { + MLMqttMsg_t xMsg; + if (xQueueReceive (xMlMqttQueue, &xMsg, portMAX_DELAY) == pdTRUE) { + prvMqttSendMessage(xMsg.pcResult); + free(reinterpret_cast(xMsg.pcResult)); + } else { + LogError( ( "xQueueReceive ML MQTT msg queue failed\r\n" ) ); + } + } +} + +} // extern "C" diff --git a/applications/object_detection/ml_interface.h b/applications/object_detection/ml_interface.h new file mode 100644 index 00000000..c33559af --- /dev/null +++ b/applications/object_detection/ml_interface.h @@ -0,0 +1,71 @@ +/* Copyright 2021-2024, Arm Limited and/or its affiliates + * + * SPDX-License-Identifier: MIT + */ + +#ifndef ML_INTERFACE_H + #define ML_INTERFACE_H + + #include + + #ifdef __cplusplus + extern "C" { + #endif + + struct DetectRegion_t + { + uint32_t ulX; + uint32_t ulY; + uint32_t ulW; + uint32_t ulH; + }; + + typedef struct + { + char * pcResult; + } MLMqttMsg_t; + +/** + * @brief Start the inference task. + */ + void vMlTaskInferenceStart( void ); + +/** + * @brief Stop the inference task. + */ + void vMlTaskInferenceStop( void ); + +/** + * @brief Task to perform ML processing. + * It is gated by the net task which lets it run + * if no ota job is present. + * @param pvParameters Contextual data for the task. + */ + void vMlTask( void * pvParameters ); + +/** + * @brief Task to communicate ML results via MQTT. + * @param pvParameters Contextual data for the task. + */ + void vMlMqttTask( void * pvParameters ); + +/** + * @brief Create ML MQTT task. + */ + void vStartMlMqttTask( void ); + +/** + * @brief Create ML task. + * @param pvParameters Contextual data for the task. + */ + void vStartMlTask( void * pvParameters ); + + int32_t lMLRunInference( const uint8_t * pucImg, + struct DetectRegion_t * pxResults, + uint32_t * pulResultsNum ); + + #ifdef __cplusplus + } + #endif + +#endif /* ! ML_INTERFACE_H */ diff --git a/applications/object_detection/resources/test.frm b/applications/object_detection/resources/test.frm new file mode 100644 index 0000000000000000000000000000000000000000..18ade61da260bfb5b581660d6ed9bb7507bca266 GIT binary patch literal 663616 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 <~XKhr0wo}J1S30R+kKTXX0sXl*j0^ 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

Nu_-wM8o2;;s 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 --private_key_path -t corstone315 --toolchain GNU +``` +- The `certificate pem's path` and `private key pem's path` should be the downloaded key's and certificate's path if you chose the **Auto-generate a new certificate** during the Thing creation. If you chose **Skip creating a certificate at this time** then these paths should locate the generated credential files that were created by the `./tools/scripts/generate_credentials.py` script in the previous step. + +- The `toolchain` is used to select the `GNU`, that supports the `Mali-C55`. + +### Or, run the command below to perform a clean build: +```bash +./tools/scripts/build.sh object-detection --certificate_path --private_key_path -t corstone315 --toolchain GNU -c +``` + +## Provisioning the device credentials into Protected Storage +Check [Device Provisioning](./device_provisioning.md) for detailed information. + +## Running the application + +```bash +./tools/scripts/run.sh object-detection -t corstone315 --frames applications/object_detection/resources/test.frm +``` +- The `frames` is used to select the input frames file. + +### Expected output + +```log +[INF] Starting bootloader +[INF] Primary image: magic=good, swap_type=0x1, copy_done=0x3, image_ok=0x1 +[INF] Scratch: magic=bad, swap_type=0x1, copy_done=0x2, image_ok=0x2 +[INF] Boot source: primary slot +[INF] Image index: 1, Swap type: test +[INF] Starting swap using scratch algorithm. +[INF] Primary image: magic=good, swap_type=0x1, copy_done=0x3, image_ok=0x1 +[INF] Scratch: magic=unset, swap_type=0x1, copy_done=0x3, image_ok=0x3 +[INF] Boot source: primary slot +[INF] Image index: 0, Swap type: none +[INF] Bootloader chainload address offset: 0x0 +[INF] Jumping to the first image slot +Booting TF-M v2.0.0 +[Sec Thread] Secure image initializing! +[INF][Crypto] Provisioning entropy seed... complete. +0 0 [None] [INFO] PSA Framework version is: 257 +1 0 [None] Write certificate... +2 0 [None] [INFO] Device key provisioning succeeded +3 0 [None] [INFO] OTA signing key provisioning succeeded +4 0 [None] [INFO] Signal task inference stop +5 0 [None] FreeRTOS_AddEndPoint: MAC: 44-21 IPv4: c0a80069ip +6 41 [OTA Task ] [INFO] OTA over MQTT, Application version from appFirmwareVersion 0.0.20 +7 54 [ML_TASK] [INFO] ML Task start +8 59 [BLINK_TASK ] [INFO] Blink task started +9 1000 [IP-Task] DHCP-socket[44-21]: DHCP Socket Create +10 1000 [IP-Task] prvCreateDHCPSocket[44-21]: open, user count 1 +11 1000 [IP-Task] vDHCP_RATimerReload: 250 +12 1250 [IP-Task] vDHCPProcess: offer ac143301ip for MAC address 44-21 +13 1260 [IP-Task] vDHCPProcess: offer ac143301ip for MAC address 44-21 +14 1260 [IP-Task] [INFO] Network is up +15 1260 [IP-Task] prvCloseDHCPSocket[44-21]: closed, user count 0 +16 1260 [IP-Task] vDHCP_RATimerReload: 8640000 +17 1291 [MQTT Agent Task] [INFO] Creating a TLS connection to a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883. +18 1307 [MQTT Agent Task] [INFO] Resolving host name: a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com. +19 1322 [IP-Task] ipARP_REPLY from ac1433feip to ac143301ip end-point ac143301ip +20 6334 [MQTT Agent Task] DNS_ReadReply returns -11 +21 6341 [MQTT Agent Task] prvIncreaseDNS4Index: from 0 to 0 +22 6356 [MQTT Agent Task] DNS[0xE135]: The answer to 'a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com' (99.81.255.142) will be stored +23 6374 [MQTT Agent Task] [INFO] Initiating TCP connection with host: a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883 +24 6392 [MQTT Agent Task] FreeRTOS_connect: 37762 to 6351ff8eip:8883 +25 6492 [MQTT Agent Task] [INFO] Initiating TLS handshake with host: a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883 +26 8622 [MQTT Agent Task] [INFO] Successfully created a TLS connection to a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883. +27 8640 [MQTT Agent Task] [INFO] Creating an MQTT connection to the broker. +28 8993 [MQTT Agent Task] [INFO] MQTT connection established with the broker. +29 9004 [MQTT Agent Task] [INFO] Successfully connected to the MQTT broker. +30 9014 [MQTT Agent Task] [INFO] Session present: 0 +31 9022 [MQTT Agent Task] [INFO] Starting a clean MQTT Session. +32 9031 [OTA Task ] [INFO] Received: 0 Queued: 0 Processed: 0 Dropped: 0 +33 9042 [OTA Agent Task] [INFO] Current State=[RequestingJob], Event=[Start], New state=[RequestingJob] +34 9413 [OTA Agent Task] [INFO] Subscribed to topic $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/notify-next. +35 9432 [OTA Agent Task] [INFO] Subscribed to MQTT topic: $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/notify-next +36 10163 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/$next/get. +37 10484 [MQTT Agent Task] [INFO] Ack packet deserialized with result: MQTTSuccess. +38 10496 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +39 10508 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/$next/get to broker. +40 10530 [OTA Agent Task] [WARN] OTA Timer handle NULL for Timerid=0, can't stop. +41 10542 [OTA Agent Task] [INFO] Current State=[WaitingForJob], Event=[RequestJobDocument], New state=[WaitingForJob] +42 11258 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +43 11272 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +44 11284 [MQTT Agent Task] [INFO] Received job message callback, size 1112. +45 11295 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.jobId: AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a] +46 11318 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.statusDetails.updatedBy: 10] +47 11335 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.jobDocument.afr_ota.streamname: AFR_OTA-eac26517-1848-4705-8242-97bda52cab40] +48 11358 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.jobDocument.afr_ota.protocols: ["MQTT"]] +49 11375 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[filepath: non_secure image] +50 11389 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[filesize: 1141641] +51 11402 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[fileid: 0] +52 11413 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[certfile: 0] +53 11426 [OTA Agent Task] [INFO] Extracted parameter [ sig-sha256-rsa: LUOGFk3dx7VoXvS7tJl5MJObgHtbCJgZ... ] +54 11442 [OTA Agent Task] [INFO] In self test mode. +55 11450 [OTA Agent Task] [INFO] New image has a higher version number than the current image: New image version=0.0.20, Previous image version=0.0.10 +56 11471 [OTA Agent Task] [INFO] Image version is valid: Begin testing file: File ID=0 +57 12044 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/update. +58 12372 [MQTT Agent Task] [INFO] Ack packet deserialized with result: MQTTSuccess. +59 12383 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +60 12395 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +61 12410 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +62 12421 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/update to broker. +63 12452 [OTA Agent Task] [INFO] Job parsing success: OtaJobParseErr_t=OtaJobParseErrNone, Job name=AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a +64 12475 [OTA Agent Task] [INFO] Signal task inference stop +65 12484 [OTA Agent Task] [INFO] Current State=[CreatingFile], Event=[ReceivedJobDocument], New state=[CreatingFile] +66 12500 [OTA Agent Task] [INFO] Beginning self-test. +67 12508 [OTA Agent Task] [INFO] Received OtaJobEventStartTest callback from OTA Agent. +68 13171 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/update. +69 13498 [MQTT Agent Task] [INFO] Ack packet deserialized with result: MQTTSuccess. +70 13510 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +71 13521 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +72 13536 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +73 13548 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/update to broker. +74 13579 [OTA Agent Task] [INFO] New image validation succeeded in self test mode. +75 13590 [OTA Agent Task] [INFO] OTA active state `4` from OTA Agent. +76 13600 [OTA Agent Task] [INFO] Signal task inference stop +77 13608 [OTA Agent Task] [WARN] OTA Timer handle NULL for Timerid=1, can't stop. +78 13620 [OTA Agent Task] [INFO] Current State=[WaitingForJob], Event=[StartSelfTest], New state=[WaitingForJob] +79 17048 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +80 17062 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPubAckSend. +81 17074 [MQTT Agent Task] [INFO] Received job message callback, size 24. +82 17085 [OTA Agent Task] [INFO] No active job available in received job document: OtaJobParseErr_t=OtaJobParseErrNoActiveJobs +83 17102 [OTA Agent Task] [INFO] Signal task inference start +84 17111 [OTA Agent Task] [INFO] Current State=[WaitingForJob], Event=[ReceivedJobDocument], New state=[CreatingFile] +85 17127 [ML_TASK] [INFO] Initial start of image processing +INFO - Ethos-U device initialised +INFO - Ethos-U version info: +INFO - Arch: v1.1.0 +INFO - Driver: v0.16.0 +INFO - MACs/cc: 256 +INFO - Cmd stream: v0 +86 17159 [ML_TASK] [INFO] Ethos-U65 device initialised +87 17167 [ML_TASK] [INFO] Ethos-U version info: +88 17173 [ML_TASK] [INFO] Arch: v1.1.0 +89 17180 [ML_TASK] [INFO] Driver: v0.16.0 +90 17187 [ML_TASK] [INFO] MACs/cc: 256 +91 17192 [ML_TASK] [INFO] Cmd stream: v0 +INFO - Added ethos-u support to op resolver +INFO - Creating allocator using tensor arena at 0x0x6006b440 +INFO - Allocating tensors +INFO - Model INPUT tensors: +INFO - tensor type is INT8 +INFO - tensor occupies 36864 bytes with dimensions +INFO - 0: 1 +INFO - 1: 192 +INFO - 2: 192 +INFO - 3: 1 +INFO - Quant dimension: 0 +INFO - Scale[0] = 0.003921 +INFO - ZeroPoint[0] = -128 +INFO - Model OUTPUT tensors: +INFO - tensor type is INT8 +INFO - tensor occupies 648 bytes with dimensions +INFO - 0: 1 +INFO - 1: 6 +INFO - 2: 6 +INFO - 3: 18 +INFO - Quant dimension: 0 +INFO - Scale[0] = 0.134084 +INFO - ZeroPoint[0] = 47 +INFO - tensor type is INT8 +INFO - tensor occupies 2592 bytes with dimensions +INFO - 0: 1 +INFO - 1: 12 +INFO - 2: 12 +INFO - 3: 18 +INFO - Quant dimension: 0 +INFO - Scale[0] = 0.185359 +INFO - ZeroPoint[0] = 10 +INFO - Activation buffer (a.k.a tensor arena) size used: 370024 +INFO - Number of operators: 1 +INFO - Operator 0: ethos-u +INFO - Model address: 0x0x60000000INFO - Model size: 439360 bytes.INFO - Model info: +INFO - Model INPUT tensors: +INFO - tensor type is INT8 +INFO - tensor occupies 36864 bytes with dimensions +INFO - 0: 1 +INFO - 1: 192 +INFO - 2: 192 +INFO - 3: 1 +INFO - Quant dimension: 0 +INFO - Scale[0] = 0.003921 +INFO - ZeroPoint[0] = -128 +INFO - Model OUTPUT tensors: +INFO - tensor type is INT8 +INFO - tensor occupies 648 bytes with dimensions +INFO - 0: 1 +INFO - 1: 6 +INFO - 2: 6 +INFO - 3: 18 +INFO - Quant dimension: 0 +INFO - Scale[0] = 0.134084 +INFO - ZeroPoint[0] = 47 +INFO - tensor type is INT8 +INFO - tensor occupies 2592 bytes with dimensions +INFO - 0: 1 +INFO - 1: 12 +INFO - 2: 12 +INFO - 3: 18 +INFO - Quant dimension: 0 +INFO - Scale[0] = 0.185359 +INFO - ZeroPoint[0] = 10 +INFO - Activation buffer (a.k.a tensor arena) size used: 370024 +INFO - Number of operators: 1 +INFO - Operator 0: ethos-u +INFO - The model is optimised for Ethos-U NPU: yes. +92 17474 [ML_TASK] [INFO] *** ML interface initialised +93 17481 [ML_TASK] [INFO] ISP init +94 17486 [ML_TASK] [INFO] Starting HDLCD config! +95 17493 [ML_TASK] [INFO] Starting ISP init! +00T00:00:17.500Z GENERIC(ERR) :PINGISP +1:4830ab6c(58140) <-> 0x212f6f7c(58140) +2:48318e88(16384) <-> 0x21305298(16384) +00T00:00:17.517Z GENERIC(ERR) :PoNGISP +1:48322b2c(58140) <-> 0x212f6f7c(58140) +2:48330e48(16384) <-> 0x21305298(16384) +00T00:00:17.534Z GENERIC(ERR) :PINGMET +1:483054a8(4096) <-> 0x212f18b8(4096) +2:483095ac(4672) <-> 0x212f59bc(4672) +00T00:00:17.550Z GENERIC(ERR) :PoNGMET +1:483054a8(4096) <-> 0x212f18b8(4096) +2:4832156c(4672) <-> 0x212f59bc(4672) +00T00:00:17.567Z SENSOR(NOTICE) :Sensor initialization is complete, ID 0xFFFF resolution 576x576 +00T00:00:17.581Z GENERIC(ERR) :DMA alloc: 0xca800, Memory left: 0x47b800 +00T00:00:17.596Z CROP(NOTICE) :FR update: Crop: e 0 x 0, y 0, w 576, h 576, Downscaler: e 0, w 576, h 576 +00T00:00:17.611Z CROP(NOTICE) :DS update: Crop: e 0 x 0, y 0, w 576, h 576, Downscaler: e 0, w 576, h 576 +00T00:00:17.636Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #0 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #0 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #0 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #0 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #0 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #0 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #0 is ready. +00T00:00:17.646Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 0 +00T00:00:17.646Z GENERIC(ERR) :Attempt to start a new frame before processing is done for the previous one. Skip this frame. +00T00:00:17.762Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #1 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #1 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #1 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #1 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #1 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #1 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #1 is ready. +00T00:00:17.772Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 1 +00T00:00:17.888Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #2 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #2 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #2 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #2 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #2 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #2 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #2 is ready. +00T00:00:17.898Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 2 +00T00:00:18.014Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #3 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #3 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #3 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #3 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #3 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #3 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #3 is ready. +00T00:00:18.024Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 3 +00T00:00:18.140Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #4 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #4 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #4 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #4 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #4 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #4 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #4 is ready. +00T00:00:18.150Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 4 +00T00:00:18.155Z GENERIC(CRIT) :-- aframes->frame_id = 1 -- +00T00:00:18.165Z GENERIC(CRIT) :-- 576 X 576 @ 2 bytes per pixel -- +96 18177 [acamera] [INFO] Displaying image... +97 18183 [acamera] [INFO] Image displayed +00T00:00:18.189Z GENERIC(CRIT) :-- aframes->frame_id = 1 -- +00T00:00:18.199Z GENERIC(CRIT) :-- 576 X 576 @ 2 bytes per pixel -- +98 18211 [acamera] [ERROR] Input frame too big for inference! +00T00:00:18.235Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #5 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #5 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #5 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #5 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #5 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #5 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #5 is ready. +00T00:00:18.245Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 5 +00T00:00:18.252Z GENERIC(CRIT) :-- aframes->frame_id = 2 -- +00T00:00:18.262Z GENERIC(CRIT) :-- 576 X 576 @ 2 bytes per pixel -- +99 18274 [acamera] [INFO] Displaying image... +100 18280 [acamera] [INFO] Image displayed +00T00:00:18.286Z GENERIC(CRIT) :-- aframes->frame_id = 2 -- +00T00:00:18.296Z GENERIC(CRIT) :-- 192 X 192 @ 2 bytes per pixel -- +101 18308 [acamera] [INFO] Converting to Gray: 0x82f76000 -> 0x21224300 +INFO - Running inference on image at addr 0x21224300 +102 18340 [acamera] [INFO] Final results: +103 18346 [acamera] [INFO] Total number of inferences: 1 +104 18354 [acamera] [INFO] Detected faces: 3 +105 18360 [acamera] [INFO] 0) (0.994434) -> Detection box: {x=6,y=71,w=33,h=44} +106 18371 [acamera] [INFO] 1) (0.950913) -> Detection box: {x=108,y=71,w=13,h=21} +107 18383 [acamera] [INFO] 2) (0.884787) -> Detection box: {x=47,y=78,w=20,h=31} +108 18394 [acamera] [INFO] Complete recognition: Detected faces: 3 +109 18534 [ML_MQTT] [INFO] Attempting to publish (Detected faces: 3) to the MQTT topic iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/ml/inference. +110 18557 [MQTT Agent Task] [INFO] Publishing message to iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/ml/inference. +00T00:00:18.575Z GENERIC(CRIT) :-- CAMERA STREAM TRIGGER #6 -- +abstract_port__vsync_s__setValue : (idx = 0) Frame #6 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #6 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #6 is ready. +abstract_port__vsync_s__setValue : (idx = 0) Frame #6 is ready. +abstract_port__vsync_s__setValue : (idx = 1) Frame #6 is ready. +abstract_port__vsync_s__setValue : (idx = 2) Frame #6 is ready. +00T00:00:18.585Z GENERIC(CRIT) :[KeyMsg]: FS interrupt: 6 +00T00:00:18.588Z GENERIC(CRIT) :-- aframes->frame_id = 3 -- +00T00:00:18.598Z GENERIC(CRIT) :-- 576 X 576 @ 2 bytes per pixel -- +111 18610 [acamera] [INFO] Displaying image... +112 18616 [acamera] [INFO] Image displayed +00T00:00:18.622Z GENERIC(CRIT) :-- aframes->frame_id = 3 -- +00T00:00:18.633Z GENERIC(CRIT) :-- 192 X 192 @ 2 bytes per pixel -- +113 18644 [acamera] [INFO] Converting to Gray: 0x837e6000 -> 0x21224300 +INFO - Running inference on image at addr 0x21224300 +114 18676 [acamera] [INFO] Final results: +115 18682 [acamera] [INFO] Total number of inferences: 1 +116 18690 [acamera] [INFO] Detected faces: 4 +117 18696 [acamera] [INFO] 0) (0.996794) -> Detection box: {x=6,y=71,w=33,h=44} +118 18707 [acamera] [INFO] 1) (0.986066) -> Detection box: {x=108,y=71,w=13,h=21} +119 18719 [acamera] [INFO] 2) (0.958923) -> Detection box: {x=47,y=78,w=20,h=31} +120 18730 [acamera] [INFO] 3) (0.546118) -> Detection box: {x=170,y=91,w=16,h=21} +121 18741 [acamera] [INFO] Complete recognition: Detected faces: 4 +. +OTA update status: DELETE_IN_PROGRESS +An error occurred (ResourceNotFoundException) when calling the GetOTAUpdate operation: Requested otaUpdateJob ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a does not exist +Deleted update ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a +Deleted thing arn:aws:iot:eu-west-1:965412032949:thing/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a +Deleted S3 object object-detection-update_signed.bin from iotmsw-ci-test-bucket-9ba0432f-e481-4fde-b3e5-a45bcf076e7a +Deleted S3 bucket iotmsw-ci-test-bucket-9ba0432f-e481-4fde-b3e5-a45bcf076e7a +``` + +## Observing MQTT connectivity +Follow the instructions described in the [Observing MQTT connectivity](./aws_iot_cloud_connection.md) section. + +## Firmware update with AWS +Follow the instructions described in the [Firmware update with AWS](./aws_iot_cloud_connection.md) section. + +### Expected output + +```log +[INF] Starting bootloader +[INF] Primary image: magic=good, swap_type=0x1, copy_done=0x3, image_ok=0x1 +[INF] Scratch: magic=bad, swap_type=0x1, copy_done=0x2, image_ok=0x2 +[INF] Boot source: primary slot +[INF] Image index: 1, Swap type: none +[INF] Primary image: magic=good, swap_type=0x1, copy_done=0x3, image_ok=0x1 +[INF] Scratch: magic=bad, swap_type=0x1, copy_done=0x2, image_ok=0x2 +[INF] Boot source: primary slot +[INF] Image index: 0, Swap type: none +[INF] Bootloader chainload address offset: 0x0 +[INF] Jumping to the first image slot +Booting TF-M v2.0.0 +[Sec Thread] Secure image initializing! +Creating an empty ITS flash layout. +Creating an empty PS flash layout. +[INF][Crypto] Provisioning entropy seed... complete. +0 0 [None] [INFO] PSA Framework version is: 257 +1 0 [None] Write certificate... +2 0 [None] [INFO] Device key provisioning succeeded +3 0 [None] [INFO] OTA signing key provisioning succeeded +4 0 [None] [INFO] Signal task inference stop +5 0 [None] FreeRTOS_AddEndPoint: MAC: 44-21 IPv4: c0a80069ip +6 41 [OTA Task ] [INFO] OTA over MQTT, Application version from appFirmwareVersion 0.0.10 +7 54 [ML_TASK] [INFO] ML Task start +8 59 [BLINK_TASK ] [INFO] Blink task started +9 1000 [IP-Task] DHCP-socket[44-21]: DHCP Socket Create +10 1000 [IP-Task] prvCreateDHCPSocket[44-21]: open, user count 1 +11 1000 [IP-Task] vDHCP_RATimerReload: 250 +12 1250 [IP-Task] vDHCPProcess: offer ac143301ip for MAC address 44-21 +13 1260 [IP-Task] vDHCPProcess: offer ac143301ip for MAC address 44-21 +14 1260 [IP-Task] [INFO] Network is up +15 1260 [IP-Task] prvCloseDHCPSocket[44-21]: closed, user count 0 +16 1260 [IP-Task] vDHCP_RATimerReload: 8640000 +17 1291 [MQTT Agent Task] [INFO] Creating a TLS connection to a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883. +18 1307 [MQTT Agent Task] [INFO] Resolving host name: a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com. +19 1323 [IP-Task] ipARP_REPLY from ac1433feip to ac143301ip end-point ac143301ip +20 6334 [MQTT Agent Task] DNS_ReadReply returns -11 +21 6341 [MQTT Agent Task] prvIncreaseDNS4Index: from 0 to 0 +22 6352 [MQTT Agent Task] DNS[0xB324]: The answer to 'a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com' (52.208.143.40) will be stored +23 6371 [MQTT Agent Task] [INFO] Initiating TCP connection with host: a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883 +24 6388 [MQTT Agent Task] FreeRTOS_connect: 25748 to 34d08f28ip:8883 +25 6482 [MQTT Agent Task] [INFO] Initiating TLS handshake with host: a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883 +26 8653 [MQTT Agent Task] [INFO] Successfully created a TLS connection to a1nn4qspc1z547-ats.iot.eu-west-1.amazonaws.com:8883. +27 8671 [MQTT Agent Task] [INFO] Creating an MQTT connection to the broker. +28 9099 [MQTT Agent Task] [INFO] MQTT connection established with the broker. +29 9110 [MQTT Agent Task] [INFO] Successfully connected to the MQTT broker. +30 9121 [MQTT Agent Task] [INFO] Session present: 0 +31 9128 [MQTT Agent Task] [INFO] Starting a clean MQTT Session. +32 9137 [OTA Task ] [INFO] Received: 0 Queued: 0 Processed: 0 Dropped: 0 +33 9149 [OTA Agent Task] [INFO] Current State=[RequestingJob], Event=[Start], New state=[RequestingJob] +34 9633 [OTA Agent Task] [INFO] Subscribed to topic $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/notify-next. +35 9653 [OTA Agent Task] [INFO] Subscribed to MQTT topic: $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/notify-next +36 10383 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/$next/get. +37 10922 [MQTT Agent Task] [INFO] Ack packet deserialized with result: MQTTSuccess. +38 10934 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +39 10946 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +40 10961 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +41 10973 [MQTT Agent Task] [INFO] Received job message callback, size 1020. +42 10983 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/$next/get to broker. +43 11006 [OTA Agent Task] [WARN] OTA Timer handle NULL for Timerid=0, can't stop. +44 11017 [OTA Agent Task] [INFO] Current State=[WaitingForJob], Event=[RequestJobDocument], New state=[WaitingForJob] +45 11035 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.jobId: AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a] +46 11060 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.jobDocument.afr_ota.streamname: AFR_OTA-eac26517-1848-4705-8242-97bda52cab40] +47 11082 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[execution.jobDocument.afr_ota.protocols: ["MQTT"]] +48 11100 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[filepath: non_secure image] +49 11114 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[filesize: 1141641] +50 11127 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[fileid: 0] +51 11138 [OTA Agent Task] [INFO] Extracted parameter: [key: value]=[certfile: 0] +52 11151 [OTA Agent Task] [INFO] Extracted parameter [ sig-sha256-rsa: LUOGFk3dx7VoXvS7tJl5MJObgHtbCJgZ... ] +53 11167 [OTA Agent Task] [INFO] Job document was accepted. Attempting to begin the update. +54 11180 [OTA Agent Task] [INFO] Job parsing success: OtaJobParseErr_t=OtaJobParseErrNone, Job name=AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a +55 11203 [OTA Agent Task] [INFO] Signal task inference stop +56 11253 [OTA Agent Task] [INFO] Setting OTA data interface. +57 11261 [OTA Agent Task] [INFO] Current State=[CreatingFile], Event=[ReceivedJobDocument], New state=[CreatingFile] +58 12464 [OTA Agent Task] [INFO] Subscribed to topic $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/data/cbor. +59 12490 [OTA Agent Task] [INFO] Current State=[RequestingFileBlock], Event=[CreateFile], New state=[RequestingFileBlock] +60 13214 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor. +61 13399 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor to broker. +62 13428 [OTA Agent Task] [INFO] Published to MQTT topic to request the next block: topic=$aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor +63 13459 [OTA Agent Task] [INFO] Current State=[WaitingForFileBlock], Event=[RequestFileBlock], New state=[WaitingForFileBlock] +64 14010 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +65 14024 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +66 14036 [MQTT Agent Task] [INFO] Received data message callback, size 4120. +67 14047 [OTA Agent Task] [INFO] Received valid file block: Block index=0, Size=4096 +68 14059 [OTA Agent Task] [INFO] Number of blocks remaining: 278 + +... + +3487 567949 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor. +3488 568137 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor to broker. +3489 568166 [OTA Agent Task] [INFO] Published to MQTT topic to request the next block: topic=$aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor +3490 568198 [OTA Agent Task] [INFO] Current State=[WaitingForFileBlock], Event=[RequestFileBlock], New state=[WaitingForFileBlock] +3491 569141 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +3492 569156 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +3493 569168 [MQTT Agent Task] [INFO] Received data message callback, size 4122. +3494 569179 [OTA Agent Task] [INFO] Received valid file block: Block index=277, Size=4096 +3495 569192 [OTA Agent Task] [INFO] Number of blocks remaining: 1 +3496 569201 [OTA Agent Task] [INFO] OTA active state `6` from OTA Agent. +3497 569212 [OTA Agent Task] [INFO] Signal task inference stop +3498 569221 [OTA Agent Task] [INFO] Current State=[WaitingForFileBlock], Event=[ReceivedFileBlock], New state=[WaitingForFileBlock] +3499 569872 [OTA Task ] [INFO] Received: 278 Queued: 278 Processed: 278 Dropped: 0 +3500 569983 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor. +3501 570169 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor to broker. +3502 570198 [OTA Agent Task] [INFO] Published to MQTT topic to request the next block: topic=$aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/streams/AFR_OTA-eac26517-1848-4705-8242-97bda52cab40/get/cbor +3503 570230 [OTA Agent Task] [INFO] Current State=[WaitingForFileBlock], Event=[RequestFileBlock], New state=[WaitingForFileBlock] +3504 571172 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +3505 571187 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +3506 571199 [MQTT Agent Task] [INFO] Received data message callback, size 2979. +3507 571210 [OTA Agent Task] [INFO] Received valid file block: Block index=278, Size=2953 +3508 571223 [OTA Agent Task] [INFO] Received final block of the update. +3509 571894 [OTA Agent Task] [INFO] Received entire update and validated the signature. +3510 572014 [MQTT Agent Task] [INFO] Publishing message to $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/update. +3511 573260 [MQTT Agent Task] [INFO] Ack packet deserialized with result: MQTTSuccess. +3512 573272 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +3513 573284 [MQTT Agent Task] [INFO] De-serialized incoming PUBLISH packet: DeserializerResult=MQTTSuccess. +3514 573299 [MQTT Agent Task] [INFO] State record updated. New state=MQTTPublishDone. +3515 573312 [OTA Agent Task] [INFO] Sent PUBLISH packet to broker $aws/things/iotmsw-ci-test-thing-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/jobs/AFR_OTA-ota-test-update-id-9ba0432f-e481-4fde-b3e5-a45bcf076e7a/update to broker. +3516 573343 [OTA Agent Task] [INFO] Received OtaJobEventActivate callback from OTA Agent. +``` + +## Running AWS IoT Core Device Advisor tests +Follow the instructions described in [Running AWS IoT Core Device Advisor tests](./running_aws_iot_core_device_advisor_tests.md). diff --git a/tools/ci/license/license.schema b/tools/ci/license/license.schema index fb510f42..f9dde918 100644 --- a/tools/ci/license/license.schema +++ b/tools/ci/license/license.schema @@ -73,7 +73,7 @@ "properties": { "path": { "type": "string", - "pattern": "\\.(html|json|md|log|in|map|rst|patch|wav|release_ignore|ini|conf|options|toml|change|pylintrc)$" + "pattern": "\\.(html|json|md|log|in|map|rst|patch|wav|frm|release_ignore|ini|conf|options|toml|change|pylintrc)$" }, "type": { "type": "string", diff --git a/tools/scripts/build.sh b/tools/scripts/build.sh index 582aea77..41a37a2f 100755 --- a/tools/scripts/build.sh +++ b/tools/scripts/build.sh @@ -93,7 +93,7 @@ Options: --certificate_path Path to the AWS device certificate --private_key_path Path to the AWS device private key Examples: - blinky, freertos-iot-libraries-tests, keyword-detection, speech-recognition + blinky, freertos-iot-libraries-tests, keyword-detection, object-detection, speech-recognition EOF } @@ -188,8 +188,12 @@ case "$1" in EXAMPLE="$1" PATH_TO_SOURCE="$ROOT/applications/speech_recognition" ;; + object-detection) + EXAMPLE="$1" + PATH_TO_SOURCE="$ROOT/applications/object_detection" + ;; *) - echo "Missing example " + echo "Missing example " show_usage exit 2 ;; @@ -282,4 +286,9 @@ if [ "$EXAMPLE" == "speech-recognition" ] && [ "$ML_INFERENCE_ENGINE" == "SOFTWA exit 1 fi +if [ "$EXAMPLE" == "object-detection" ] && [ "$TARGET" != "corstone315" ]; then + echo "Error: Invalid combination of example and target. object-detection only supports corstone315" >&2 + exit 2 +fi + build_with_cmake diff --git a/tools/scripts/run.sh b/tools/scripts/run.sh index 60ced60d..67bc2481 100755 --- a/tools/scripts/run.sh +++ b/tools/scripts/run.sh @@ -28,14 +28,15 @@ Options: -t,--target Target to run -s, --audio Audio source (ROM, VSI) -n, --npu-id NPU ID to use (U55, U65) + --frames Path to camera frames for the ISP to stream Examples: - blinky, keyword-detection, speech-recognition + blinky, keyword-detection, speech-recognition, object-detection EOF } SHORT=t:,n:,s:,h,p: -LONG=target:,npu-id:,audio:,help,path: +LONG=target:,npu-id:,audio:,help,path:,frames: OPTS=$(getopt -n run --options $SHORT --longoptions $LONG -- "$@") eval set -- "$OPTS" @@ -63,6 +64,10 @@ do AUDIO_SOURCE=$2 shift 2 ;; + --frames ) + FRAMES=$2 + shift 2 + ;; --) shift; break @@ -121,8 +126,12 @@ case "$1" in EXAMPLE="$1" MERGED_IMAGE_PATH="$BUILD_PATH/speech-recognition_merged.elf" ;; + object-detection) + EXAMPLE="$1" + MERGED_IMAGE_PATH="$BUILD_PATH/object-detection_merged.elf" + ;; *) - echo "Usage: $0 " >&2 + echo "Usage: $0 " >&2 exit 1 ;; esac @@ -167,9 +176,17 @@ case "$TARGET" in -C mps4_board.uart0.out_file="-" \ -C mps4_board.uart0.unbuffered_output=1 \ -C mps4_board.subsystem.ethosu.extra_args="--fast" \ + -C mps4_board.isp_c55_capture_ds.do_capture=0 \ + -C mps4_board.isp_c55_capture_fr.do_capture=0 \ + -C mps4_board.isp_c55_camera.image_file=${FRAMES} \ --stat" ;; esac +if [ "$EXAMPLE" == "object-detection" ] && [ "$TARGET" != "corstone315" ]; then + echo "Error: Invalid combination of example and target. object-detection only supports corstone315" >&2 + exit 2 +fi + # Start the FVP $FVP_BIN -a $MERGED_IMAGE_PATH $OPTIONS $AVH_AUDIO_OPTIONS diff --git a/tools/tests/conftest.py b/tools/tests/conftest.py index e2e0bcc9..eaeaaa14 100644 --- a/tools/tests/conftest.py +++ b/tools/tests/conftest.py @@ -1,4 +1,4 @@ -# Copyright (c) 2023 Arm Limited. All rights reserved. +# Copyright (c) 2023-2024, Arm Limited. All rights reserved. # SPDX-License-Identifier: MIT import pytest @@ -22,6 +22,7 @@ def pytest_addoption(parser): default="", help="Path to the input audio file to be used by applications.", ) + parser.addoption("--frm-file-path", action="store", default="") parser.addoption("--build-artefacts-path", action="store", default="") parser.addoption("--credentials-path", action="store", default="credentials") parser.addoption("--fvp", action="store", default="FVP_Corstone_SSE-310") @@ -63,6 +64,11 @@ def audio_file_path(pytestconfig): yield pytestconfig.getoption("--audio-file-path") +@pytest.fixture +def frm_file_path(pytestconfig): + yield pytestconfig.getoption("--frm-file-path") + + @pytest.fixture def fvp_options(pytestconfig): raw_options = pytestconfig.getoption("--fvp-options") @@ -116,39 +122,93 @@ def pythonhome_path(pytestconfig): @pytest.fixture(scope="function") def fvp_process( - fvp_path, merged_elf_name, fvp_options, audio_file_path, pythonhome_path + fvp_path, + merged_elf_name, + fvp_options, + audio_file_path, + pythonhome_path, + frm_file_path, ): # Fixture of the FVP, when it returns, the FVP is started and # traces are accessible through the .stdout of the object returned. # When the test is terminated, the FVP subprocess is closed. # Note: It can take few seconds to terminate the FVP - cmdline = [ - fvp_path, - "-a", - f"{merged_elf_name}", - "-C", - "core_clk.mul=200000000", - "-C", - "mps3_board.visualisation.disable-visualisation=1", - "-C", - "mps3_board.telnetterminal0.start_telnet=0", - "-C", - "mps3_board.uart0.out_file=-", - "-C", - "mps3_board.uart0.unbuffered_output=1", - "-C", - "mps3_board.uart0.shutdown_on_eot=1", - "-C", - "cpu0.semihosting-enable=1", - "-C", - "mps3_board.smsc_91c111.enabled=1", - "-C", - "mps3_board.hostbridge.userNetworking=1", - "-C", - "mps3_board.DISABLE_GATING=1", - "-C", - "ethosu.extra_args=--fast", - ] + + if "SSE-315" in fvp_path: + cmdline = [ + fvp_path, + "-a", + f"{merged_elf_name}", + "-C", + "core_clk.mul=200000000", + "-C", + "mps4_board.visualisation.disable-visualisation=1", + "-C", + "mps4_board.telnetterminal0.start_telnet=0", + "-C", + "mps4_board.uart0.out_file=-", + "-C", + "mps4_board.uart0.unbuffered_output=1", + "-C", + "mps4_board.uart0.shutdown_on_eot=1", + "-C", + "mps4_board.subsystem.cpu0.semihosting-enable=1", + "-C", + "mps4_board.smsc_91c111.enabled=1", + "-C", + "mps4_board.hostbridge.userNetworking=1", + "-C", + "mps4_board.subsystem.ethosu.extra_args=--fast", + ] + + vsi_options = [ + "-C", + f"mps4_board.v_path={Path(__file__).parent.parent / 'scripts'}", + ] + + frm_options = [ + "-C", + "mps4_board.isp_c55_capture_ds.do_capture=0", + "-C", + "mps4_board.isp_c55_capture_fr.do_capture=0", + "-C", + f"mps4_board.isp_c55_camera.image_file={frm_file_path}", + ] + else: + cmdline = [ + fvp_path, + "-a", + f"{merged_elf_name}", + "-C", + "core_clk.mul=200000000", + "-C", + "mps3_board.visualisation.disable-visualisation=1", + "-C", + "mps3_board.telnetterminal0.start_telnet=0", + "-C", + "mps3_board.uart0.out_file=-", + "-C", + "mps3_board.uart0.unbuffered_output=1", + "-C", + "mps3_board.uart0.shutdown_on_eot=1", + "-C", + "cpu0.semihosting-enable=1", + "-C", + "mps3_board.smsc_91c111.enabled=1", + "-C", + "mps3_board.hostbridge.userNetworking=1", + "-C", + "mps3_board.DISABLE_GATING=1", + "-C", + "ethosu.extra_args=--fast", + ] + + vsi_options = [ + "-C", + f"mps3_board.v_path={Path(__file__).parent.parent / 'scripts'}", + ] + + frm_options = [] cmdline.extend(fvp_options) @@ -159,11 +219,9 @@ def fvp_process( # If this option is used with ROM as audio source, it would break if PYTHONHOME # path is not set. if audio_file_path: - vsi_options = [ - "-C", - f"mps3_board.v_path={Path(__file__).parent.parent / 'scripts'}", - ] cmdline.extend(vsi_options) + if frm_file_path: + cmdline.extend(frm_options) if pythonhome_path != "": fvp_env["PYTHONHOME"] = pythonhome_path proc = subprocess.Popen(cmdline, stdout=subprocess.PIPE, env=fvp_env)