From 20c564bedb47070c838869dde00a8729b22fe5f3 Mon Sep 17 00:00:00 2001 From: Seth Rickard Date: Mon, 29 Nov 2021 23:28:00 -0600 Subject: [PATCH] Add ECC hardware acceleration for TI platform (#11682) * Add ECC hardware acceleration for TI platform Add MbedTLS ALT source files for the Texas Instruments CC13X2_26X2 platform to enable hardware acceleration of ECDSA and ECDHE. This was done to address timeout issues with PASE and CASE test cases in TE#6. The test cases would fail due to a gross timeout even though the sessions were established. This change requires a new version of the SimpleLink SDK to avoid a BLE ROM symbol collision. A new version of the SimpleLink SDK based on the 5.30 SDK is used here. Documentation has been updated in associated project READMEs. The button middleware driver API has been updated and associated calls in examples have been updated to match. Address an issue where the SHA-256 hardware acceleration did not copy the digest buffer. It was incorrectly assumed that the digest buffer would be empty when a SHA context was cloned. This addresses an issue in decoding ZCL messages. * Restyled by gn * Restyled by prettier-markdown * fix copy/paste error * update pump README * Add DMM Compatible BLE User config for Matter Replace TI SDK BLE user config to enable DMM hooks. Co-authored-by: Alexander D'abreu * update copyright and make pretty * update HEAP and cstack size from TE#7 * Restyled by gn Co-authored-by: Restyled.io Co-authored-by: Alexander D'abreu --- .../images/matter_ti_overview_simplified.png | Bin 0 -> 75738 bytes docs/guides/ti_platform_overview.md | 121 ++++ examples/lock-app/cc13x2x7_26x2x7/README.md | 47 +- examples/lock-app/cc13x2x7_26x2x7/chip.syscfg | 18 +- .../lock-app/cc13x2x7_26x2x7/main/AppTask.cpp | 22 +- .../main/include/CHIPProjectConfig.h | 16 - .../lock-app/cc13x2x7_26x2x7/main/main.cpp | 8 +- .../cc13x2x7_26x2x7/README.md | 26 +- .../cc13x2x7_26x2x7/chip.syscfg | 18 +- examples/pump-app/cc13x2x7_26x2x7/README.md | 33 +- examples/pump-app/cc13x2x7_26x2x7/chip.syscfg | 49 +- .../pump-app/cc13x2x7_26x2x7/main/AppTask.cpp | 29 +- .../pump-app/cc13x2x7_26x2x7/main/main.cpp | 8 +- .../cc13x2x7_26x2x7/README.md | 33 +- .../cc13x2x7_26x2x7/chip.syscfg | 48 +- .../cc13x2x7_26x2x7/main/AppTask.cpp | 29 +- .../cc13x2x7_26x2x7/main/main.cpp | 8 +- src/platform/cc13x2_26x2/BLEManagerImpl.cpp | 17 - .../cc13x2_26x2/CHIPDevicePlatformConfig.h | 2 +- src/platform/cc13x2_26x2/FreeRTOSConfig.h | 2 +- .../cc13x2_26x2/ThreadStackManagerImpl.cpp | 10 + src/platform/cc13x2_26x2/ble_user_config.c | 530 ++++++++++++++++++ .../crypto/cc13x2_26x2-mbedtls-config.h | 8 +- src/platform/cc13x2_26x2/crypto/ecdh_alt.c | 314 +++++++++++ src/platform/cc13x2_26x2/crypto/ecdsa_alt.c | 289 ++++++++++ src/platform/cc13x2_26x2/crypto/sha256_alt.c | 2 + third_party/ti_simplelink_sdk/BUILD.gn | 2 + .../ti_simplelink_sdk/ti_simplelink_sdk.gni | 16 +- 28 files changed, 1476 insertions(+), 229 deletions(-) create mode 100755 docs/guides/images/matter_ti_overview_simplified.png create mode 100644 docs/guides/ti_platform_overview.md create mode 100644 src/platform/cc13x2_26x2/ble_user_config.c create mode 100644 src/platform/cc13x2_26x2/crypto/ecdh_alt.c create mode 100755 src/platform/cc13x2_26x2/crypto/ecdsa_alt.c diff --git a/docs/guides/images/matter_ti_overview_simplified.png b/docs/guides/images/matter_ti_overview_simplified.png new file mode 100755 index 0000000000000000000000000000000000000000..2586fa41fdd97aecf59f78ac5060af273da89c03 GIT binary patch literal 75738 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 CG+h<| literal 0 HcmV?d00001 diff --git a/docs/guides/ti_platform_overview.md b/docs/guides/ti_platform_overview.md new file mode 100644 index 00000000000000..37a8e065232810 --- /dev/null +++ b/docs/guides/ti_platform_overview.md @@ -0,0 +1,121 @@ +# Texas Instruments platform overview + +The TI platform is a [Matter][matter_gh] platform based on the Texas Instruments +Incorporated SimpleLink SDK. + +The following diagram is a simplified representation of a Matter application +which built on the TI Platform. + +

+ matter_ti_overview_simplified +

+ +## Texas Instruments SimpleLink SDK + +The SimpleLink™ CC13xx and CC26xx Software Development Kit (SDK) delivers +components that enable engineers to develop applications on the Texas +Instruments SimpleLink CC13xx and CC26xx family of wireless microcontrollers +(MCUs). This software toolkit provides a cohesive and consistent software +experience for all SimpleLink CC13xx and CC26xx wireless MCU users by packaging +essential software components, such as a Bluetooth® Low Energy (BLE) protocol +stack supporting Bluetooth 5.2, Bluetooth Mesh, Thread 1.1.1 networking stack +based on OpenThread, Zigbee 3.0 compliant protocol suite, RF-Proprietary +examples, TI’s 15.4 Stack as well as the TI-RTOS kernel and TI Drivers in one +easy-to-use software package along with example applications and documentation. +In addition, the Dynamic Multi-Protocol Manager (DMM) software component enables +multiprotocol development on a single SimpleLink wireless MCU through +time-division multiplexing. + +The SimpleLink MCU portfolio offers a single development environment that +delivers flexible hardware, software, and tool options for customers developing +wired and wireless applications. With 100 percent code reuse across host MCUs, +Wi-Fi™, Bluetooth Low Energy, Sub-1GHz devices and more, choose the MCU or +connectivity standard that fits your design. A one-time investment with the +SimpleLink software development kit allows you to reuse often, opening the door +to create unlimited applications. For more information, visit +www.ti.com/simplelink. + +
+ +## BLE and Thread stacks + +In the TI example applications the Bluetooth Low Energy protocol is used to +provision the Thread protocol to enable Matter communication. Then Thread is +used for IP communication with other Matter devices. + +The TI applications leverage the Bluetooth Low Energy stack on the CC13X2 and +CC26X2 families. This BLE software is distributed in binary form within the TI +SimpleLink SDK. The BLE stack leverages code that is present in the device ROM +for certain common BLE operations. + +These applications leverage the OpenThread stack available within the Matter +repository for Thread communication. Platform support source is built from the +SimpleLink SDK. + +These connection protocols can be run concurrently by using the Texas +Instruments Dynamic Multi-protocol Manager. + +
+ +## LwIP stack + +The Lightweight IP stack interfaces with the OpenThread stack to offer standard +IP connectivity protocols that OpenThread does not natively support. This offers +a standard socket based interface to the Matter platform. + +
+ +## MbedTLS + +The MbedTLS library is used by OpenThread and Matter for a wide variety of +protocols. This ranges from basic AES and SHA to cryptographic protocols like +ECDSA and ECDH. + +The MbedTLS library is hardware accelerated using the TI SimpleLink SDK drivers. +This is achieved through the usage of `_ALT` defines in the MbedTLS +configuration file. + +
+ +## Matter integration + +Matter interacts with LwIP, OpenThread, and the TI BLE stack to achieve the +protocol and application functionality. A BLE profile is registered with the +TI-BLE stack to enable provisioning and configuration. Once the device is +provisioned Matter will configure the OpenThread interface to connect to an +existing Thread network or to start its own network. From there the Matter IP +messages are sent to the LwIP stack to be routed to the OpenThread stack for +transmission. + +Overall, applications generally only need to interface with the Cluster Library +from Matter. The transport of messages and configuration of the device is all +handled by the platform implementation files. + +
+ +# Matter example applications + +Sample Matter applications are provided for the TI platform. These can be used +as reference for your own application. + +- [lock-app](../../examples/lock-app/cc13x2x7_26x2x7/README.md) +- [pump-app](../../examples/pump-app/cc13x2x7_26x2x7/README.md) +- [pump-controller-app](../../examples/pump-controller-app/cc13x2x7_26x2x7/README.md) + +
+ +## Build system + +The TI platform uses GN to generate ninja build scripts. Build files have +already been written to build and link the TI specific code within the +SimpleLink SDK. + +
+ +## TI Support + +For technical support, please consider creating a post on TI's [E2E forum][e2e]. +Additionally, we welcome any feedback. + +[e2e]: https://e2e.ti.com/support/wireless-connectivity/zigbee-and-thread +[matter_gh]: https://github.com/project-chip/connectedhomeip diff --git a/examples/lock-app/cc13x2x7_26x2x7/README.md b/examples/lock-app/cc13x2x7_26x2x7/README.md index 857dc6bf552159..d2c14aa666cc18 100644 --- a/examples/lock-app/cc13x2x7_26x2x7/README.md +++ b/examples/lock-app/cc13x2x7_26x2x7/README.md @@ -1,11 +1,11 @@ -# CHIP CC1352 CC2652 Lock Example Application +# Matter CC1352 CC2652 Lock Example Application -An example application showing the use [CHIP][chip] on the Texas Instruments -CC13X2_26X2 family of Wireless MCUs. +An example application showing the use of [Matter][matter] on the Texas +Instruments CC13X2_26X2 family of Wireless MCUs. --- -- [CHIP CC1352 CC2652 Lock Example Application](#chip-cc1352-cc2652-lock-example-application) +- [Matter CC1352 CC2652 Lock Example Application](#matter-cc1352-cc2652-lock-example-application) - [Introduction](#introduction) - [Device UI](#device-ui) - [Building](#building) @@ -19,7 +19,7 @@ CC13X2_26X2 family of Wireless MCUs. - [Provisioning](#provisioning) - [Bluetooth LE Advertising](#bluetooth-le-advertising) - [Bluetooth LE Rendezvous](#bluetooth-le-rendezvous) - - [CHIP Remote Commands](#chip-remote-commands) + - [Matter Remote Commands](#matter-remote-commands) - [TI Support](#ti-support) --- @@ -29,16 +29,16 @@ CC13X2_26X2 family of Wireless MCUs. ![CC1352R1_LAUNCHXL](doc/images/cc1352r1_launchxl.jpg) The CC13X2_26X2 lock example application provides a working demonstration of a -connected door lock device. This uses the open-source CHIP implementation and +connected door lock device. This uses the open-source Matter implementation and the Texas Instruments SimpleLinkâ„¢ CC13x2 and CC26x2 software development kit. This example is enabled to build for CC2652R7 devices. This upcoming devices are currently not yet in full production. For more information on device -availability or early access to an engineering build of our CHIP-enabled SDK, -please reach out [here][ti_cc13x2_26x2_r7_chip_request]. +availability or early access to an engineering build of our Matter-enabled SDK, +please reach out [here][ti_cc13x2_26x2_r7_matter_request]. The lock example is intended to serve both as a means to explore the workings of -CHIP, as well as a template for creating real products based on the Texas +Matter, as well as a template for creating real products based on the Texas Instruments devices. ## Device UI @@ -66,7 +66,7 @@ section will need to be done when migrating to new versions of the SDK. This guide assumes that the environment is linux based, and recommends Ubuntu 20.04. - An engineering SDK from TI is required. Please request access for it - [here][ti_cc13x2_26x2_r7_chip_request]. + [here][ti_cc13x2_26x2_r7_matter_request]. - Follow the default installation instructions when executing the installer. @@ -152,15 +152,15 @@ Ninja to build the executable. - Run the build to produce a default executable. By default on Linux both the TI SimpleLink SDK and Sysconfig are located in a `ti` folder in the user's home directory, and you must provide the absolute path to them. For example - `/home/username/ti/simplelink_cc13x2_26x2_sdk_4_40_05_02_eng` and - `/home/username/ti/sysconfig_1.7.0`. On Windows the default directory is + `/home/username/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng` and + `/home/username/ti/sysconfig_1.10.0`. On Windows the default directory is `C:\ti`. Take note of this install path, as it will be used in the next step. ``` $ cd ~/connectedhomeip/examples/lock-app/cc13x2x7_26x2x7 - $ export TI_SIMPLELINK_SDK_ROOT=$HOME/ti/simplelink_cc13x2_26x2_sdk_4_40_05_02_eng - $ export TI_SYSCONFIG_ROOT=$HOME/ti/sysconfig_1.7.0 + $ export TI_SIMPLELINK_SDK_ROOT=$HOME/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng + $ export TI_SYSCONFIG_ROOT=$HOME/ti/sysconfig_1.10.0 $ gn gen out/debug --args="ti_simplelink_sdk_root=\"${TI_SIMPLELINK_SDK_ROOT}\" ti_sysconfig_root=\"${TI_SYSCONFIG_ROOT}\"" $ ninja -C out/debug @@ -252,8 +252,8 @@ Router][ot_border_router_setup]. ### Provisioning -The first step to bring the CHIP device onto the network is to provision it. Our -example accomplishes this with Bluetooth Low Energy (BLE) and the +The first step to bring the Matter device onto the network is to provision it. +Our example accomplishes this with Bluetooth Low Energy (BLE) and the [CHIPTool](../../../src/android/CHIPTool/README.md) mobile app. #### Bluetooth LE Advertising @@ -265,9 +265,9 @@ fully provisioned, BLE advertising will stop. #### Bluetooth LE Rendezvous -To commission and control this application within a CHIP-enabled Thread network, -consult the [CHIPTool README](../../../src/android/CHIPTool/README.md) for -information on the Android smartphone application. Reference the Device +To commission and control this application within a Matter-enabled Thread +network, consult the [CHIPTool README](../../../src/android/CHIPTool/README.md) +for information on the Android smartphone application. Reference the Device Configuration information printed in the Logging Output of this application. ## TI Support @@ -275,7 +275,7 @@ Configuration information printed in the Logging Output of this application. For technical support, please consider creating a post on TI's [E2E forum][e2e]. Additionally, we welcome any feedback. -[chip]: https://github.com/project-chip/connectedhomeip +[matter]: https://github.com/project-chip/connectedhomeip [ccs]: https://www.ti.com/tool/CCSTUDIO [ccs_after_launch]: https://software-dl.ti.com/ccs/esd/documents/users_guide/ccs_debug-main.html?configuration#after-launch @@ -287,14 +287,11 @@ Additionally, we welcome any feedback. https://software-dl.ti.com/ccs/esd/documents/users_guide/ccs_debug-main.html?configuration#manual-method [cc1352r1_launchxl]: https://www.ti.com/tool/LAUNCHXL-CC1352R1 [e2e]: https://e2e.ti.com/support/wireless-connectivity/zigbee-and-thread -[simplelink_sdk]: https://www.ti.com/tool/SIMPLELINK-CC13X2-26X2-SDK -[simplelink_sdk_4.30.00.54]: - https://www.ti.com/tool/download/SIMPLELINK-CC13X2-26X2-SDK/4.30.00.54 [sysconfig]: https://www.ti.com/tool/SYSCONFIG [sysconfig_recommended]: - https://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.7.0_1746-setup.run + https://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.10.0_2163-setup.run [ti_thread_dnd]: https://www.ti.com/wireless-connectivity/thread/design-development.html -[ti_cc13x2_26x2_r7_chip_request]: https://ti.com/chip_sdk +[ti_cc13x2_26x2_r7_matter_request]: https://ti.com/chip_sdk [ot_border_router_setup]: https://openthread.io/guides/border-router/build [uniflash]: https://www.ti.com/tool/download/UNIFLASH diff --git a/examples/lock-app/cc13x2x7_26x2x7/chip.syscfg b/examples/lock-app/cc13x2x7_26x2x7/chip.syscfg index c93e60f4846aaa..f195ca15ec32d7 100644 --- a/examples/lock-app/cc13x2x7_26x2x7/chip.syscfg +++ b/examples/lock-app/cc13x2x7_26x2x7/chip.syscfg @@ -17,6 +17,7 @@ * limitations under the License. */ + /* Modules */ var AESCCM = scripting.addModule("/ti/drivers/AESCCM"); var AESECB = scripting.addModule("/ti/drivers/AESECB"); @@ -38,7 +39,6 @@ var ECDH = scripting.addModule("/ti/drivers/ECDH"); /* Instances */ var AESCCM1 = AESCCM.addInstance(); -var AESCCM2 = AESCCM.addInstance(); var AESECB1 = AESECB.addInstance(); var AESECB2 = AESECB.addInstance(); var Button1 = Button.addInstance(); @@ -52,22 +52,18 @@ var TRNG1 = TRNG.addInstance(); var TRNG2 = TRNG.addInstance(); var TRNG3 = TRNG.addInstance(); var UART1 = UART.addInstance(); -var UART2 = UART.addInstance(); var AESCTRDRBG1 = AESCTRDRBG.addInstance(); var ECDH1 = ECDH.addInstance(); -var ECDH2 = ECDH.addInstance(); AESCTRDRBG1.$name = "CONFIG_AESCTRDRBG_0"; AESCTRDRBG1.aesctrObject.$name = "CONFIG_AESCTR_0"; AESCCM1.$name = "CONFIG_AESCCM0"; -AESCCM2.$name = "CONFIG_AESCCM_1"; AESECB1.$name = "CONFIG_AESECB0"; AESECB2.$name = "CONFIG_AESECB_1"; ECDH1.$name = "CONFIG_ECDH0"; -ECDH2.$name = "CONFIG_ECDH_1"; /* RTOS */ RTOS.name = "FreeRTOS"; @@ -78,7 +74,6 @@ Button1.$hardware = system.deviceData.board.components["BTN-1" Button1.gpioPin.$name = "CONFIG_GPIO_BTN1"; Button1.gpioPin.pull = "Pull Up"; Button1.gpioPin.interruptTrigger = "Falling Edge"; -Button1.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN1"; /* Left Button */ Button2.$name = "CONFIG_BTN_RIGHT"; @@ -86,7 +81,6 @@ Button2.$hardware = system.deviceData.board.components["BTN-2" Button2.gpioPin.$name = "CONFIG_GPIO_BTN2"; Button2.gpioPin.pull = "Pull Up"; Button2.gpioPin.interruptTrigger = "Falling Edge"; -Button2.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN2"; /* ======== CCFG ======== */ var CCFG = scripting.addModule("/ti/devices/CCFG"); @@ -104,11 +98,10 @@ if(deviceName.includes("RB")) } /* NVS */ NVS1.$name = "CONFIG_NVSINTERNAL"; -NVS1.internalFlash.regionBase = 0x78000; +NVS1.internalFlash.regionBase = 0xAA000; NVS1.internalFlash.regionSize = 0x4000; /* RF */ -RF.$name = "CONFIG_RF0"; /* if an antenna component exists, assign it to the rf instance */ if (system.deviceData.board && system.deviceData.board.components.RF) { RF.$hardware = system.deviceData.board.components.RF; @@ -126,7 +119,6 @@ LED1.$hardware = system.deviceData.board.components.LED_RED; LED1.gpioPin.$name = "CONFIG_GPIO_RLED"; LED1.gpioPin.mode = "Output"; LED1.gpioPin.callbackFunction = ""; -LED1.gpioPin.pinInstance.$name = "CONFIG_PIN_RLED"; /* Green LED */ LED2.$name = "CONFIG_LED_GREEN"; @@ -134,17 +126,11 @@ LED2.$hardware = system.deviceData.board.components.LED_GREEN; LED2.gpioPin.$name = "CONFIG_GPIO_GLED"; LED2.gpioPin.mode = "Output"; LED2.gpioPin.callbackFunction = ""; -LED2.gpioPin.pinInstance.$name = "CONFIG_PIN_GLED"; /* Debug UART */ UART1.$hardware = system.deviceData.board.components.XDS110UART; UART1.$name = "CONFIG_UART_DEBUG"; -/* Display UART */ -UART2.$name = "CONFIG_DISPLAY_UART"; -UART2.uart.txPin.$suggestSolution = "boosterpack.32"; -UART2.uart.rxPin.$suggestSolution = "boosterpack.18"; - /* TRNG */ TRNG1.$name = "CONFIG_TRNG_0"; TRNG2.$name = "CONFIG_TRNG_1"; diff --git a/examples/lock-app/cc13x2x7_26x2x7/main/AppTask.cpp b/examples/lock-app/cc13x2x7_26x2x7/main/AppTask.cpp index e0d522d588560f..8649f916b3ebb1 100644 --- a/examples/lock-app/cc13x2x7_26x2x7/main/AppTask.cpp +++ b/examples/lock-app/cc13x2x7_26x2x7/main/AppTask.cpp @@ -82,7 +82,7 @@ int AppTask::StartAppTask() int AppTask::Init() { LED_Params ledParams; - Button_Params buttionParams; + Button_Params buttonParams; cc13x2_26x2LogInit(); @@ -152,15 +152,17 @@ int AppTask::Init() PLAT_LOG("Initialize buttons"); Button_init(); - Button_Params_init(&buttionParams); - buttionParams.buttonEventMask = Button_EV_CLICKED | Button_EV_LONGCLICKED; - buttionParams.longPressDuration = 1000U; // ms - sAppLeftHandle = Button_open(CONFIG_BTN_LEFT, ButtonLeftEventHandler, &buttionParams); - - Button_Params_init(&buttionParams); - buttionParams.buttonEventMask = Button_EV_CLICKED | Button_EV_LONGCLICKED; - buttionParams.longPressDuration = 1000U; // ms - sAppRightHandle = Button_open(CONFIG_BTN_RIGHT, ButtonRightEventHandler, &buttionParams); + Button_Params_init(&buttonParams); + buttonParams.buttonEventMask = Button_EV_CLICKED | Button_EV_LONGCLICKED; + buttonParams.longPressDuration = 1000U; // ms + sAppLeftHandle = Button_open(CONFIG_BTN_LEFT, &buttonParams); + Button_setCallback(sAppLeftHandle, ButtonLeftEventHandler); + + Button_Params_init(&buttonParams); + buttonParams.buttonEventMask = Button_EV_CLICKED | Button_EV_LONGCLICKED; + buttonParams.longPressDuration = 1000U; // ms + sAppRightHandle = Button_open(CONFIG_BTN_RIGHT, &buttonParams); + Button_setCallback(sAppRightHandle, ButtonRightEventHandler); // Initialize BoltLock module PLAT_LOG("Initialize BoltLock"); diff --git a/examples/lock-app/cc13x2x7_26x2x7/main/include/CHIPProjectConfig.h b/examples/lock-app/cc13x2x7_26x2x7/main/include/CHIPProjectConfig.h index 96d0e31bcee6b0..5b5c95c5721145 100644 --- a/examples/lock-app/cc13x2x7_26x2x7/main/include/CHIPProjectConfig.h +++ b/examples/lock-app/cc13x2x7_26x2x7/main/include/CHIPProjectConfig.h @@ -30,20 +30,8 @@ #if BUILD_RELEASE // release build -// Security and Authentication enabled for release build. -#define CHIP_CONFIG_SECURITY_TEST_MODE 0 -#define CHIP_CONFIG_REQUIRE_AUTH 1 - #else // development build -// Security and Authentication disabled for development build. -// For convenience, enable CHIP Security Test Mode and disable the requirement for -// authentication in various protocols. -// WARNING: These options make it possible to circumvent basic CHIP security functionality, -// including message encryption. Because of this they MUST NEVER BE ENABLED IN PRODUCTION BUILDS. -#define CHIP_CONFIG_SECURITY_TEST_MODE 0 -#define CHIP_CONFIG_REQUIRE_AUTH 0 - // Use a default pairing code if one hasn't been provisioned in flash. #define CHIP_DEVICE_CONFIG_USE_TEST_SETUP_PIN_CODE 20202021 #define CHIP_DEVICE_CONFIG_USE_TEST_SETUP_DISCRIMINATOR 0xF00 @@ -138,10 +126,6 @@ */ #define CHIP_DEVICE_CONFIG_ENABLE_THREAD_SRP_CLIENT 1 -#define CHIP_DEVICE_CONFIG_ENABLE_SED 1 -#define CHIP_DEVICE_CONFIG_SED_SLOW_POLLING_INTERVAL 5000 -#define CHIP_DEVICE_CONFIG_SED_FAST_POLLING_INTERVAL 5000 - /** * CHIP_CONFIG_EVENT_LOGGING_DEFAULT_IMPORTANCE * diff --git a/examples/lock-app/cc13x2x7_26x2x7/main/main.cpp b/examples/lock-app/cc13x2x7_26x2x7/main/main.cpp index 4db8ab020823a4..cc1c08bd82cf78 100644 --- a/examples/lock-app/cc13x2x7_26x2x7/main/main.cpp +++ b/examples/lock-app/cc13x2x7_26x2x7/main/main.cpp @@ -33,11 +33,13 @@ #include #include +#include +#include #include #include #include -#define TOTAL_ICALL_HEAP_SIZE (0xf000) +#define TOTAL_ICALL_HEAP_SIZE (0xc800) using namespace ::chip; using namespace ::chip::Inet; @@ -71,6 +73,10 @@ int main(void) UART_init(); + ECDH_init(); + + ECDSA_init(); + ECJPAKE_init(); AESECB_init(); diff --git a/examples/persistent-storage/cc13x2x7_26x2x7/README.md b/examples/persistent-storage/cc13x2x7_26x2x7/README.md index dba97e2896a48c..c07fd6b3864f54 100644 --- a/examples/persistent-storage/cc13x2x7_26x2x7/README.md +++ b/examples/persistent-storage/cc13x2x7_26x2x7/README.md @@ -1,11 +1,11 @@ -# CHIP CC1352 CC2652 Persistent Storage Example Application +# Matter CC1352 CC2652 Persistent Storage Example Application An example test application showing the persistent storage system for -[CHIP][chip] on the Texas Instruments CC13X2_26X2 family of Wireless MCUs. +[Matter][matter] on the Texas Instruments CC13X2_26X2 family of Wireless MCUs. --- -- [CHIP CC1352 CC2652 Persistent Storage Application](#chip-cc1352-cc2652-persistent-storage-example-application) +- [Matter CC1352 CC2652 Persistent Storage Application](#matter-cc1352-cc2652-persistent-storage-example-application) - [Introduction](#introduction) - [Device UI](#device-ui) - [Building](#building) @@ -26,8 +26,8 @@ offers information on proper usage of the KVS system. This example is enabled to build for CC2652R7 devices. This upcoming devices are currently not yet in full production. For more information on device -availability or early access to an engineering build of our CHIP-enabled SDK, -please reach out [here][ti_cc13x2_26x2_r7_chip_request]. +availability or early access to an engineering build of our Matter-enabled SDK, +please reach out [here][ti_cc13x2_26x2_r7_matter_request]. ## Device UI @@ -55,7 +55,7 @@ section will need to be done when migrating to new versions of the SDK. This guide assumes that the environment is linux based, and recommends Ubuntu 20.04. - An engineering SDK from TI is required. Please request access for it - [here][ti_cc13x2_26x2_r7_chip_request]. + [here][ti_cc13x2_26x2_r7_matter_request]. - Follow the default installation instructions when executing the installer. @@ -141,15 +141,15 @@ Ninja to build the executable. - Run the build to produce a default executable. By default on Linux both the TI SimpleLink SDK and Sysconfig are located in a `ti` folder in the user's home directory, and you must provide the absolute path to them. For example - `/home/username/ti/simplelink_cc13x2_26x2_sdk_4_40_05_02_eng` and - `/home/username/ti/sysconfig_1.7.0`. On Windows the default directory is + `/home/username/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng` and + `/home/username/ti/sysconfig_1.10.0`. On Windows the default directory is `C:\ti`. Take note of this install path, as it will be used in the next step. ``` $ cd ~/connectedhomeip/examples/lock-app/cc13x2x7_26x2x7 - $ export TI_SIMPLELINK_SDK_ROOT=$HOME/ti/simplelink_cc13x2_26x2_sdk_4_40_05_02_eng - $ export TI_SYSCONFIG_ROOT=$HOME/ti/sysconfig_1.7.0 + $ export TI_SIMPLELINK_SDK_ROOT=$HOME/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng + $ export TI_SYSCONFIG_ROOT=$HOME/ti/sysconfig_1.10.0 $ gn gen out/debug --args="ti_simplelink_sdk_root=\"${TI_SIMPLELINK_SDK_ROOT}\" ti_sysconfig_root=\"${TI_SYSCONFIG_ROOT}\"" $ ninja -C out/debug @@ -237,7 +237,7 @@ terminal emulator to that port to see the output with the following options: For technical support, please consider creating a post on TI's [E2E forum][e2e]. Additionally, we welcome any feedback. -[chip]: https://github.com/project-chip/connectedhomeip +[matter]: https://github.com/project-chip/connectedhomeip [ccs]: https://www.ti.com/tool/CCSTUDIO [ccs_after_launch]: https://software-dl.ti.com/ccs/esd/documents/users_guide/ccs_debug-main.html?configuration#after-launch @@ -250,6 +250,6 @@ Additionally, we welcome any feedback. [e2e]: https://e2e.ti.com/support/wireless-connectivity/zigbee-and-thread [sysconfig]: https://www.ti.com/tool/SYSCONFIG [sysconfig_recommended]: - https://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.7.0_1746-setup.run -[ti_cc13x2_26x2_r7_chip_request]: https://ti.com/chip_sdk + https://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.10.0_2163-setup.run +[ti_cc13x2_26x2_r7_matter_request]: https://ti.com/chip_sdk [uniflash]: https://www.ti.com/tool/download/UNIFLASH diff --git a/examples/persistent-storage/cc13x2x7_26x2x7/chip.syscfg b/examples/persistent-storage/cc13x2x7_26x2x7/chip.syscfg index c93e60f4846aaa..f195ca15ec32d7 100644 --- a/examples/persistent-storage/cc13x2x7_26x2x7/chip.syscfg +++ b/examples/persistent-storage/cc13x2x7_26x2x7/chip.syscfg @@ -17,6 +17,7 @@ * limitations under the License. */ + /* Modules */ var AESCCM = scripting.addModule("/ti/drivers/AESCCM"); var AESECB = scripting.addModule("/ti/drivers/AESECB"); @@ -38,7 +39,6 @@ var ECDH = scripting.addModule("/ti/drivers/ECDH"); /* Instances */ var AESCCM1 = AESCCM.addInstance(); -var AESCCM2 = AESCCM.addInstance(); var AESECB1 = AESECB.addInstance(); var AESECB2 = AESECB.addInstance(); var Button1 = Button.addInstance(); @@ -52,22 +52,18 @@ var TRNG1 = TRNG.addInstance(); var TRNG2 = TRNG.addInstance(); var TRNG3 = TRNG.addInstance(); var UART1 = UART.addInstance(); -var UART2 = UART.addInstance(); var AESCTRDRBG1 = AESCTRDRBG.addInstance(); var ECDH1 = ECDH.addInstance(); -var ECDH2 = ECDH.addInstance(); AESCTRDRBG1.$name = "CONFIG_AESCTRDRBG_0"; AESCTRDRBG1.aesctrObject.$name = "CONFIG_AESCTR_0"; AESCCM1.$name = "CONFIG_AESCCM0"; -AESCCM2.$name = "CONFIG_AESCCM_1"; AESECB1.$name = "CONFIG_AESECB0"; AESECB2.$name = "CONFIG_AESECB_1"; ECDH1.$name = "CONFIG_ECDH0"; -ECDH2.$name = "CONFIG_ECDH_1"; /* RTOS */ RTOS.name = "FreeRTOS"; @@ -78,7 +74,6 @@ Button1.$hardware = system.deviceData.board.components["BTN-1" Button1.gpioPin.$name = "CONFIG_GPIO_BTN1"; Button1.gpioPin.pull = "Pull Up"; Button1.gpioPin.interruptTrigger = "Falling Edge"; -Button1.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN1"; /* Left Button */ Button2.$name = "CONFIG_BTN_RIGHT"; @@ -86,7 +81,6 @@ Button2.$hardware = system.deviceData.board.components["BTN-2" Button2.gpioPin.$name = "CONFIG_GPIO_BTN2"; Button2.gpioPin.pull = "Pull Up"; Button2.gpioPin.interruptTrigger = "Falling Edge"; -Button2.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN2"; /* ======== CCFG ======== */ var CCFG = scripting.addModule("/ti/devices/CCFG"); @@ -104,11 +98,10 @@ if(deviceName.includes("RB")) } /* NVS */ NVS1.$name = "CONFIG_NVSINTERNAL"; -NVS1.internalFlash.regionBase = 0x78000; +NVS1.internalFlash.regionBase = 0xAA000; NVS1.internalFlash.regionSize = 0x4000; /* RF */ -RF.$name = "CONFIG_RF0"; /* if an antenna component exists, assign it to the rf instance */ if (system.deviceData.board && system.deviceData.board.components.RF) { RF.$hardware = system.deviceData.board.components.RF; @@ -126,7 +119,6 @@ LED1.$hardware = system.deviceData.board.components.LED_RED; LED1.gpioPin.$name = "CONFIG_GPIO_RLED"; LED1.gpioPin.mode = "Output"; LED1.gpioPin.callbackFunction = ""; -LED1.gpioPin.pinInstance.$name = "CONFIG_PIN_RLED"; /* Green LED */ LED2.$name = "CONFIG_LED_GREEN"; @@ -134,17 +126,11 @@ LED2.$hardware = system.deviceData.board.components.LED_GREEN; LED2.gpioPin.$name = "CONFIG_GPIO_GLED"; LED2.gpioPin.mode = "Output"; LED2.gpioPin.callbackFunction = ""; -LED2.gpioPin.pinInstance.$name = "CONFIG_PIN_GLED"; /* Debug UART */ UART1.$hardware = system.deviceData.board.components.XDS110UART; UART1.$name = "CONFIG_UART_DEBUG"; -/* Display UART */ -UART2.$name = "CONFIG_DISPLAY_UART"; -UART2.uart.txPin.$suggestSolution = "boosterpack.32"; -UART2.uart.rxPin.$suggestSolution = "boosterpack.18"; - /* TRNG */ TRNG1.$name = "CONFIG_TRNG_0"; TRNG2.$name = "CONFIG_TRNG_1"; diff --git a/examples/pump-app/cc13x2x7_26x2x7/README.md b/examples/pump-app/cc13x2x7_26x2x7/README.md index b8e270f63d9e4d..62a688e3aa53b5 100644 --- a/examples/pump-app/cc13x2x7_26x2x7/README.md +++ b/examples/pump-app/cc13x2x7_26x2x7/README.md @@ -48,13 +48,13 @@ pump and to control the state. The user LEDs on the LaunchPad are set on when the pump is started, and are set off when stopped. The LEDs will flash when in the transition state between started and stopped. -Short presses (less than 1000ms) of the left user button (`BTN-1`) are used for +Short presses (less than 1000ms) of the right user button (`BTN-2`) are used for toggling the pump state. -Short presses (less than 1000ms) of the right user button (`BTN-2`) are used for +Short presses (less than 1000ms) of the left user button (`BTN-1`) are used for toggling Matter BLE advertisements. -Long presses (greater than 5000ms) of the right user button (`BTN-2`) will +Long presses (greater than 5000ms) of the left user button (`BTN-1`) will initiate a factory reset of the device clearing all stored provisioning information to allow for a new network setup. @@ -75,8 +75,8 @@ section will need to be done when migrating to new versions of the SDK. packaged with the TI SDK. Check the following section for a list of changes needed. -- Download and install [SysConfig][sysconfig] - ([sysconfig-1.5.0_1397][sysconfig-1.5.0_1397]) +- Download and install [SysConfig][sysconfig] ([recommended + version][sysconfig_recommended]) - This may have already been installed with your SimpleLink SDK install. @@ -100,7 +100,7 @@ section will need to be done when migrating to new versions of the SDK. ``` $ cd ~/connectedhomeip - $ source ./script/bootstrap.sh + $ source ./scripts/bootstrap.sh ``` @@ -113,21 +113,21 @@ Ninja to build the executable. ``` $ cd ~/connectedhomeip - $ source ./script/activate.sh + $ source ./scripts/activate.sh ``` - Run the build to produce a default executable. By default on Linux both the TI SimpleLink SDK and Sysconfig are located in a `ti` folder in the user's home directory, and you must provide the absolute path to them. For example - `/home/username/ti/simplelink_cc13x2_26x2_sdk_4_40_05_02_eng` and - `/home/username/ti/sysconfig_1.6.0`. On Windows the default directory is - `C:\ti` + `/home/username/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng` and + `/home/username/ti/sysconfig_1.10.0`. On Windows the default directory is + `C:\ti`. ``` - $ cd ~/connectedhomeip/examples/pump-app/cc13x2_26x2 - $ export TI_SIMPLELINK_SDK_ROOT= - $ export TI_SYSCONFIG_ROOT= + $ cd ~/connectedhomeip/examples/pump-app/cc13x2x7_26x2x7 + $ export TI_SIMPLELINK_SDK_ROOT=$HOME/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng + $ export TI_SYSCONFIG_ROOT=$HOME/ti/sysconfig_1.10.0 $ gn gen out/debug --args="ti_simplelink_sdk_root=\"${TI_SIMPLELINK_SDK_ROOT}\" ti_sysconfig_root=\"${TI_SYSCONFIG_ROOT}\"" $ ninja -C out/debug @@ -220,12 +220,9 @@ Additionally, we welcome any feedback. [matter]: https://github.com/project-chip/connectedhomeip [cc1352r1_launchxl]: https://www.ti.com/tool/LAUNCHXL-CC1352R1 [e2e]: https://e2e.ti.com/support/wireless-connectivity/zigbee-and-thread -[simplelink_sdk]: https://www.ti.com/tool/SIMPLELINK-CC13X2-26X2-SDK -[simplelink_sdk_4.30.00.54]: - https://www.ti.com/tool/download/SIMPLELINK-CC13X2-26X2-SDK/4.30.00.54 [sysconfig]: https://www.ti.com/tool/SYSCONFIG -[sysconfig-1.5.0_1397]: - http://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.5.0_1397-setup.run +[sysconfig_recommended]: + https://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.10.0_2163-setup.run [ti_thread_dnd]: https://www.ti.com/wireless-connectivity/thread/design-development.html [ti_cc13x2_26x2_r7_matter_request]: https://ti.com/chip_sdk diff --git a/examples/pump-app/cc13x2x7_26x2x7/chip.syscfg b/examples/pump-app/cc13x2x7_26x2x7/chip.syscfg index 38a53e57d500fd..3f47781b9f518a 100644 --- a/examples/pump-app/cc13x2x7_26x2x7/chip.syscfg +++ b/examples/pump-app/cc13x2x7_26x2x7/chip.syscfg @@ -145,7 +145,7 @@ ECJPAKE1.$name = "CONFIG_ECJPAKE_0"; NVS1.$name = "CONFIG_NVSINTERNAL"; NVS1.internalFlash.$name = "ti_drivers_nvs_NVSCC26XX0"; -NVS1.internalFlash.regionBase = 0x78000; +NVS1.internalFlash.regionBase = 0xAA000; NVS1.internalFlash.regionSize = 0x4000; RTOS.name = "FreeRTOS"; @@ -169,25 +169,34 @@ UART2.$name = "CONFIG_DISPLAY_UART"; UART2.txPinInstance.$name = "CONFIG_PIN_6"; UART2.rxPinInstance.$name = "CONFIG_PIN_7"; -Button1.$name = "CONFIG_BTN_LEFT"; -Button1.gpioPin.$name = "CONFIG_GPIO_BTN1"; -Button1.gpioPin.gpioPin.$assign = "boosterpack.32"; -Button1.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN1"; - -Button2.$name = "CONFIG_BTN_RIGHT"; -Button2.gpioPin.$name = "CONFIG_GPIO_BTN2"; -Button2.gpioPin.gpioPin.$assign = "boosterpack.13"; -Button2.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN2"; - -LED1.$name = "CONFIG_LED_RED"; -LED1.gpioPin.$name = "CONFIG_GPIO_RLED"; -LED1.gpioPin.gpioPin.$assign = "boosterpack.10"; -LED1.gpioPin.pinInstance.$name = "CONFIG_PIN_RLED"; - -LED2.$name = "CONFIG_LED_GREEN"; -LED2.gpioPin.$name = "CONFIG_GPIO_GLED"; -LED2.gpioPin.gpioPin.$assign = "boosterpack.39"; -LED2.gpioPin.pinInstance.$name = "CONFIG_PIN_GLED"; + +/* Left Button */ +Button1.$name = "CONFIG_BTN_LEFT"; +Button1.$hardware = system.deviceData.board.components["BTN-1"]; +Button1.gpioPin.$name = "CONFIG_GPIO_BTN1"; +Button1.gpioPin.pull = "Pull Up"; +Button1.gpioPin.interruptTrigger = "Falling Edge"; + +/* Left Button */ +Button2.$name = "CONFIG_BTN_RIGHT"; +Button2.$hardware = system.deviceData.board.components["BTN-2"]; +Button2.gpioPin.$name = "CONFIG_GPIO_BTN2"; +Button2.gpioPin.pull = "Pull Up"; +Button2.gpioPin.interruptTrigger = "Falling Edge"; + +/* Red LED */ +LED1.$name = "CONFIG_LED_RED"; +LED1.$hardware = system.deviceData.board.components.LED_RED; +LED1.gpioPin.$name = "CONFIG_GPIO_RLED"; +LED1.gpioPin.mode = "Output"; +LED1.gpioPin.callbackFunction = ""; + +/* Green LED */ +LED2.$name = "CONFIG_LED_GREEN"; +LED2.$hardware = system.deviceData.board.components.LED_GREEN; +LED2.gpioPin.$name = "CONFIG_GPIO_GLED"; +LED2.gpioPin.mode = "Output"; +LED2.gpioPin.callbackFunction = ""; thread.deviceTypeReadOnly = true; thread.pm.$name = "ti_thread_pm_thread_pm0"; diff --git a/examples/pump-app/cc13x2x7_26x2x7/main/AppTask.cpp b/examples/pump-app/cc13x2x7_26x2x7/main/AppTask.cpp index a5883059abf71c..745949c9f30b56 100644 --- a/examples/pump-app/cc13x2x7_26x2x7/main/AppTask.cpp +++ b/examples/pump-app/cc13x2x7_26x2x7/main/AppTask.cpp @@ -163,15 +163,17 @@ int AppTask::Init() PLAT_LOG("Initialize buttons"); Button_init(); - Button_Params_init(&buttonParams); - buttonParams.buttonEventMask = Button_EV_CLICKED; - buttonParams.longPressDuration = 1000U; // ms - sAppLeftHandle = Button_open(CONFIG_BTN_LEFT, ButtonLeftEventHandler, &buttonParams); - Button_Params_init(&buttonParams); buttonParams.buttonEventMask = Button_EV_CLICKED | Button_EV_LONGPRESSED; buttonParams.longPressDuration = 5000U; // ms - sAppRightHandle = Button_open(CONFIG_BTN_RIGHT, ButtonRightEventHandler, &buttonParams); + sAppLeftHandle = Button_open(CONFIG_BTN_LEFT, &buttonParams); + Button_setCallback(sAppLeftHandle, ButtonLeftEventHandler); + + Button_Params_init(&buttonParams); + buttonParams.buttonEventMask = Button_EV_CLICKED; + buttonParams.longPressDuration = 1000U; // ms + sAppRightHandle = Button_open(CONFIG_BTN_RIGHT, &buttonParams); + Button_setCallback(sAppRightHandle, ButtonRightEventHandler); // Initialize Pump module PLAT_LOG("Initialize Pump"); @@ -182,7 +184,7 @@ int AppTask::Init() ConfigurationMgr().LogDeviceConfig(); // QR code will be used with CHIP Tool - PrintOnboardingCodes(chip::RendezvousInformationFlag::kBLE); + PrintOnboardingCodes(chip::RendezvousInformationFlags(chip::RendezvousInformationFlag::kBLE)); return 0; } @@ -220,7 +222,10 @@ void AppTask::ButtonLeftEventHandler(Button_Handle handle, Button_EventMask even { event.ButtonEvent.Type = AppEvent::kAppEventButtonType_Clicked; } - + else if (events & Button_EV_LONGPRESSED) + { + event.ButtonEvent.Type = AppEvent::kAppEventButtonType_LongPressed; + } // button callbacks are in ISR context if (xQueueSendFromISR(sAppEventQueue, &event, NULL) != pdPASS) { @@ -237,10 +242,6 @@ void AppTask::ButtonRightEventHandler(Button_Handle handle, Button_EventMask eve { event.ButtonEvent.Type = AppEvent::kAppEventButtonType_Clicked; } - else if (events & Button_EV_LONGPRESSED) - { - event.ButtonEvent.Type = AppEvent::kAppEventButtonType_LongPressed; - } // button callbacks are in ISR context if (xQueueSendFromISR(sAppEventQueue, &event, NULL) != pdPASS) { @@ -300,7 +301,7 @@ void AppTask::DispatchEvent(AppEvent * aEvent) { switch (aEvent->Type) { - case AppEvent::kEventType_ButtonLeft: + case AppEvent::kEventType_ButtonRight: if (AppEvent::kAppEventButtonType_Clicked == aEvent->ButtonEvent.Type) { // Toggle Pump state @@ -315,7 +316,7 @@ void AppTask::DispatchEvent(AppEvent * aEvent) } break; - case AppEvent::kEventType_ButtonRight: + case AppEvent::kEventType_ButtonLeft: if (AppEvent::kAppEventButtonType_Clicked == aEvent->ButtonEvent.Type) { // Toggle BLE advertisements diff --git a/examples/pump-app/cc13x2x7_26x2x7/main/main.cpp b/examples/pump-app/cc13x2x7_26x2x7/main/main.cpp index 03a1b6badebe6f..fb7c602527fea5 100644 --- a/examples/pump-app/cc13x2x7_26x2x7/main/main.cpp +++ b/examples/pump-app/cc13x2x7_26x2x7/main/main.cpp @@ -33,11 +33,13 @@ #include #include +#include +#include #include #include #include -#define TOTAL_ICALL_HEAP_SIZE (0xC000) +#define TOTAL_ICALL_HEAP_SIZE (0xc000) using namespace ::chip; using namespace ::chip::Inet; @@ -71,6 +73,10 @@ int main(void) UART_init(); + ECDH_init(); + + ECDSA_init(); + ECJPAKE_init(); AESECB_init(); diff --git a/examples/pump-controller-app/cc13x2x7_26x2x7/README.md b/examples/pump-controller-app/cc13x2x7_26x2x7/README.md index b252a2fce7d6a1..7c9e21ce726d91 100644 --- a/examples/pump-controller-app/cc13x2x7_26x2x7/README.md +++ b/examples/pump-controller-app/cc13x2x7_26x2x7/README.md @@ -49,13 +49,13 @@ pump and to control the state. The user LEDs on the LaunchPad are set on when the pump is started, and are set off when stopped. The LEDs will flash when in the transition state between started and stopped. -Short presses (less than 1000ms) of the left user button (`BTN-1`) are used for +Short presses (less than 1000ms) of the right user button (`BTN-2`) are used for toggling the pump state. -Short presses (less than 1000ms) of the right user button (`BTN-2`) are used for +Short presses (less than 1000ms) of the left user button (`BTN-1`) are used for toggling Matter BLE advertisements. -Long presses (greater than 5000ms) of the right user button (`BTN-2`) will +Long presses (greater than 5000ms) of the left user button (`BTN-1`) will initiate a factory reset of the device clearing all stored provisioning information to allow for a new network setup. @@ -76,8 +76,8 @@ section will need to be done when migrating to new versions of the SDK. packaged with the TI SDK. Check the following section for a list of changes needed. -- Download and install [SysConfig][sysconfig] - ([sysconfig-1.5.0_1397][sysconfig-1.5.0_1397]) +- Download and install [SysConfig][sysconfig] ([recommended + version][sysconfig_recommended]) - This may have already been installed with your SimpleLink SDK install. @@ -101,7 +101,7 @@ section will need to be done when migrating to new versions of the SDK. ``` $ cd ~/connectedhomeip - $ source ./script/bootstrap.sh + $ source ./scripts/bootstrap.sh ``` @@ -114,21 +114,21 @@ Ninja to build the executable. ``` $ cd ~/connectedhomeip - $ source ./script/activate.sh + $ source ./scripts/activate.sh ``` - Run the build to produce a default executable. By default on Linux both the TI SimpleLink SDK and Sysconfig are located in a `ti` folder in the user's home directory, and you must provide the absolute path to them. For example - `/home/username/ti/simplelink_cc13x2_26x2_sdk_4_40_05_02_eng` and - `/home/username/ti/sysconfig_1.6.0`. On Windows the default directory is - `C:\ti` + `/home/username/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng` and + `/home/username/ti/sysconfig_1.10.0`. On Windows the default directory is + `C:\ti`. ``` - $ cd ~/connectedhomeip/examples/pump-controller-app/cc13x2_26x2 - $ export TI_SIMPLELINK_SDK_ROOT= - $ export TI_SYSCONFIG_ROOT= + $ cd ~/connectedhomeip/examples/pump-controller-app/cc13x2x7_26x2x7 + $ export TI_SIMPLELINK_SDK_ROOT=$HOME/ti/simplelink_cc13xx_cc26xx_sdk_5_30_03_01_eng + $ export TI_SYSCONFIG_ROOT=$HOME/ti/sysconfig_1.10.0 $ gn gen out/debug --args="ti_simplelink_sdk_root=\"${TI_SIMPLELINK_SDK_ROOT}\" ti_sysconfig_root=\"${TI_SYSCONFIG_ROOT}\"" $ ninja -C out/debug @@ -221,12 +221,9 @@ Additionally, we welcome any feedback. [matter]: https://github.com/project-chip/connectedhomeip [cc1352r1_launchxl]: https://www.ti.com/tool/LAUNCHXL-CC1352R1 [e2e]: https://e2e.ti.com/support/wireless-connectivity/zigbee-and-thread -[simplelink_sdk]: https://www.ti.com/tool/SIMPLELINK-CC13X2-26X2-SDK -[simplelink_sdk_4.30.00.54]: - https://www.ti.com/tool/download/SIMPLELINK-CC13X2-26X2-SDK/4.30.00.54 [sysconfig]: https://www.ti.com/tool/SYSCONFIG -[sysconfig-1.5.0_1397]: - http://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.5.0_1397-setup.run +[sysconfig_recommended]: + https://software-dl.ti.com/ccs/esd/sysconfig/sysconfig-1.10.0_2163-setup.run [ti_thread_dnd]: https://www.ti.com/wireless-connectivity/thread/design-development.html [ti_cc13x2_26x2_r7_matter_request]: https://ti.com/chip_sdk diff --git a/examples/pump-controller-app/cc13x2x7_26x2x7/chip.syscfg b/examples/pump-controller-app/cc13x2x7_26x2x7/chip.syscfg index 38a53e57d500fd..cd153a744fea1d 100644 --- a/examples/pump-controller-app/cc13x2x7_26x2x7/chip.syscfg +++ b/examples/pump-controller-app/cc13x2x7_26x2x7/chip.syscfg @@ -145,7 +145,7 @@ ECJPAKE1.$name = "CONFIG_ECJPAKE_0"; NVS1.$name = "CONFIG_NVSINTERNAL"; NVS1.internalFlash.$name = "ti_drivers_nvs_NVSCC26XX0"; -NVS1.internalFlash.regionBase = 0x78000; +NVS1.internalFlash.regionBase = 0xAA000; NVS1.internalFlash.regionSize = 0x4000; RTOS.name = "FreeRTOS"; @@ -169,25 +169,33 @@ UART2.$name = "CONFIG_DISPLAY_UART"; UART2.txPinInstance.$name = "CONFIG_PIN_6"; UART2.rxPinInstance.$name = "CONFIG_PIN_7"; -Button1.$name = "CONFIG_BTN_LEFT"; -Button1.gpioPin.$name = "CONFIG_GPIO_BTN1"; -Button1.gpioPin.gpioPin.$assign = "boosterpack.32"; -Button1.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN1"; - -Button2.$name = "CONFIG_BTN_RIGHT"; -Button2.gpioPin.$name = "CONFIG_GPIO_BTN2"; -Button2.gpioPin.gpioPin.$assign = "boosterpack.13"; -Button2.gpioPin.pinInstance.$name = "CONFIG_PIN_BTN2"; - -LED1.$name = "CONFIG_LED_RED"; -LED1.gpioPin.$name = "CONFIG_GPIO_RLED"; -LED1.gpioPin.gpioPin.$assign = "boosterpack.10"; -LED1.gpioPin.pinInstance.$name = "CONFIG_PIN_RLED"; - -LED2.$name = "CONFIG_LED_GREEN"; -LED2.gpioPin.$name = "CONFIG_GPIO_GLED"; -LED2.gpioPin.gpioPin.$assign = "boosterpack.39"; -LED2.gpioPin.pinInstance.$name = "CONFIG_PIN_GLED"; +/* Left Button */ +Button1.$name = "CONFIG_BTN_LEFT"; +Button1.$hardware = system.deviceData.board.components["BTN-1"]; +Button1.gpioPin.$name = "CONFIG_GPIO_BTN1"; +Button1.gpioPin.pull = "Pull Up"; +Button1.gpioPin.interruptTrigger = "Falling Edge"; + +/* Left Button */ +Button2.$name = "CONFIG_BTN_RIGHT"; +Button2.$hardware = system.deviceData.board.components["BTN-2"]; +Button2.gpioPin.$name = "CONFIG_GPIO_BTN2"; +Button2.gpioPin.pull = "Pull Up"; +Button2.gpioPin.interruptTrigger = "Falling Edge"; + +/* Red LED */ +LED1.$name = "CONFIG_LED_RED"; +LED1.$hardware = system.deviceData.board.components.LED_RED; +LED1.gpioPin.$name = "CONFIG_GPIO_RLED"; +LED1.gpioPin.mode = "Output"; +LED1.gpioPin.callbackFunction = ""; + +/* Green LED */ +LED2.$name = "CONFIG_LED_GREEN"; +LED2.$hardware = system.deviceData.board.components.LED_GREEN; +LED2.gpioPin.$name = "CONFIG_GPIO_GLED"; +LED2.gpioPin.mode = "Output"; +LED2.gpioPin.callbackFunction = ""; thread.deviceTypeReadOnly = true; thread.pm.$name = "ti_thread_pm_thread_pm0"; diff --git a/examples/pump-controller-app/cc13x2x7_26x2x7/main/AppTask.cpp b/examples/pump-controller-app/cc13x2x7_26x2x7/main/AppTask.cpp index fd87dd8e650e05..ee332a4dadbd10 100644 --- a/examples/pump-controller-app/cc13x2x7_26x2x7/main/AppTask.cpp +++ b/examples/pump-controller-app/cc13x2x7_26x2x7/main/AppTask.cpp @@ -158,15 +158,17 @@ int AppTask::Init() PLAT_LOG("Initialize buttons"); Button_init(); - Button_Params_init(&buttonParams); - buttonParams.buttonEventMask = Button_EV_CLICKED; - buttonParams.longPressDuration = 1000U; // ms - sAppLeftHandle = Button_open(CONFIG_BTN_LEFT, ButtonLeftEventHandler, &buttonParams); - Button_Params_init(&buttonParams); buttonParams.buttonEventMask = Button_EV_CLICKED | Button_EV_LONGPRESSED; buttonParams.longPressDuration = 5000U; // ms - sAppRightHandle = Button_open(CONFIG_BTN_RIGHT, ButtonRightEventHandler, &buttonParams); + sAppLeftHandle = Button_open(CONFIG_BTN_LEFT, &buttonParams); + Button_setCallback(sAppLeftHandle, ButtonLeftEventHandler); + + Button_Params_init(&buttonParams); + buttonParams.buttonEventMask = Button_EV_CLICKED; + buttonParams.longPressDuration = 1000U; // ms + sAppRightHandle = Button_open(CONFIG_BTN_RIGHT, &buttonParams); + Button_setCallback(sAppRightHandle, ButtonRightEventHandler); // Initialize Pump module PLAT_LOG("Initialize Pump"); @@ -177,7 +179,7 @@ int AppTask::Init() ConfigurationMgr().LogDeviceConfig(); // QR code will be used with CHIP Tool - PrintOnboardingCodes(chip::RendezvousInformationFlag::kBLE); + PrintOnboardingCodes(chip::RendezvousInformationFlags(chip::RendezvousInformationFlag::kBLE)); return 0; } @@ -215,7 +217,10 @@ void AppTask::ButtonLeftEventHandler(Button_Handle handle, Button_EventMask even { event.ButtonEvent.Type = AppEvent::kAppEventButtonType_Clicked; } - + else if (events & Button_EV_LONGPRESSED) + { + event.ButtonEvent.Type = AppEvent::kAppEventButtonType_LongPressed; + } // button callbacks are in ISR context if (xQueueSendFromISR(sAppEventQueue, &event, NULL) != pdPASS) { @@ -232,10 +237,6 @@ void AppTask::ButtonRightEventHandler(Button_Handle handle, Button_EventMask eve { event.ButtonEvent.Type = AppEvent::kAppEventButtonType_Clicked; } - else if (events & Button_EV_LONGPRESSED) - { - event.ButtonEvent.Type = AppEvent::kAppEventButtonType_LongPressed; - } // button callbacks are in ISR context if (xQueueSendFromISR(sAppEventQueue, &event, NULL) != pdPASS) { @@ -295,7 +296,7 @@ void AppTask::DispatchEvent(AppEvent * aEvent) { switch (aEvent->Type) { - case AppEvent::kEventType_ButtonLeft: + case AppEvent::kEventType_ButtonRight: if (AppEvent::kAppEventButtonType_Clicked == aEvent->ButtonEvent.Type) { // Toggle Pump state @@ -310,7 +311,7 @@ void AppTask::DispatchEvent(AppEvent * aEvent) } break; - case AppEvent::kEventType_ButtonRight: + case AppEvent::kEventType_ButtonLeft: if (AppEvent::kAppEventButtonType_Clicked == aEvent->ButtonEvent.Type) { // Toggle BLE advertisements diff --git a/examples/pump-controller-app/cc13x2x7_26x2x7/main/main.cpp b/examples/pump-controller-app/cc13x2x7_26x2x7/main/main.cpp index 03a1b6badebe6f..fb7c602527fea5 100644 --- a/examples/pump-controller-app/cc13x2x7_26x2x7/main/main.cpp +++ b/examples/pump-controller-app/cc13x2x7_26x2x7/main/main.cpp @@ -33,11 +33,13 @@ #include #include +#include +#include #include #include #include -#define TOTAL_ICALL_HEAP_SIZE (0xC000) +#define TOTAL_ICALL_HEAP_SIZE (0xc000) using namespace ::chip; using namespace ::chip::Inet; @@ -71,6 +73,10 @@ int main(void) UART_init(); + ECDH_init(); + + ECDSA_init(); + ECJPAKE_init(); AESECB_init(); diff --git a/src/platform/cc13x2_26x2/BLEManagerImpl.cpp b/src/platform/cc13x2_26x2/BLEManagerImpl.cpp index 15d28f75ffef88..e917cedeee1066 100644 --- a/src/platform/cc13x2_26x2/BLEManagerImpl.cpp +++ b/src/platform/cc13x2_26x2/BLEManagerImpl.cpp @@ -692,8 +692,6 @@ CHIP_ERROR BLEManagerImpl::CreateEventHandler(void) */ uint8_t BLEManagerImpl::ProcessStackEvent(ICall_Hdr * pMsg) { - BLEMGR_LOG("BLEMGR: BLE Process Stack Event"); - // Always dealloc pMsg unless set otherwise uint8_t safeToDealloc = TRUE; @@ -783,8 +781,6 @@ void BLEManagerImpl::ProcessEvtHdrMsg(QueuedEvt_t * pMsg) { bool dealloc = TRUE; - BLEMGR_LOG("BLEMGR: ProcessEvtHdrMsg"); - switch (pMsg->event) { /* External CHIPoBLE Event trigger */ @@ -1391,7 +1387,6 @@ status_t BLEManagerImpl::EnqueueEvtHdrMsg(uint8_t event, void * pData) if (sInstance.mFlags.Has(Flags::kBLEStackInitialized)) { QueuedEvt_t * pMsg = (QueuedEvt_t *) ICall_malloc(sizeof(QueuedEvt_t)); - BLEMGR_LOG("BLEMGR: EnqueueEvtHdrMsg"); // Create dynamic pointer to message. if (pMsg) @@ -1402,8 +1397,6 @@ status_t BLEManagerImpl::EnqueueEvtHdrMsg(uint8_t event, void * pData) // Enqueue the message. success = Util_enqueueMsg(sEventHandlerMsgQueueID, BLEManagerImpl::sSyncEvent, (uint8_t *) pMsg); - BLEMGR_LOG("BLEMGR: Util_enqueueMsg compelte"); - return (success) ? SUCCESS : FAILURE; } @@ -1620,7 +1613,6 @@ void BLEManagerImpl::ClearPendingBLEParamUpdate(uint16_t connHandle) void BLEManagerImpl::UpdateBLERPA(void) { uint8_t * pRpaNew; - BLEMGR_LOG("BLEMGR: UpdateBLERPA"); // Read the current RPA. pRpaNew = GAP_GetDevAddress(FALSE); @@ -1633,7 +1625,6 @@ void BLEManagerImpl::UpdateBLERPA(void) void BLEManagerImpl::EventHandler(void * arg) { - BLEMGR_LOG("BLEMGR: EventHandler"); PlatformMgr().LockChipStack(); sInstance.EventHandler_init(); @@ -1657,12 +1648,9 @@ void BLEManagerImpl::EventHandler(void * arg) /* Lock CHIP Stack while processing BLE Stack/App events */ PlatformMgr().LockChipStack(); - BLEMGR_LOG("BLEMGR: EventHandler: Events received"); // Fetch any available messages that might have been sent from the stack if (ICall_fetchServiceMsg(&src, &dest, (void **) &hcipMsg) == ICALL_ERRNO_SUCCESS) { - BLEMGR_LOG("BLEMGR: EventHandler: Stack Event"); - uint8 safeToDealloc = TRUE; if ((src == ICALL_SERVICE_CLASS_BLE) && (dest == BLEManagerImpl::sSelfEntity)) @@ -1686,8 +1674,6 @@ void BLEManagerImpl::EventHandler(void * arg) // If RTOS queue is not empty, process CHIP messages. if (events & QUEUE_EVT) { - BLEMGR_LOG("BLEMGR: EventHandler: App Event"); - QueuedEvt_t * pMsg; for (;;) { @@ -1767,11 +1753,9 @@ void BLEManagerImpl::AdvTimeoutHandler(uintptr_t arg) void BLEManagerImpl::ClockHandler(uintptr_t arg) { ClockEventData_t * pData = (ClockEventData_t *) arg; - BLEMGR_LOG("BLEMGR: ClockHandler"); if (pData->event == READ_RPA_EVT) { - BLEMGR_LOG("BLEMGR: ClockHandler RPA EVT"); // Start the next period Util_startClock(&sInstance.clkRpaRead); @@ -1780,7 +1764,6 @@ void BLEManagerImpl::ClockHandler(uintptr_t arg) } else if (pData->event == SEND_PARAM_UPDATE_EVT) { - BLEMGR_LOG("BLEMGR: ClockHandler PARAM UPDATE EVT"); // Send message to app if (sInstance.EnqueueEvtHdrMsg(SEND_PARAM_UPDATE_EVT, pData) != SUCCESS) { diff --git a/src/platform/cc13x2_26x2/CHIPDevicePlatformConfig.h b/src/platform/cc13x2_26x2/CHIPDevicePlatformConfig.h index bddae508acea2b..26a1537fe9c178 100644 --- a/src/platform/cc13x2_26x2/CHIPDevicePlatformConfig.h +++ b/src/platform/cc13x2_26x2/CHIPDevicePlatformConfig.h @@ -39,7 +39,7 @@ // ========== CHIP Platform Configuration ========= -#define CHIP_DEVICE_CONFIG_CHIP_TASK_STACK_SIZE (5120) +#define CHIP_DEVICE_CONFIG_CHIP_TASK_STACK_SIZE (6144) #define BLEMANAGER_EVENT_HANDLER_STACK_SIZE (4096) #define BLEMANAGER_EVENT_HANDLER_PRIORITY (2) diff --git a/src/platform/cc13x2_26x2/FreeRTOSConfig.h b/src/platform/cc13x2_26x2/FreeRTOSConfig.h index 532b7767270d67..1d3798b6c8dcc3 100644 --- a/src/platform/cc13x2_26x2/FreeRTOSConfig.h +++ b/src/platform/cc13x2_26x2/FreeRTOSConfig.h @@ -106,7 +106,7 @@ #define configTIMER_TASK_PRIORITY (6) #define configTIMER_QUEUE_LENGTH (20) #define configTIMER_TASK_STACK_DEPTH (configMINIMAL_STACK_SIZE * 2) - +#define configIDLE_TASK_STACK_DEPTH (configMINIMAL_STACK_SIZE) #define configENABLE_BACKWARD_COMPATIBILITY 1 #if defined(__TI_COMPILER_VERSION__) || defined(__ti_version__) diff --git a/src/platform/cc13x2_26x2/ThreadStackManagerImpl.cpp b/src/platform/cc13x2_26x2/ThreadStackManagerImpl.cpp index efb645eaa0b3e6..ca5bfaacba426a 100644 --- a/src/platform/cc13x2_26x2/ThreadStackManagerImpl.cpp +++ b/src/platform/cc13x2_26x2/ThreadStackManagerImpl.cpp @@ -210,3 +210,13 @@ extern "C" void otSysProcessDrivers(otInstance * aInstance) { ThreadStackMgrImpl()._ProcMessage(aInstance); } + +/** + * Get a pointer to the OpenThread instance object. + * + * @return Pointer to the OpenThread instance object. + */ +extern "C" otInstance * OtInstance_get(void) +{ + return ThreadStackMgrImpl().OTInstance(); +} diff --git a/src/platform/cc13x2_26x2/ble_user_config.c b/src/platform/cc13x2_26x2/ble_user_config.c new file mode 100644 index 00000000000000..4cc4e2f12c1695 --- /dev/null +++ b/src/platform/cc13x2_26x2/ble_user_config.c @@ -0,0 +1,530 @@ +/* + * + * Copyright (c) 2020 Project CHIP Authors + * Copyright (c) 2020 Texas Instruments Incorporated + * + * 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. + */ + +/** + * @file This file contains user configurable variables for the BLE + * Application. + * + */ + +/******************************************************************************* + * INCLUDES + */ + +#include "ble_user_config.h" +#include "hal_types.h" +#ifndef FREERTOS +#include +#endif +#ifdef SYSCFG +#include "ti_ble_config.h" +#else +#include +#endif + +#if defined(AOA_SYSCFG) +#include "ti_aoa_config.h" +#endif + +#include "ble_overrides.h" +#include "ti_radio_config.h" + +#include + +#include "ecc/ECCROMCC26XX.h" + +#include +#include +#include +#include +#include + +#ifdef FREERTOS +#define Swi_restore SwiP_restore +#define Swi_disable SwiP_disable +#include +#else +#include +#endif +#include +#include + +#if !defined(DeviceFamily_CC26X1) +#include +#include +#else +#include +#endif + +/******************************************************************************* + * MACROS + */ + +/******************************************************************************* + * CONSTANTS + */ + +// Tx Power +#define NUM_TX_POWER_VALUES (RF_BLE_TX_POWER_TABLE_SIZE - 1) + +#ifndef SYSCFG +// Default Tx Power Index +#if defined(CC13X2P) +#define DEFAULT_TX_POWER HCI_EXT_TX_POWER_0_DBM +#else // !CC13X2 +#define DEFAULT_TX_POWER HCI_EXT_TX_POWER_0_DBM +#endif // CC13X2 +#endif // SYSCFG + +// Override NOP +#define OVERRIDE_NOP 0xC0000001 + +/******************************************************************************* + * TYPEDEFS + */ + +/******************************************************************************* + * LOCAL VARIABLES + */ + +/********************************************************************* + * LOCAL FUNCTIONS + */ + +void driverTable_fnSpinlock(void); + +/******************************************************************************* + * GLOBAL VARIABLES + */ + +// Tx Power Table +txPwrTbl_t appTxPwrTbl = { (txPwrVal_t *) RF_BLE_txPowerTable, + NUM_TX_POWER_VALUES, // max + DEFAULT_TX_POWER }; // default + +#if defined(CC13X2P) && defined(CC13X2P_2_LAUNCHXL) + +// Tx Power Backoff Values (txPwrBackoff1MPhy,txPwrBackoff2MPhy ,txPwrBackoffCoded) +const txPwrBackoffVal_t TxPowerBackoffTable[] = { + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 0 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 1 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 2 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 3 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 4 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 5 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 6 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 7 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 8 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 9 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 10 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 11 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 12 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 13 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 14 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 15 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 16 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 17 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 18 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 19 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 20 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 21 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 22 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 23 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 24 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 25 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 26 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 27 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 28 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 29 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 30 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 31 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 32 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 33 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_17_DBM_P4_9_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 34 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_18_DBM_P4_10_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 35 + { HCI_EXT_TX_POWER_P2_19_DBM, HCI_EXT_TX_POWER_P2_18_DBM_P4_10_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 36 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 37 + { HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM }, // max tx power for channel 38 + { HCI_EXT_TX_POWER_P2_14_DBM_P4_6_DBM, HCI_EXT_TX_POWER_P2_20_DBM, HCI_EXT_TX_POWER_P2_20_DBM } +}; // max tx power for channel 39 + +txPwrBackoffTbl_t appTxPwrBackoffTbl = { TxPowerBackoffTable }; + +#endif // defined(CC13X2P) && defined(CC13X2P_2_LAUNCHXL) + +#if !defined(AOA_SYSCFG) +// Antenna board configurations (example for a 12-antenna board) +// Maximum number of antennas +#define ANTENNA_TABLE_SIZE 12 +// BitMask of all the relevant GPIOs which needed for the antennas +#define ANTENNA_IO_MASK BV(27) | BV(28) | BV(29) | BV(30) + +// Antenna GPIO configuration (should be adapted to the antenna board design) +antennaIOEntry_t antennaTbl[ANTENNA_TABLE_SIZE] = { + 0, // antenna 0 GPIO configuration (all GPIOs in ANTENNA_IO_MASK are LOW) + BV(28), // antenna 1 + BV(29), // antenna 2 + BV(28) | BV(29), // antenna 3 + BV(30), // antenna 4 + BV(28) | BV(30), // antenna 5 + BV(27), // antenna 6 + BV(27) | BV(28), // antenna 7 + BV(27) | BV(29), // antenna 8 + BV(27) | BV(28) | BV(29), // antenna 9 + BV(27) | BV(30), // antenna 10 + BV(27) | BV(28) | BV(30) // antenna 11 +}; + +#endif //! defined(AOA_SYSCFG) + +// Antenna properties passes to the stack +cteAntProp_t appCTEAntProp = { ANTENNA_IO_MASK, ANTENNA_TABLE_SIZE, antennaTbl }; + +#if !defined(DeviceFamily_CC26X1) +ECCParams_CurveParams eccParams_NISTP256 = { .curveType = ECCParams_CURVE_TYPE_SHORT_WEIERSTRASS_AN3, + .length = NISTP256_PARAM_SIZE_BYTES, + .prime = NISTP256_prime.byte, + .order = NISTP256_order.byte, + .a = NISTP256_a.byte, + .b = NISTP256_b.byte, + .generatorX = NISTP256_generator.x.byte, + .generatorY = NISTP256_generator.y.byte, + .cofactor = 1 }; +#else +ECCParams_CurveParams eccParams_NISTP256 = { .curveType = ECCParams_CURVE_TYPE_SHORT_WEIERSTRASS_AN3, + .length = ECC_NISTP256_PARAM_LENGTH_BYTES, + .prime = ECC_NISTP256_prime.byte, + .order = ECC_NISTP256_order.byte, + .a = ECC_NISTP256_a.byte, + .b = ECC_NISTP256_b.byte, + .generatorX = ECC_NISTP256_generatorX.byte, + .generatorY = ECC_NISTP256_generatorY.byte, + .cofactor = 1 }; +#endif + +#if defined(USE_COEX) +// 2 structures generated by the sysconfig RF for coexistence configuration +extern rfCoreHal_bleCoExConfig_t coexConfig; +extern RF_CoexOverride_BLEUseCases coexConfigBle; +// from coexConfig struct we will get the coex enable and coex type (3 or 1 wire) +// from coexConfigBle struct we will get the priority and rx request per use case: +// initiator, connected, Broadcaster and Observer. +coexUseCaseConfig_t coexSysConfig = { &coexConfig, &coexConfigBle }; +#endif + +#ifdef ICALL_JT +#include + +// RF Driver API Table +rfDrvTblPtr_t rfDriverTableBLE[] = { + (uint32) RF_open, + (uint32) driverTable_fnSpinlock, // RF_close +#ifdef RF_SINGLEMODE + (uint32) RF_postCmd, +#else // !RF_SINGLEMODE + (uint32) driverTable_fnSpinlock, // RF_postCmd +#endif // RF_SINGLEMODE + (uint32) driverTable_fnSpinlock, // RF_pendCmd +#ifdef RF_SINGLEMODE + (uint32) RF_runCmd, +#else // !RF_SINGLEMODE + (uint32) driverTable_fnSpinlock, // RF_runCmd +#endif // RF_SINGLEMODE + (uint32) RF_cancelCmd, + (uint32) RF_flushCmd, + (uint32) driverTable_fnSpinlock, // RF_yield + (uint32) RF_Params_init, + (uint32) RF_runImmediateCmd, + (uint32) RF_runDirectCmd, + (uint32) RF_ratCompare, + (uint32) driverTable_fnSpinlock, // RF_ratCapture + (uint32) RF_ratDisableChannel, + (uint32) RF_getCurrentTime, + (uint32) RF_getRssi, + (uint32) RF_getInfo, + (uint32) RF_getCmdOp, + (uint32) RF_control, + (uint32) driverTable_fnSpinlock, // RF_getTxPower + (uint32) RF_setTxPower, // RF_setTxPower + (uint32) driverTable_fnSpinlock, // RF_TxPowerTable_findPowerLevel + (uint32) driverTable_fnSpinlock, // RF_TxPowerTable_findValue +#ifndef RF_SINGLEMODE + (uint32) RF_scheduleCmd, + (uint32) RF_runScheduleCmd, + (uint32) driverTable_fnSpinlock, // RF_requestAccess +#endif // !RF_SINGLEMODE +}; + +cryptoDrvTblPtr_t cryptoDriverTableBLE[] = { (uint32) AESCCM_init, + (uint32) AESCCM_open, + (uint32) AESCCM_close, + (uint32) AESCCM_Params_init, + (uint32) AESCCM_Operation_init, + (uint32) AESCCM_oneStepEncrypt, + (uint32) AESCCM_oneStepDecrypt, + (uint32) AESECB_init, + (uint32) AESECB_open, + (uint32) AESECB_close, + (uint32) AESECB_Params_init, + (uint32) AESECB_Operation_init, + (uint32) AESECB_oneStepEncrypt, + (uint32) AESECB_oneStepDecrypt, + (uint32) CryptoKeyPlaintext_initKey, + (uint32) CryptoKeyPlaintext_initBlankKey }; + +// Swi APIs needed by BLE controller +rtosApiTblPtr_t rtosApiTable[] = { (uint32_t) Swi_disable, (uint32_t) Swi_restore }; + +// BLE Stack Configuration Structure +const stackSpecific_t bleStackConfig = { .maxNumConns = MAX_NUM_BLE_CONNS, + .maxNumPDUs = MAX_NUM_PDU, + .maxPduSize = 0, + .maxNumPSM = L2CAP_NUM_PSM, + .maxNumCoChannels = L2CAP_NUM_CO_CHANNELS, + .maxWhiteListElems = MAX_NUM_WL_ENTRIES, + .maxResolvListElems = CFG_MAX_NUM_RL_ENTRIES, + .pfnBMAlloc = &pfnBMAlloc, + .pfnBMFree = &pfnBMFree, + .rfDriverParams.powerUpDurationMargin = RF_POWER_UP_DURATION_MARGIN, + .rfDriverParams.inactivityTimeout = RF_INACTIVITY_TIMEOUT, + .rfDriverParams.powerUpDuration = RF_POWER_UP_DURATION, + .rfDriverParams.pErrCb = &(RF_ERR_CB), + .eccParams = &eccParams_NISTP256, + .fastStateUpdateCb = NULL, + .bleStackType = 0, +#ifdef CC2652RB_LAUNCHXL + .extStackSettings = EXTENDED_STACK_SETTINGS | CC2652RB_OVERRIDE_USED, +#else + .extStackSettings = EXTENDED_STACK_SETTINGS, +#endif + .maxNumCteBuffers = MAX_NUM_CTE_BUFS }; + +uint16_t bleUserCfg_maxPduSize = MAX_PDU_SIZE; + +#ifdef OSAL_SNV_EXTFLASH +const extflashDrvTblPtr_t extflashDriverTable[] = { (uint32) ExtFlash_open, (uint32) ExtFlash_close, (uint32) ExtFlash_read, + (uint32) ExtFlash_write, (uint32) ExtFlash_erase }; +#endif // OSAL_SNV_EXTFLASH + +// Table for Driver can be found in icall_user_config.c +// if a driver is not to be used, then the pointer shoul dbe set to NULL, +// for this example, this is done in ble_user_config.h +const drvTblPtr_t driverTable = { + .rfDrvTbl = rfDriverTableBLE, + .eccDrvTbl = eccDriverTable, + .cryptoDrvTbl = cryptoDriverTableBLE, + .trngDrvTbl = trngDriverTable, + .rtosApiTbl = rtosApiTable, + .nvintfStructPtr = &nvintfFncStruct, +#ifdef OSAL_SNV_EXTFLASH + .extflashDrvTbl = extflashDriverTable, +#endif // OSAL_SNV_EXTFLASH +}; + +const boardConfig_t boardConfig = { + .rfFeModeBias = RF_FE_MODE_AND_BIAS, + .rfRegTbl = (regOverride_t *) pOverrides_bleCommon, + .rfRegTbl1M = (regOverride_t *) pOverrides_ble1Mbps, +#if defined(BLE_V50_FEATURES) && (BLE_V50_FEATURES & (PHY_2MBPS_CFG | PHY_LR_CFG)) + // Currently, no overrides for 2M and Coded, so exclude from build. + .rfRegTbl2M = (regOverride_t *) pOverrides_ble2Mbps, + .rfRegTblCoded = (regOverride_t *) pOverrides_bleCoded, +#endif // PHY_2MBPS_CFG | PHY_LR_CFG + .txPwrTbl = &appTxPwrTbl, +#if defined(CC13X2P) +#if defined(CC13X2P_2_LAUNCHXL) + .txPwrBackoffTbl = &appTxPwrBackoffTbl, +#else + .txPwrBackoffTbl = NULL, +#endif // defined(CC13X2P_2_LAUNCHXL) + .rfRegOverrideTxStdTblptr = (regOverride_t *) pOverrides_bleTxStd, // Default PA + .rfRegOverrideTx20TblPtr = (regOverride_t *) pOverrides_bleTx20, // High power PA +#endif // CC13X2P +#if defined(RTLS_CTE) + .rfRegOverrideCtePtr = (regOverride_t *) (pOverrides_bleCommon + BLE_STACK_OVERRIDES_OFFSET + CTE_OVERRIDES_OFFSET), + .cteAntennaPropPtr = &appCTEAntProp, +#else + .rfRegOverrideCtePtr = NULL, + .cteAntennaPropPtr = NULL, +#endif + .privOverrideOffset = BLE_STACK_OVERRIDES_OFFSET + PRIVACY_OVERRIDE_OFFSET, +#if defined(USE_COEX) + .coexUseCaseConfigPtr = &coexSysConfig, +#else + .coexUseCaseConfigPtr = NULL, +#endif +}; + +#else /* !(ICALL_JT) */ + +// RF Driver API Table +rfDrvTblPtr_t rfDriverTable[] = { + (uint32) RF_open, + (uint32) driverTable_fnSpinlock, // RF_close +#ifdef RF_SINGLEMODE + (uint32) RF_postCmd, +#else // !RF_SINGLEMODE + (uint32) driverTable_fnSpinlock, // RF_postCmd +#endif // RF_SINGLEMODE + (uint32) driverTable_fnSpinlock, // RF_pendCmd +#ifdef RF_SINGLEMODE + (uint32) RF_runCmd, +#else // !RF_SINGLEMODE + (uint32) driverTable_fnSpinlock, // RF_runCmd +#endif // RF_SINGLEMODE + (uint32) RF_cancelCmd, + (uint32) RF_flushCmd, + (uint32) driverTable_fnSpinlock, // RF_yield + (uint32) RF_Params_init, + (uint32) RF_runImmediateCmd, + (uint32) RF_runDirectCmd, + (uint32) RF_ratCompare(uint32) driverTable_fnSpinlock, // RF_ratCapture + (uint32) driverTable_fnSpinlock, // RF_ratDisableChannel + (uint32) RF_getCurrentTime, + (uint32) RF_getRssi, + (uint32) RF_getInfo, + (uint32) RF_getCmdOp, + (uint32) RF_control, + (uint32) driverTable_fnSpinlock, // RF_getTxPower + (uint32) RF_setTxPower, // RF_setTxPower + (uint32) driverTable_fnSpinlock, // RF_TxPowerTable_findPowerLevel + (uint32) driverTable_fnSpinlock, // RF_TxPowerTable_findValue +#ifndef RF_SINGLEMODE + (uint32) RF_scheduleCmd, + (uint32) RF_runScheduleCmd, + (uint32) driverTable_fnSpinlock // RF_requestAccess +#endif // !RF_SINGLEMODE +}; + +// ECC Driver API Table +eccDrvTblPtr_t eccDriverTable[] = { (uint32) ECDH_init, + (uint32) ECDH_Params_init, + (uint32) ECDH_open, + (uint32) ECDH_close, + (uint32) ECDH_OperationGeneratePublicKey_init, + (uint32) ECDH_OperationComputeSharedSecret_init, + (uint32) ECDH_generatePublicKey, + (uint32) ECDH_computeSharedSecret }; + +// Crypto Driver API Table +cryptoDrvTblPtr_t cryptoDriverTable[] = { (uint32) AESCCM_init, + (uint32) AESCCM_open, + (uint32) AESCCM_close, + (uint32) AESCCM_Params_init, + (uint32) AESCCM_Operation_init, + (uint32) AESCCM_oneStepEncrypt, + (uint32) AESCCM_oneStepDecrypt, + (uint32) AESECB_init, + (uint32) AESECB_open, + (uint32) AESECB_close, + (uint32) AESECB_Params_init, + (uint32) AESECB_Operation_init, + (uint32) AESECB_oneStepEncrypt, + (uint32) AESECB_oneStepDecrypt, + (uint32) CryptoKeyPlaintext_initKey, + (uint32) CryptoKeyPlaintext_initBlankKey }; + +trngDrvTblPtr_t trngDriverTable[] = { (uint32) TRNG_init, (uint32) TRNG_open, (uint32) TRNG_generateEntropy, (uint32) TRNG_close }; + +#endif /* ICALL_JT */ + +/******************************************************************************* + * @fn RegisterAssertCback + * + * @brief This routine registers the Application's assert handler. + * + * input parameters + * + * @param appAssertHandler - Application's assert handler. + * + * output parameters + * + * @param None. + * + * @return None. + */ +void RegisterAssertCback(assertCback_t appAssertHandler) +{ + appAssertCback = appAssertHandler; + +#ifdef EXT_HAL_ASSERT + // also set the Assert callback pointer used by halAssertHandlerExt + // Note: Normally, this pointer will be intialized by the stack, but in the + // event HAL_ASSERT is used by the Application, we initialize it + // directly here. + halAssertCback = appAssertHandler; +#endif // EXT_HAL_ASSERT + + return; +} + +/******************************************************************************* + * @fn driverTable_fnSpinLock + * + * @brief This routine is used to trap calls to unpopulated indexes of + * driver function pointer tables. + * + * input parameters + * + * @param None. + * + * output parameters + * + * @param None. + * + * @return None. + */ +void driverTable_fnSpinlock(void) +{ + volatile uint8 i = 1; + + while (i) + ; +} + +/******************************************************************************* + * @fn DefaultAssertCback + * + * @brief This is the Application default assert callback, in the event + * none is registered. + * + * input parameters + * + * @param assertCause - Assert cause as defined in hal_assert.h. + * @param assertSubcause - Optional assert subcause (see hal_assert.h). + * + * output parameters + * + * @param None. + * + * @return None. + */ +void DefaultAssertCback(uint8 assertCause, uint8 assertSubcause1) +{ +#ifdef HAL_ASSERT_SPIN + driverTable_fnSpinlock(); +#endif // HAL_ASSERT_SPIN + + return; +} + +// Application Assert Callback Function Pointer +assertCback_t appAssertCback = DefaultAssertCback; + +/******************************************************************************* + */ diff --git a/src/platform/cc13x2_26x2/crypto/cc13x2_26x2-mbedtls-config.h b/src/platform/cc13x2_26x2/crypto/cc13x2_26x2-mbedtls-config.h index efca6840f5d25d..04e91bae54616b 100644 --- a/src/platform/cc13x2_26x2/crypto/cc13x2_26x2-mbedtls-config.h +++ b/src/platform/cc13x2_26x2/crypto/cc13x2_26x2-mbedtls-config.h @@ -39,10 +39,14 @@ /* Enable Hardware Acceleration */ -#define MBEDTLS_ECJPAKE_ALT #define MBEDTLS_AES_ALT -#define MBEDTLS_SHA256_ALT +#define MBEDTLS_ECDH_COMPUTE_SHARED_ALT +#define MBEDTLS_ECDH_GEN_PUBLIC_ALT +#define MBEDTLS_ECDSA_SIGN_ALT +#define MBEDTLS_ECDSA_VERIFY_ALT +#define MBEDTLS_ECJPAKE_ALT #define MBEDTLS_ENTROPY_HARDWARE_ALT +#define MBEDTLS_SHA256_ALT /** * Enable Crypto and Entropy modules diff --git a/src/platform/cc13x2_26x2/crypto/ecdh_alt.c b/src/platform/cc13x2_26x2/crypto/ecdh_alt.c new file mode 100644 index 00000000000000..a3074a9c8a2435 --- /dev/null +++ b/src/platform/cc13x2_26x2/crypto/ecdh_alt.c @@ -0,0 +1,314 @@ +/* + * + * Copyright (c) 2021 Project CHIP Authors + * Copyright (c) 2021 Texas Instruments Incorporated + * + * 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. + */ + +#if !defined(MBEDTLS_CONFIG_FILE) +#include "mbedtls/config.h" +#else +#include MBEDTLS_CONFIG_FILE +#endif + +#if defined MBEDTLS_ECDH_COMPUTE_SHARED_ALT || defined MBEDTLS_ECDH_GEN_PUBLIC_ALT + +#include + +#include +#include +#include +#include + +#include "ti_drivers_config.h" + +#include +#include +#include + +/* Parameter validation macros based on platform_util.h */ +#define ECDH_VALIDATE_RET(cond) MBEDTLS_INTERNAL_VALIDATE_RET(cond, MBEDTLS_ERR_ECP_BAD_INPUT_DATA) +#define ECDH_VALIDATE(cond) MBEDTLS_INTERNAL_VALIDATE(cond) + +#if defined MBEDTLS_ECDH_GEN_PUBLIC_ALT +/* + * Generate public key (restartable version) + */ +static int ecdh_gen_public_restartable(mbedtls_ecp_group * grp, mbedtls_mpi * d, mbedtls_ecp_point * Q, + int (*f_rng)(void *, unsigned char *, size_t), void * p_rng, + mbedtls_ecp_restart_ctx * rs_ctx) +{ + int ret = 0; + size_t plen = grp->nbits / 8U; + uint8_t * q_buf = (uint8_t *) mbedtls_calloc(1, (plen * 2U) + 1); + uint8_t * d_buf = (uint8_t *) mbedtls_calloc(1, plen); + CryptoKey myPrivateKey; + CryptoKey myPublicKey; + ECDH_Config config = { 0 }; + ECDH_Handle handle = NULL; + ECDH_OperationGeneratePublicKey operationGeneratePublicKey; + ECDHCC26X2_HWAttrs hwAttrs = { 0 }; + ECDHCC26X2_Object object = { 0 }; + + if (NULL == q_buf || NULL == d_buf) + { + ret = MBEDTLS_ERR_ECP_ALLOC_FAILED; + goto cleanup; + } + + hwAttrs.intPriority = (1 << 5); + + config.object = (void *) &object; + config.hwAttrs = (void *) &hwAttrs; + + handle = ECDH_construct(&config, NULL); + + if (NULL == handle) + { + ret = MBEDTLS_ERR_ECP_HW_ACCEL_FAILED; + goto cleanup; + } + + CryptoKeyPlaintext_initKey(&myPrivateKey, d_buf, plen); + CryptoKeyPlaintext_initKey(&myPublicKey, q_buf, (plen * 2U) + 1); + + if (0 != f_rng(p_rng, d_buf, plen)) + { + ret = MBEDTLS_ERR_ECP_RANDOM_FAILED; + goto cleanup; + } + + ECDH_OperationGeneratePublicKey_init(&operationGeneratePublicKey); + operationGeneratePublicKey.myPrivateKey = &myPrivateKey; + operationGeneratePublicKey.myPublicKey = &myPublicKey; + + /* Load the ROM curve params */ + switch (grp->id) + { + case MBEDTLS_ECP_DP_SECP224R1: + operationGeneratePublicKey.curve = &ECCParams_NISTP224; + break; + + case MBEDTLS_ECP_DP_SECP256R1: + operationGeneratePublicKey.curve = &ECCParams_NISTP256; + break; + + case MBEDTLS_ECP_DP_SECP384R1: + operationGeneratePublicKey.curve = &ECCParams_NISTP384; + break; + + case MBEDTLS_ECP_DP_SECP521R1: + operationGeneratePublicKey.curve = &ECCParams_NISTP521; + break; + + case MBEDTLS_ECP_DP_BP256R1: + operationGeneratePublicKey.curve = &ECCParams_BrainpoolP256R1; + break; + + case MBEDTLS_ECP_DP_BP384R1: + operationGeneratePublicKey.curve = &ECCParams_BrainpoolP384R1; + break; + + case MBEDTLS_ECP_DP_BP512R1: + operationGeneratePublicKey.curve = &ECCParams_BrainpoolP512R1; + break; + + case MBEDTLS_ECP_DP_CURVE25519: + operationGeneratePublicKey.curve = &ECCParams_Curve25519; + break; + + default: + /* Possible extension to load an arbitrary curve */ + ret = MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE; + goto cleanup; + } + + if (ECDH_STATUS_SUCCESS == ECDH_generatePublicKey(handle, &operationGeneratePublicKey)) + { + ret = 0; + } + else + { + ret = MBEDTLS_ERR_ECP_BAD_INPUT_DATA; + } + + MBEDTLS_MPI_CHK(mbedtls_mpi_read_binary(d, d_buf, plen)); + MBEDTLS_MPI_CHK(mbedtls_mpi_read_binary(&(Q->X), q_buf + 1, plen)); + MBEDTLS_MPI_CHK(mbedtls_mpi_read_binary(&(Q->Y), q_buf + 1 + plen, plen)); + +cleanup: + mbedtls_free(q_buf); + mbedtls_free(d_buf); + + if (NULL != handle) + { + ECDH_close(handle); + } + + return (ret); +} + +/* + * Generate public key + */ +int mbedtls_ecdh_gen_public(mbedtls_ecp_group * grp, mbedtls_mpi * d, mbedtls_ecp_point * Q, + int (*f_rng)(void *, unsigned char *, size_t), void * p_rng) +{ + ECDH_VALIDATE_RET(grp != NULL); + ECDH_VALIDATE_RET(d != NULL); + ECDH_VALIDATE_RET(Q != NULL); + ECDH_VALIDATE_RET(f_rng != NULL); + return (ecdh_gen_public_restartable(grp, d, Q, f_rng, p_rng, NULL)); +} +#endif /* MBEDTLS_ECDH_GEN_PUBLIC_ALT */ + +#if defined MBEDTLS_ECDH_COMPUTE_SHARED_ALT +/* + * Compute shared secret (SEC1 3.3.1) + */ +static int ecdh_compute_shared_restartable(mbedtls_ecp_group * grp, mbedtls_mpi * z, const mbedtls_ecp_point * Q, + const mbedtls_mpi * d, int (*f_rng)(void *, unsigned char *, size_t), void * p_rng, + mbedtls_ecp_restart_ctx * rs_ctx) +{ + int ret = 0; + size_t plen = grp->nbits / 8U; + uint8_t * q_buf = (uint8_t *) mbedtls_calloc(1, (plen * 2U) + 1); + uint8_t * s_buf = (uint8_t *) mbedtls_calloc(1, (plen * 2U) + 1); + uint8_t * d_buf = (uint8_t *) mbedtls_calloc(1, plen); + CryptoKey myPrivateKey; + CryptoKey theirPublicKey; + CryptoKey sharedSecret; + ECDH_Config config = { 0 }; + ECDH_Handle handle = NULL; + ECDH_OperationComputeSharedSecret operationComputeSharedSecret; + ECDHCC26X2_HWAttrs hwAttrs = { 0 }; + ECDHCC26X2_Object object = { 0 }; + + if (NULL == q_buf || NULL == s_buf || NULL == d_buf) + { + ret = MBEDTLS_ERR_ECP_ALLOC_FAILED; + goto cleanup; + } + + hwAttrs.intPriority = (1 << 5); + + config.object = (void *) &object; + config.hwAttrs = (void *) &hwAttrs; + + handle = ECDH_construct(&config, NULL); + + if (NULL == handle) + { + ret = MBEDTLS_ERR_ECP_HW_ACCEL_FAILED; + goto cleanup; + } + + CryptoKeyPlaintext_initKey(&sharedSecret, s_buf, (2U * plen) + 1); + + /* The driver consumes network byte order. Copy this number and reverse them again. */ + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(d, d_buf, plen)); + CryptoKeyPlaintext_initKey(&myPrivateKey, d_buf, plen); + + /* The driver consumes an octet string in network byte order. Copy this + * point and reverse them again. */ + q_buf[0] = 0x04; + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(&(Q->X), q_buf + 1, plen)); + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(&(Q->Y), q_buf + 1 + plen, plen)); + CryptoKeyPlaintext_initKey(&theirPublicKey, q_buf, (2U * plen) + 1); + + ECDH_OperationComputeSharedSecret_init(&operationComputeSharedSecret); + operationComputeSharedSecret.myPrivateKey = &myPrivateKey; + operationComputeSharedSecret.theirPublicKey = &theirPublicKey; + operationComputeSharedSecret.sharedSecret = &sharedSecret; + + /* Load the ROM curve params */ + switch (grp->id) + { + case MBEDTLS_ECP_DP_SECP224R1: + operationComputeSharedSecret.curve = &ECCParams_NISTP224; + break; + + case MBEDTLS_ECP_DP_SECP256R1: + operationComputeSharedSecret.curve = &ECCParams_NISTP256; + break; + + case MBEDTLS_ECP_DP_SECP384R1: + operationComputeSharedSecret.curve = &ECCParams_NISTP384; + break; + + case MBEDTLS_ECP_DP_SECP521R1: + operationComputeSharedSecret.curve = &ECCParams_NISTP521; + break; + + case MBEDTLS_ECP_DP_BP256R1: + operationComputeSharedSecret.curve = &ECCParams_BrainpoolP256R1; + break; + + case MBEDTLS_ECP_DP_BP384R1: + operationComputeSharedSecret.curve = &ECCParams_BrainpoolP384R1; + break; + + case MBEDTLS_ECP_DP_BP512R1: + operationComputeSharedSecret.curve = &ECCParams_BrainpoolP512R1; + break; + + case MBEDTLS_ECP_DP_CURVE25519: + operationComputeSharedSecret.curve = &ECCParams_Curve25519; + break; + + default: + /* Possible extension to load an arbitrary curve */ + ret = MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE; + goto cleanup; + } + + if (ECDH_STATUS_SUCCESS == ECDH_computeSharedSecret(handle, &operationComputeSharedSecret)) + { + ret = 0; + } + else + { + ret = MBEDTLS_ERR_ECP_BAD_INPUT_DATA; + } + + MBEDTLS_MPI_CHK(mbedtls_mpi_read_binary(z, s_buf + 1, plen)); + +cleanup: + mbedtls_free(q_buf); + mbedtls_free(d_buf); + mbedtls_free(s_buf); + + if (NULL != handle) + { + ECDH_close(handle); + } + + return (ret); +} + +/* + * Compute shared secret (SEC1 3.3.1) + */ +int mbedtls_ecdh_compute_shared(mbedtls_ecp_group * grp, mbedtls_mpi * z, const mbedtls_ecp_point * Q, const mbedtls_mpi * d, + int (*f_rng)(void *, unsigned char *, size_t), void * p_rng) +{ + ECDH_VALIDATE_RET(grp != NULL); + ECDH_VALIDATE_RET(Q != NULL); + ECDH_VALIDATE_RET(d != NULL); + ECDH_VALIDATE_RET(z != NULL); + return (ecdh_compute_shared_restartable(grp, z, Q, d, f_rng, p_rng, NULL)); +} +#endif /* MBEDTLS_ECDH_COMPUTE_SHARED_ALT */ + +#endif /* MBEDTLS_ECDH_COMPUTE_SHARED_ALT || MBEDTLS_ECDH_GEN_PUBLIC_ALT */ diff --git a/src/platform/cc13x2_26x2/crypto/ecdsa_alt.c b/src/platform/cc13x2_26x2/crypto/ecdsa_alt.c new file mode 100755 index 00000000000000..18bb44fd1364e3 --- /dev/null +++ b/src/platform/cc13x2_26x2/crypto/ecdsa_alt.c @@ -0,0 +1,289 @@ +/* + * + * Copyright (c) 2021 Project CHIP Authors + * Copyright (c) 2021 Texas Instruments Incorporated + * + * 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. + */ + +#if !defined(MBEDTLS_CONFIG_FILE) +#include "mbedtls/config.h" +#else +#include MBEDTLS_CONFIG_FILE +#endif + +#if defined MBEDTLS_ECDSA_VERIFY_ALT || defined MBEDTLS_ECDSA_SIGN_ALT + +/* NOTE: The TI drivers consume these points and big numbers in network byte + * order. This is in contrast to the mbedtls_mpi structures which story + * these numbers and points in little endian byte order. This file uses + * the mpi functions to re-write the buffers into network byte order. + */ + +#include + +#include +#include +#include +#include + +#include "ti_drivers_config.h" + +#include +#include +#include + +#ifdef MBEDTLS_ECDSA_VERIFY_ALT + +/* + * Verify ECDSA signature of hashed message + */ +int mbedtls_ecdsa_verify(mbedtls_ecp_group * grp, const unsigned char * buf, size_t blen, const mbedtls_ecp_point * Q, + const mbedtls_mpi * r, const mbedtls_mpi * s) +{ + int ret = 0; + size_t plen = grp->nbits / 8U; + uint8_t * r_buf = (uint8_t *) mbedtls_calloc(1, plen); + uint8_t * s_buf = (uint8_t *) mbedtls_calloc(1, plen); + uint8_t * q_buf = (uint8_t *) mbedtls_calloc(1, (plen * 2U) + 1); + CryptoKey theirPublicKey; + ECDSA_Config config = { 0 }; + ECDSA_Handle handle = NULL; + ECDSA_OperationVerify operationVerify; + ECDSACC26X2_HWAttrs hwAttrs = { 0 }; + ECDSACC26X2_Object object = { 0 }; + + if (NULL == r_buf || NULL == s_buf || NULL == q_buf) + { + ret = MBEDTLS_ERR_ECP_ALLOC_FAILED; + goto cleanup; + } + + hwAttrs.intPriority = (1 << 5); + hwAttrs.trngIntPriority = (1 << 5); + + config.object = (void *) &object; + config.hwAttrs = (void *) &hwAttrs; + + handle = ECDSA_construct(&config, NULL); + + if (NULL == handle) + { + ret = MBEDTLS_ERR_ECP_HW_ACCEL_FAILED; + goto cleanup; + } + + /* The driver consumes an octet string in network byte order. Copy this + * point and reverse them again. */ + q_buf[0] = 0x04; + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(&(Q->X), q_buf + 1, plen)); + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(&(Q->Y), q_buf + 1 + plen, plen)); + CryptoKeyPlaintext_initKey(&theirPublicKey, q_buf, (2U * plen) + 1); + + /* The driver also consumes the r and s in network byte order. Copy these + * buffers and them reverse them again */ + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(r, r_buf, plen)); + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(s, s_buf, plen)); + + /* prepare the operation */ + ECDSA_OperationVerify_init(&operationVerify); + operationVerify.theirPublicKey = &theirPublicKey; + operationVerify.hash = buf; + operationVerify.r = r_buf; + operationVerify.s = s_buf; + + /* Load the ROM curve params */ + switch (grp->id) + { + case MBEDTLS_ECP_DP_SECP224R1: + operationVerify.curve = &ECCParams_NISTP224; + break; + + case MBEDTLS_ECP_DP_SECP256R1: + operationVerify.curve = &ECCParams_NISTP256; + break; + + case MBEDTLS_ECP_DP_SECP384R1: + operationVerify.curve = &ECCParams_NISTP384; + break; + + case MBEDTLS_ECP_DP_SECP521R1: + operationVerify.curve = &ECCParams_NISTP521; + break; + + case MBEDTLS_ECP_DP_BP256R1: + operationVerify.curve = &ECCParams_BrainpoolP256R1; + break; + + case MBEDTLS_ECP_DP_BP384R1: + operationVerify.curve = &ECCParams_BrainpoolP384R1; + break; + + case MBEDTLS_ECP_DP_BP512R1: + operationVerify.curve = &ECCParams_BrainpoolP512R1; + break; + + case MBEDTLS_ECP_DP_CURVE25519: + operationVerify.curve = &ECCParams_Curve25519; + break; + + default: + /* Possible extension to load an arbitrary curve */ + ret = MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE; + goto cleanup; + } + + if (ECDSA_STATUS_SUCCESS == ECDSA_verify(handle, &operationVerify)) + { + ret = 0; + } + else + { + ret = MBEDTLS_ERR_ECP_VERIFY_FAILED; + } + +cleanup: + mbedtls_free(r_buf); + mbedtls_free(s_buf); + mbedtls_free(q_buf); + + if (NULL != handle) + { + ECDSA_close(handle); + } + + return ret; +} + +#endif /* MBEDTLS_ECDSA_VERIFY_ALT */ + +#ifdef MBEDTLS_ECDSA_SIGN_ALT + +/* + * Compute ECDSA signature of a hashed message + */ +int mbedtls_ecdsa_sign(mbedtls_ecp_group * grp, mbedtls_mpi * r, mbedtls_mpi * s, const mbedtls_mpi * d, const unsigned char * buf, + size_t blen, int (*f_rng)(void *, unsigned char *, size_t), void * p_rng) +{ + int ret = 0; + size_t plen = grp->nbits / 8U; + uint8_t * r_buf = (uint8_t *) mbedtls_calloc(1, plen); + uint8_t * s_buf = (uint8_t *) mbedtls_calloc(1, plen); + uint8_t * d_buf = (uint8_t *) mbedtls_calloc(1, plen); + CryptoKey myPrivateKey; + ECDSA_Config config = { 0 }; + ECDSA_Handle handle = NULL; + ECDSA_OperationSign operationSign; + ECDSACC26X2_HWAttrs hwAttrs = { 0 }; + ECDSACC26X2_Object object = { 0 }; + + if (NULL == r_buf || NULL == s_buf || NULL == d_buf) + { + ret = MBEDTLS_ERR_ECP_ALLOC_FAILED; + goto cleanup; + } + + hwAttrs.intPriority = (1 << 5); + hwAttrs.trngIntPriority = (1 << 5); + + config.object = (void *) &object; + config.hwAttrs = (void *) &hwAttrs; + + handle = ECDSA_construct(&config, NULL); + + if (NULL == handle) + { + ret = MBEDTLS_ERR_ECP_HW_ACCEL_FAILED; + goto cleanup; + } + + /* The driver consumes numbers in network byte order */ + MBEDTLS_MPI_CHK(mbedtls_mpi_write_binary(d, d_buf, plen)); + + CryptoKeyPlaintext_initKey(&myPrivateKey, d_buf, plen); + ECDSA_OperationSign_init(&operationSign); + operationSign.myPrivateKey = &myPrivateKey; + operationSign.hash = buf; + operationSign.r = r_buf; + operationSign.s = s_buf; + + /* Load the ROM curve params */ + switch (grp->id) + { + case MBEDTLS_ECP_DP_SECP224R1: + operationSign.curve = &ECCParams_NISTP224; + break; + + case MBEDTLS_ECP_DP_SECP256R1: + operationSign.curve = &ECCParams_NISTP256; + break; + + case MBEDTLS_ECP_DP_SECP384R1: + operationSign.curve = &ECCParams_NISTP384; + break; + + case MBEDTLS_ECP_DP_SECP521R1: + operationSign.curve = &ECCParams_NISTP521; + break; + + case MBEDTLS_ECP_DP_BP256R1: + operationSign.curve = &ECCParams_BrainpoolP256R1; + break; + + case MBEDTLS_ECP_DP_BP384R1: + operationSign.curve = &ECCParams_BrainpoolP384R1; + break; + + case MBEDTLS_ECP_DP_BP512R1: + operationSign.curve = &ECCParams_BrainpoolP512R1; + break; + + case MBEDTLS_ECP_DP_CURVE25519: + operationSign.curve = &ECCParams_Curve25519; + break; + + default: + /* Possible extension to load an arbitrary curve */ + ret = MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE; + goto cleanup; + } + + if (ECDSA_STATUS_SUCCESS == ECDSA_sign(handle, &operationSign)) + { + ret = 0; + } + else + { + ret = MBEDTLS_ERR_ECP_VERIFY_FAILED; + } + + /* The driver produces r and s in network byte order. copy into mbedtls mpi + * format. This incurs an extra byte reversal when written to ASN1. */ + MBEDTLS_MPI_CHK(mbedtls_mpi_read_binary(r, r_buf, plen)); + MBEDTLS_MPI_CHK(mbedtls_mpi_read_binary(s, s_buf, plen)); + +cleanup: + mbedtls_free(r_buf); + mbedtls_free(s_buf); + mbedtls_free(d_buf); + + if (NULL != handle) + { + ECDSA_close(handle); + } + + return ret; +} +#endif /* MBEDTLS_ECDSA_SIGN_ALT */ + +#endif /* defined MBEDTLS_ECDSA_VERIFY_ALT || defined MBEDTLS_ECDSA_SIGN_ALT */ diff --git a/src/platform/cc13x2_26x2/crypto/sha256_alt.c b/src/platform/cc13x2_26x2/crypto/sha256_alt.c index 8dabab6f88dff3..0ef01f48fe34ba 100644 --- a/src/platform/cc13x2_26x2/crypto/sha256_alt.c +++ b/src/platform/cc13x2_26x2/crypto/sha256_alt.c @@ -136,6 +136,8 @@ void mbedtls_sha256_clone(mbedtls_sha256_context * dst, const mbedtls_sha256_con dst->object.bytesInBuffer = src->object.bytesInBuffer; dst->object.bytesProcessed = src->object.bytesProcessed; memcpy(dst->object.buffer, src->object.buffer, sizeof(dst->object.buffer)); + // copy the digest to support cloning after finalization has happened + memcpy(dst->object.digest, src->object.digest, sizeof(dst->object.digest)); } int mbedtls_sha256_finish_ret(mbedtls_sha256_context * ctx, unsigned char output[32]) diff --git a/third_party/ti_simplelink_sdk/BUILD.gn b/third_party/ti_simplelink_sdk/BUILD.gn index 73e9b3e965603e..f968324d78170f 100644 --- a/third_party/ti_simplelink_sdk/BUILD.gn +++ b/third_party/ti_simplelink_sdk/BUILD.gn @@ -48,6 +48,8 @@ mbedtls_target("mbedtls") { # Hardware acceleration sources = [ "${chip_root}/src/platform/cc13x2_26x2/crypto/aes_alt.c", + "${chip_root}/src/platform/cc13x2_26x2/crypto/ecdh_alt.c", + "${chip_root}/src/platform/cc13x2_26x2/crypto/ecdsa_alt.c", "${chip_root}/src/platform/cc13x2_26x2/crypto/ecjpake_alt.c", "${chip_root}/src/platform/cc13x2_26x2/crypto/sha256_alt.c", ] diff --git a/third_party/ti_simplelink_sdk/ti_simplelink_sdk.gni b/third_party/ti_simplelink_sdk/ti_simplelink_sdk.gni index aebec3c96940f2..538bae5ee02413 100644 --- a/third_party/ti_simplelink_sdk/ti_simplelink_sdk.gni +++ b/third_party/ti_simplelink_sdk/ti_simplelink_sdk.gni @@ -87,6 +87,8 @@ template("ti_sysconfig") { "gcc", "--board", "/ti/boards/" + ti_simplelink_board, + "--rtos", + "freertos", ] foreach(source, sources) { @@ -316,6 +318,8 @@ template("ti_simplelink_sdk") { "${ti_simplelink_sdk_root}/source/ti/", "${ti_simplelink_sdk_root}/source/ti/dmm/", "${ti_simplelink_sdk_root}/source/ti/dmm/apps/", + "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/thread/", + "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/thread/source/activity/", "${ti_simplelink_sdk_root}/source/ti/common/nv", "${ti_simplelink_sdk_root}/source/ti/devices/cc13x2x7_cc26x2x7", @@ -374,7 +378,7 @@ template("ti_simplelink_sdk") { if (ti_simplelink_device_family == "cc13x2x7_26x2x7") { libs += [ - "${ti_simplelink_sdk_root}/source/ti/dmm/library/freertos/gcc/bin/dmmlib_cc26x2.a", + "${ti_simplelink_sdk_root}/source/ti/dmm/lib/gcc/m4f/libdmmlib_freertos.a", "${ti_simplelink_sdk_root}/source/ti/ble5stack/libraries/cc26x2r1/OneLib.a", "${ti_simplelink_sdk_root}/source/ti/ble5stack/libraries/cc26x2r1/StackWrapper.a", "${ti_simplelink_sdk_root}/source/ti/ble5stack/libraries/cc26x2r1/ble_r2.symbols", @@ -391,8 +395,10 @@ template("ti_simplelink_sdk") { source_set("${sdk_target_name}_dmm") { defines = [] - include_dirs = [] - + include_dirs = [ + "${chip_root}/third_party/openthread/repo/include/", + "${chip_root}/src/", + ] configs -= [ "${build_root}/config/compiler:warnings_default" ] # removed for # -Wshadow @@ -409,7 +415,6 @@ template("ti_simplelink_sdk") { "${ti_simplelink_sdk_root}/source/ti/ble5stack/common/cc26xx/rcosc/rcosc_calibration.c", "${ti_simplelink_sdk_root}/source/ti/ble5stack/host/gatt_uuid.c", "${ti_simplelink_sdk_root}/source/ti/ble5stack/host/gattservapp_util.c", - "${ti_simplelink_sdk_root}/source/ti/ble5stack/icall/app/ble_user_config.c", "${ti_simplelink_sdk_root}/source/ti/ble5stack/icall/app/icall_api_lite.c", "${ti_simplelink_sdk_root}/source/ti/ble5stack/icall/src/icall_cc2650.c", "${ti_simplelink_sdk_root}/source/ti/ble5stack/icall/src/icall_user_config.c", @@ -418,11 +423,12 @@ template("ti_simplelink_sdk") { "${ti_simplelink_sdk_root}/source/ti/ble5stack/rom/agama_r1/rom_init.c", "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/ble_remote_display/stack/osal_icall_ble.c", "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/freertos/icall_FreeRTOS.c", - "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/freertos/itm/itm.c", "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/freertos/util.c", + "${ti_simplelink_sdk_root}/source/ti/dmm/apps/common/thread/source/activity/dmm_thread_activity.c", "${ti_simplelink_sdk_root}/source/ti/dmm/dmm_priority_ble_thread.c", # Move this into the SDK + "${chip_root}/src/platform/cc13x2_26x2/ble_user_config.c", "${chip_root}/src/platform/cc13x2_26x2/chipOBleProfile.c", ]