diff --git a/config/esp32/components/chip/CMakeLists.txt b/config/esp32/components/chip/CMakeLists.txt index e405201395f065..2a53fbde064d25 100644 --- a/config/esp32/components/chip/CMakeLists.txt +++ b/config/esp32/components/chip/CMakeLists.txt @@ -107,6 +107,13 @@ if (CONFIG_ENABLE_OTA_REQUESTOR) chip_gn_arg_append("chip_enable_ota_requestor" "true") endif() +if (CONFIG_ENABLE_EXTENDED_DATA) + chip_gn_arg_append("chip_enable_additional_data_advertising" "true") + if (CONFIG_ENABLE_ROTATING_DEVICE_ID) + chip_gn_arg_append("chip_enable_rotating_device_id" "true") + endif() +endif() + set(args_gn_input "${CMAKE_CURRENT_BINARY_DIR}/args.gn.in") file(GENERATE OUTPUT "${args_gn_input}" CONTENT "${chip_gn_args}") diff --git a/config/esp32/components/chip/Kconfig b/config/esp32/components/chip/Kconfig index b7837b42cc9696..3b443fe4452bc1 100644 --- a/config/esp32/components/chip/Kconfig +++ b/config/esp32/components/chip/Kconfig @@ -706,6 +706,16 @@ menu "CHIP Device Layer" Setting this to y will cause the commissioner to send commissioning commands to the various clusters after establishing a PASE session. + config ENABLE_EXTENDED_DATA + depends on ENABLE_CHIPOBLE + bool "Include extended data during commissioning" + default n + + config ENABLE_ROTATING_DEVICE_ID + depends on ENABLE_EXTENDED_DATA + bool "Enable Rotating Device Identifier Support" + default n + endmenu diff --git a/src/app/server/Dnssd.cpp b/src/app/server/Dnssd.cpp index 306faf23de15ff..3ce263b7cf3afc 100644 --- a/src/app/server/Dnssd.cpp +++ b/src/app/server/Dnssd.cpp @@ -338,7 +338,7 @@ CHIP_ERROR DnssdServer::Advertise(bool commissionableNode, chip::Dnssd::Commissi #if CHIP_ENABLE_ROTATING_DEVICE_ID char rotatingDeviceIdHexBuffer[RotatingDeviceId::kHexMaxLength]; ReturnErrorOnFailure(GenerateRotatingDeviceId(rotatingDeviceIdHexBuffer, ArraySize(rotatingDeviceIdHexBuffer))); - advertiseParameters.SetRotatingId(chip::Optional::Value(rotatingDeviceIdHexBuffer)); + advertiseParameters.SetRotatingDeviceId(chip::Optional::Value(rotatingDeviceIdHexBuffer)); #endif advertiseParameters.SetMRPConfig(gDefaultMRPConfig).SetTcpSupported(Optional(INET_CONFIG_ENABLE_TCP_ENDPOINT)); diff --git a/src/ble/CHIPBleServiceData.h b/src/ble/CHIPBleServiceData.h index cd4d84e8b11df9..8a0f2f660d8499 100644 --- a/src/ble/CHIPBleServiceData.h +++ b/src/ble/CHIPBleServiceData.h @@ -46,11 +46,17 @@ enum chipBLEServiceDataType struct ChipBLEDeviceIdentificationInfo { constexpr static uint16_t kDiscriminatorMask = 0xfff; +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + constexpr static uint8_t kAdditionalDataFlagMask = 0x1; +#endif uint8_t OpCode; uint8_t DeviceDiscriminator[2]; uint8_t DeviceVendorId[2]; uint8_t DeviceProductId[2]; +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + uint8_t AdditionalDataFlag; +#endif void Init() { memset(this, 0, sizeof(*this)); } @@ -74,6 +80,22 @@ struct ChipBLEDeviceIdentificationInfo deviceDiscriminator |= static_cast(DeviceDiscriminator[1] << 8u & ~kDiscriminatorMask); chip::Encoding::LittleEndian::Put16(DeviceDiscriminator, deviceDiscriminator); } + +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + uint8_t GetAdditionalDataFlag() const { return (AdditionalDataFlag & kAdditionalDataFlagMask); } + + void SetAdditionalDataFlag(bool flag) + { + if (flag) + { + AdditionalDataFlag |= kAdditionalDataFlagMask; + } + else + { + AdditionalDataFlag &= static_cast(~kAdditionalDataFlagMask); + } + } +#endif } __attribute__((packed)); } /* namespace Ble */ diff --git a/src/include/platform/internal/GenericConfigurationManagerImpl.cpp b/src/include/platform/internal/GenericConfigurationManagerImpl.cpp index dc1fe8fb799b27..d8ef8563c3831a 100644 --- a/src/include/platform/internal/GenericConfigurationManagerImpl.cpp +++ b/src/include/platform/internal/GenericConfigurationManagerImpl.cpp @@ -484,6 +484,10 @@ GenericConfigurationManagerImpl::GetBLEDeviceIdentificationInfo(Ble SuccessOrExit(err); deviceIdInfo.SetDeviceDiscriminator(discriminator); +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + deviceIdInfo.SetAdditionalDataFlag(true); +#endif + exit: return err; } diff --git a/src/platform/BUILD.gn b/src/platform/BUILD.gn index cd12ba5da998df..b498da32bb69c2 100644 --- a/src/platform/BUILD.gn +++ b/src/platform/BUILD.gn @@ -54,7 +54,7 @@ if (chip_device_platform != "none") { chip_bypass_rendezvous = false # Enable including the additional data in the advertisement packets - chip_enable_additional_data_advertising = true + chip_enable_additional_data_advertising = false # Enable adding optional rotating device id to the additional data. chip_enable_rotating_device_id = false diff --git a/src/platform/ESP32/BLEManagerImpl.h b/src/platform/ESP32/BLEManagerImpl.h index 624da05ea2d4ce..62ba2b79ba9097 100644 --- a/src/platform/ESP32/BLEManagerImpl.h +++ b/src/platform/ESP32/BLEManagerImpl.h @@ -189,6 +189,9 @@ class BLEManagerImpl final : public BLEManager, uint16_t mServiceAttrHandle; uint16_t mRXCharAttrHandle; uint16_t mTXCharAttrHandle; +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + uint16_t mC3CharAttrHandle; +#endif uint16_t mTXCharCCCDAttrHandle; BitFlags mFlags; char mDeviceName[kMaxDeviceNameLength + 1]; @@ -246,6 +249,11 @@ class BLEManagerImpl final : public BLEManager, static int ble_svr_gap_event(struct ble_gap_event * event, void * arg); static int gatt_svr_chr_access(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt * ctxt, void * arg); +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + static int gatt_svr_chr_access_additional_data(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt * ctxt, + void * arg); + void HandleC3CharRead(struct ble_gatt_char_context * param); +#endif /* CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING */ #endif static void DriveBLEState(intptr_t arg); diff --git a/src/platform/ESP32/BUILD.gn b/src/platform/ESP32/BUILD.gn index e06792500a7de4..0b1528a074967e 100644 --- a/src/platform/ESP32/BUILD.gn +++ b/src/platform/ESP32/BUILD.gn @@ -48,7 +48,10 @@ static_library("ESP32") { "nimble/BLEManagerImpl.cpp", ] - deps = [ "${chip_root}/src/lib/dnssd:platform_header" ] + deps = [ + "${chip_root}/src/lib/dnssd:platform_header", + "${chip_root}/src/setup_payload", + ] public_deps = [ "${chip_root}/src/crypto", diff --git a/src/platform/ESP32/nimble/BLEManagerImpl.cpp b/src/platform/ESP32/nimble/BLEManagerImpl.cpp index a59ed79a80add9..bb104f19cd83b1 100644 --- a/src/platform/ESP32/nimble/BLEManagerImpl.cpp +++ b/src/platform/ESP32/nimble/BLEManagerImpl.cpp @@ -34,6 +34,7 @@ #include #include #include +#include #include #include "esp_log.h" @@ -90,6 +91,17 @@ const ble_uuid128_t UUID_CHIPoBLEChar_TX = { { BLE_UUID_TYPE_128 }, { 0x12, 0x9D, 0x9F, 0x42, 0x9C, 0x4F, 0x9F, 0x95, 0x59, 0x45, 0x3D, 0x26, 0xF5, 0x2E, 0xEE, 0x18 } }; +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING +const ble_uuid128_t UUID128_CHIPoBLEChar_C3 = { + BLE_UUID_TYPE_128, { 0x04, 0x8F, 0x21, 0x83, 0x8A, 0x74, 0x7D, 0xB8, 0xF2, 0x45, 0x72, 0x87, 0x38, 0x02, 0x63, 0x64 } +}; +const ChipBleUUID chipUUID_CHIPoBLEChar_C3 = { { 0x64, 0x63, 0x02, 0x38, 0x87, 0x72, 0x45, 0xF2, 0xB8, 0x7D, 0x74, 0x8A, 0x83, 0x21, + 0x8F, 0x04 } }; +const ble_uuid128_t UUID_CHIPoBLEChar_C3 = { + { BLE_UUID_TYPE_128 }, { 0x04, 0x8F, 0x21, 0x83, 0x8A, 0x74, 0x7D, 0xB8, 0xF2, 0x45, 0x72, 0x87, 0x38, 0x02, 0x63, 0x64 } +}; +#endif + SemaphoreHandle_t semaphoreHandle = NULL; } // unnamed namespace @@ -115,6 +127,14 @@ const struct ble_gatt_svc_def BLEManagerImpl::CHIPoBLEGATTAttrs[] = { .flags = BLE_GATT_CHR_F_READ | BLE_GATT_CHR_F_NOTIFY, .val_handle = &sInstance.mTXCharCCCDAttrHandle, }, +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + { + .uuid = (ble_uuid_t *) (&UUID_CHIPoBLEChar_C3), + .access_cb = gatt_svr_chr_access_additional_data, + .flags = BLE_GATT_CHR_F_READ, + .val_handle = &sInstance.mC3CharAttrHandle, + }, +#endif { 0, /* No more characteristics in this service */ }, @@ -133,7 +153,10 @@ CHIP_ERROR BLEManagerImpl::_Init() err = BleLayer::Init(this, this, &DeviceLayer::SystemLayer()); SuccessOrExit(err); - mRXCharAttrHandle = 0; + mRXCharAttrHandle = 0; +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING + mC3CharAttrHandle = 0; +#endif mTXCharCCCDAttrHandle = 0; mFlags.ClearAll().Set(Flags::kAdvertisingEnabled, CHIP_DEVICE_CONFIG_CHIPOBLE_ENABLE_ADVERTISING_AUTOSTART); mFlags.Set(Flags::kFastAdvertisingEnabled, true); @@ -683,6 +706,11 @@ CHIP_ERROR BLEManagerImpl::ConfigureAdvertisingData(void) uint8_t advData[MAX_ADV_DATA_LEN]; uint8_t index = 0; + constexpr uint8_t kServiceDataLenSize = 1; + constexpr uint8_t kServiceDataUUIDSize = 2; + + chip::Ble::ChipBLEDeviceIdentificationInfo deviceIdInfo; + // If a custom device name has not been specified, generate a CHIP-standard name based on the // bottom digits of the Chip device id. uint16_t discriminator; @@ -706,12 +734,11 @@ CHIP_ERROR BLEManagerImpl::ConfigureAdvertisingData(void) advData[index++] = 0x02; // length advData[index++] = CHIP_ADV_DATA_TYPE_FLAGS; // AD type : flags advData[index++] = CHIP_ADV_DATA_FLAGS; // AD value - advData[index++] = 0x0A; // length + advData[index++] = kServiceDataLenSize + kServiceDataUUIDSize + sizeof(deviceIdInfo); // length advData[index++] = CHIP_ADV_DATA_TYPE_SERVICE_DATA; // AD type: (Service Data - 16-bit UUID) advData[index++] = static_cast(ShortUUID_CHIPoBLEService.value & 0xFF); // AD value advData[index++] = static_cast((ShortUUID_CHIPoBLEService.value >> 8) & 0xFF); // AD value - chip::Ble::ChipBLEDeviceIdentificationInfo deviceIdInfo; err = ConfigurationMgr().GetBLEDeviceIdentificationInfo(deviceIdInfo); if (err != CHIP_NO_ERROR) { @@ -1046,6 +1073,69 @@ int BLEManagerImpl::ble_svr_gap_event(struct ble_gap_event * event, void * arg) return err.AsInteger(); } +#if CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING +void BLEManagerImpl::HandleC3CharRead(struct ble_gatt_char_context * param) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + chip::System::PacketBufferHandle bufferHandle; + + char serialNumber[ConfigurationManager::kMaxSerialNumberLength + 1]; + uint16_t lifetimeCounter = 0; + BitFlags additionalDataFields; + +#if CHIP_ENABLE_ROTATING_DEVICE_ID + err = ConfigurationMgr().GetSerialNumber(serialNumber, sizeof(serialNumber)); + SuccessOrExit(err); + err = ConfigurationMgr().GetLifetimeCounter(lifetimeCounter); + SuccessOrExit(err); + + additionalDataFields.Set(AdditionalDataFields::RotatingDeviceId); +#endif /* CHIP_ENABLE_ROTATING_DEVICE_ID */ + + err = AdditionalDataPayloadGenerator().generateAdditionalDataPayload(lifetimeCounter, serialNumber, strlen(serialNumber), + bufferHandle, additionalDataFields); + SuccessOrExit(err); + + os_mbuf_append(param->ctxt->om, bufferHandle->Start(), bufferHandle->DataLength()); + +exit: + if (err != CHIP_NO_ERROR) + { + ChipLogError(DeviceLayer, "Failed to generate TLV encoded Additional Data (%s)", __func__); + } + return; +} + +int BLEManagerImpl::gatt_svr_chr_access_additional_data(uint16_t conn_handle, uint16_t attr_handle, + struct ble_gatt_access_ctxt * ctxt, void * arg) +{ + struct ble_gatt_char_context param; + int err = 0; + + memset(¶m, 0, sizeof(struct ble_gatt_char_context)); + + switch (ctxt->op) + { + case BLE_GATT_ACCESS_OP_READ_CHR: + + param.conn_handle = conn_handle; + param.attr_handle = attr_handle; + param.ctxt = ctxt; + param.arg = arg; + sInstance.HandleC3CharRead(¶m); + break; + + default: + err = BLE_ATT_ERR_UNLIKELY; + break; + } + + PlatformMgr().ScheduleWork(DriveBLEState, 0); + + return err; +} +#endif /* CHIP_ENABLE_ADDITIONAL_DATA_ADVERTISING */ + int BLEManagerImpl::gatt_svr_chr_access(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt * ctxt, void * arg) { struct ble_gatt_char_context param; diff --git a/src/platform/Linux/bluez/Helper.cpp b/src/platform/Linux/bluez/Helper.cpp index 4da5da3021be03..2548210f9a8c05 100644 --- a/src/platform/Linux/bluez/Helper.cpp +++ b/src/platform/Linux/bluez/Helper.cpp @@ -47,6 +47,7 @@ * @file * Provides Bluez dbus implementatioon for BLE */ +#include #include #include