From caee5d674bb147d678b97a004f7377e255cdeac7 Mon Sep 17 00:00:00 2001 From: Boris Zbarsky Date: Thu, 3 Jun 2021 13:14:39 -0400 Subject: [PATCH] Add chip-tool support for octet string arguments with non-ASCII bytes in them. For an octet-string argument we support three formats: 1. hex: followed by hex representation of the bytes. 2. str: followed by just the chars, if they are all nicely printable 3. Just the bytes/chars, if they are all nicely printable. Format 1 is needed when the value contains non-printable chars, and especially nulls. Formats 2 and 3 are easier to use for the typical use case of Wi-Fi SSIDs and passwords. Formats 1 and 2 match chip-device-ctrl. --- .../chip-tool/commands/clusters/Commands.h | 87 ++++++++----------- .../chip-tool/commands/common/Command.cpp | 80 +++++++++++++++-- examples/chip-tool/commands/common/Command.h | 12 ++- .../commands/pairing/PairingCommand.cpp | 9 +- .../commands/pairing/PairingCommand.h | 7 +- examples/chip-tool/templates/commands.zapt | 8 +- src/app/zap-templates/README.md | 2 +- src/lib/support/BytesToHex.h | 17 ++++ 8 files changed, 148 insertions(+), 74 deletions(-) diff --git a/examples/chip-tool/commands/clusters/Commands.h b/examples/chip-tool/commands/clusters/Commands.h index 82d975cd7c5c5e..75582c71a9cce1 100644 --- a/examples/chip-tool/commands/clusters/Commands.h +++ b/examples/chip-tool/commands/clusters/Commands.h @@ -11515,9 +11515,8 @@ class NetworkCommissioningAddThreadNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.AddThreadNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mOperationalDataset), strlen(mOperationalDataset)), - mBreadcrumb, mTimeoutMs); + return cluster.AddThreadNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mOperationalDataset, mBreadcrumb, + mTimeoutMs); } private: @@ -11526,7 +11525,7 @@ class NetworkCommissioningAddThreadNetwork : public ModelCommand OnNetworkCommissioningClusterAddThreadNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mOperationalDataset; + chip::ByteSpan mOperationalDataset; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11557,9 +11556,8 @@ class NetworkCommissioningAddWiFiNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.AddWiFiNetwork( - onSuccessCallback->Cancel(), onFailureCallback->Cancel(), chip::ByteSpan(chip::Uint8::from_char(mSsid), strlen(mSsid)), - chip::ByteSpan(chip::Uint8::from_char(mCredentials), strlen(mCredentials)), mBreadcrumb, mTimeoutMs); + return cluster.AddWiFiNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mSsid, mCredentials, mBreadcrumb, + mTimeoutMs); } private: @@ -11568,8 +11566,8 @@ class NetworkCommissioningAddWiFiNetwork : public ModelCommand OnNetworkCommissioningClusterAddWiFiNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mSsid; - char * mCredentials; + chip::ByteSpan mSsid; + chip::ByteSpan mCredentials; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11599,8 +11597,7 @@ class NetworkCommissioningDisableNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.DisableNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mNetworkID), strlen(mNetworkID)), mBreadcrumb, + return cluster.DisableNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mNetworkID, mBreadcrumb, mTimeoutMs); } @@ -11610,7 +11607,7 @@ class NetworkCommissioningDisableNetwork : public ModelCommand OnNetworkCommissioningClusterDisableNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mNetworkID; + chip::ByteSpan mNetworkID; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11640,9 +11637,7 @@ class NetworkCommissioningEnableNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.EnableNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mNetworkID), strlen(mNetworkID)), mBreadcrumb, - mTimeoutMs); + return cluster.EnableNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mNetworkID, mBreadcrumb, mTimeoutMs); } private: @@ -11651,7 +11646,7 @@ class NetworkCommissioningEnableNetwork : public ModelCommand OnNetworkCommissioningClusterEnableNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mNetworkID; + chip::ByteSpan mNetworkID; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11715,9 +11710,7 @@ class NetworkCommissioningRemoveNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.RemoveNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mNetworkID), strlen(mNetworkID)), mBreadcrumb, - mTimeoutMs); + return cluster.RemoveNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mNetworkID, mBreadcrumb, mTimeoutMs); } private: @@ -11726,7 +11719,7 @@ class NetworkCommissioningRemoveNetwork : public ModelCommand OnNetworkCommissioningClusterRemoveNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mNetworkID; + chip::ByteSpan mNetworkID; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11756,8 +11749,7 @@ class NetworkCommissioningScanNetworks : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.ScanNetworks(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mSsid), strlen(mSsid)), mBreadcrumb, mTimeoutMs); + return cluster.ScanNetworks(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mSsid, mBreadcrumb, mTimeoutMs); } private: @@ -11766,7 +11758,7 @@ class NetworkCommissioningScanNetworks : public ModelCommand OnNetworkCommissioningClusterScanNetworksResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mSsid; + chip::ByteSpan mSsid; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11796,8 +11788,7 @@ class NetworkCommissioningUpdateThreadNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.UpdateThreadNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mOperationalDataset), strlen(mOperationalDataset)), + return cluster.UpdateThreadNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mOperationalDataset, mBreadcrumb, mTimeoutMs); } @@ -11807,7 +11798,7 @@ class NetworkCommissioningUpdateThreadNetwork : public ModelCommand OnNetworkCommissioningClusterUpdateThreadNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mOperationalDataset; + chip::ByteSpan mOperationalDataset; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -11838,9 +11829,8 @@ class NetworkCommissioningUpdateWiFiNetwork : public ModelCommand chip::Controller::NetworkCommissioningCluster cluster; cluster.Associate(device, endpointId); - return cluster.UpdateWiFiNetwork( - onSuccessCallback->Cancel(), onFailureCallback->Cancel(), chip::ByteSpan(chip::Uint8::from_char(mSsid), strlen(mSsid)), - chip::ByteSpan(chip::Uint8::from_char(mCredentials), strlen(mCredentials)), mBreadcrumb, mTimeoutMs); + return cluster.UpdateWiFiNetwork(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mSsid, mCredentials, mBreadcrumb, + mTimeoutMs); } private: @@ -11849,8 +11839,8 @@ class NetworkCommissioningUpdateWiFiNetwork : public ModelCommand OnNetworkCommissioningClusterUpdateWiFiNetworkResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mSsid; - char * mCredentials; + chip::ByteSpan mSsid; + chip::ByteSpan mCredentials; uint64_t mBreadcrumb; uint32_t mTimeoutMs; }; @@ -12206,10 +12196,8 @@ class OperationalCredentialsAddOpCert : public ModelCommand chip::Controller::OperationalCredentialsCluster cluster; cluster.Associate(device, endpointId); - return cluster.AddOpCert( - onSuccessCallback->Cancel(), onFailureCallback->Cancel(), chip::ByteSpan(chip::Uint8::from_char(mNoc), strlen(mNoc)), - chip::ByteSpan(chip::Uint8::from_char(mICACertificate), strlen(mICACertificate)), - chip::ByteSpan(chip::Uint8::from_char(mIPKValue), strlen(mIPKValue)), mCaseAdminNode, mAdminVendorId); + return cluster.AddOpCert(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mNoc, mICACertificate, mIPKValue, + mCaseAdminNode, mAdminVendorId); } private: @@ -12218,9 +12206,9 @@ class OperationalCredentialsAddOpCert : public ModelCommand OnOperationalCredentialsClusterOpCertResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mNoc; - char * mICACertificate; - char * mIPKValue; + chip::ByteSpan mNoc; + chip::ByteSpan mICACertificate; + chip::ByteSpan mIPKValue; chip::NodeId mCaseAdminNode; uint16_t mAdminVendorId; }; @@ -12248,8 +12236,7 @@ class OperationalCredentialsOpCSRRequest : public ModelCommand chip::Controller::OperationalCredentialsCluster cluster; cluster.Associate(device, endpointId); - return cluster.OpCSRRequest(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mCSRNonce), strlen(mCSRNonce))); + return cluster.OpCSRRequest(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mCSRNonce); } private: @@ -12258,7 +12245,7 @@ class OperationalCredentialsOpCSRRequest : public ModelCommand OnOperationalCredentialsClusterOpCSRResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mCSRNonce; + chip::ByteSpan mCSRNonce; }; /* @@ -15872,8 +15859,7 @@ class WriteTestClusterOctetString : public ModelCommand chip::Controller::TestClusterCluster cluster; cluster.Associate(device, endpointId); - return cluster.WriteAttributeOctetString(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mValue), strlen(mValue))); + return cluster.WriteAttributeOctetString(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mValue); } private: @@ -15881,7 +15867,7 @@ class WriteTestClusterOctetString : public ModelCommand new chip::Callback::Callback(OnDefaultSuccessResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mValue; + chip::ByteSpan mValue; }; /* @@ -16660,9 +16646,7 @@ class TrustedRootCertificatesAddTrustedRootCertificate : public ModelCommand chip::Controller::TrustedRootCertificatesCluster cluster; cluster.Associate(device, endpointId); - return cluster.AddTrustedRootCertificate( - onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mRootCertificate), strlen(mRootCertificate))); + return cluster.AddTrustedRootCertificate(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), mRootCertificate); } private: @@ -16670,7 +16654,7 @@ class TrustedRootCertificatesAddTrustedRootCertificate : public ModelCommand new chip::Callback::Callback(OnDefaultSuccessResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mRootCertificate; + chip::ByteSpan mRootCertificate; }; /* @@ -16696,9 +16680,8 @@ class TrustedRootCertificatesRemoveTrustedRootCertificate : public ModelCommand chip::Controller::TrustedRootCertificatesCluster cluster; cluster.Associate(device, endpointId); - return cluster.RemoveTrustedRootCertificate( - onSuccessCallback->Cancel(), onFailureCallback->Cancel(), - chip::ByteSpan(chip::Uint8::from_char(mTrustedRootIdentifier), strlen(mTrustedRootIdentifier))); + return cluster.RemoveTrustedRootCertificate(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), + mTrustedRootIdentifier); } private: @@ -16706,7 +16689,7 @@ class TrustedRootCertificatesRemoveTrustedRootCertificate : public ModelCommand new chip::Callback::Callback(OnDefaultSuccessResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - char * mTrustedRootIdentifier; + chip::ByteSpan mTrustedRootIdentifier; }; /* diff --git a/examples/chip-tool/commands/common/Command.cpp b/examples/chip-tool/commands/common/Command.cpp index f7eadccd37fd16..654552a7eec53e 100644 --- a/examples/chip-tool/commands/common/Command.cpp +++ b/examples/chip-tool/commands/common/Command.cpp @@ -23,7 +23,11 @@ #include #include +#include +#include +#include #include +#include #include bool Command::InitArguments(int argc, char ** argv) @@ -78,7 +82,7 @@ static bool ParseAddressWithInterface(const char * addressString, Command::Addre return true; } -bool Command::InitArgument(size_t argIndex, const char * argValue) +bool Command::InitArgument(size_t argIndex, char * argValue) { bool isValidArgument = false; @@ -91,10 +95,65 @@ bool Command::InitArgument(size_t argIndex, const char * argValue) break; } - case ArgumentType::String: { + case ArgumentType::CharString: { const char ** value = reinterpret_cast(arg.value); - *value = const_cast(argValue); - isValidArgument = (strcmp(argValue, *value) == 0); + *value = argValue; + isValidArgument = true; + break; + } + + case ArgumentType::OctetString: { + auto * value = static_cast(arg.value); + // We support two ways to pass an octet string argument. If it happens + // to be all-ASCII, you can just pass it in. Otherwise you can pass in + // 0x followed by the hex-encoded bytes. + size_t argLen = strlen(argValue); + static constexpr char hexPrefix[] = "hex:"; + constexpr size_t prefixLen = ArraySize(hexPrefix) - 1; // Don't count the null + if (strncmp(argValue, hexPrefix, prefixLen) == 0) + { + // Hex-encoded. Decode it into a temporary buffer first, so if we + // run into errors we can do correct "argument is not valid" logging + // that actually shows the value that was passed in. After we + // determine it's valid, modify the passed-in value to hold the + // right bytes, so we don't need to worry about allocating storage + // for this somewhere else. This works because the hex + // representation is always longer than the octet string it encodes, + // so we have enough space in argValue for the decoded version. + auto buffer = static_cast(chip::Platform::MemoryCalloc(argLen, 1)); // Bigger than needed, but it's fine. + if (!buffer) { + isValidArgument = false; + break; + } + + size_t octetCount = chip::Encoding::HexToBytes(argValue + prefixLen, argLen - prefixLen, buffer, argLen); + if (octetCount == 0) + { + isValidArgument = false; + } + else + { + memcpy(argValue, buffer, octetCount); + *value = chip::ByteSpan(chip::Uint8::from_char(argValue), octetCount); + isValidArgument = true; + } + + chip::Platform::MemoryFree(buffer); + } + else + { + // Just ASCII. Check for the "str:" prefix. + static constexpr char strPrefix[] = "str:"; + constexpr size_t strPrefixLen = ArraySize(strPrefix) - 1; // Don't count the null + if (strncmp(argValue, strPrefix, strPrefixLen) == 0) + { + // Skip the prefix + argValue += strPrefixLen; + argLen -= strPrefixLen; + } + *value = chip::ByteSpan(chip::Uint8::from_char(argValue), argLen); + isValidArgument = true; + } break; } @@ -237,7 +296,18 @@ size_t Command::AddArgument(const char * name, const char * value) size_t Command::AddArgument(const char * name, char ** value) { Argument arg; - arg.type = ArgumentType::String; + arg.type = ArgumentType::CharString; + arg.name = name; + arg.value = reinterpret_cast(value); + + mArgs.emplace_back(arg); + return mArgs.size(); +} + +size_t Command::AddArgument(const char * name, chip::ByteSpan * value) +{ + Argument arg; + arg.type = ArgumentType::OctetString; arg.name = name; arg.value = reinterpret_cast(value); diff --git a/examples/chip-tool/commands/common/Command.h b/examples/chip-tool/commands/common/Command.h index 8fa8e87b97a0b2..bcb09c30112267 100644 --- a/examples/chip-tool/commands/common/Command.h +++ b/examples/chip-tool/commands/common/Command.h @@ -20,6 +20,7 @@ #include #include +#include #include #include @@ -56,7 +57,8 @@ enum ArgumentType Number_int16, Number_int32, Number_int64, - String, + CharString, + OctetString, Attribute, Address }; @@ -100,13 +102,17 @@ class Command size_t AddArgument(const char * name, const char * value); /** * @brief - * Add a string command argument + * Add a char string command argument * * @param name The name that will be displayed in the command help * @param value A pointer to a `char *` where the argv value will be stored * @returns The number of arguments currently added to the command */ size_t AddArgument(const char * name, char ** value); + /** + * Add an octet string command argument + */ + size_t AddArgument(const char * name, chip::ByteSpan * value); size_t AddArgument(const char * name, AddressWithInterface * out); size_t AddArgument(const char * name, int64_t min, uint64_t max, int8_t * out) { @@ -154,7 +160,7 @@ class Command void WaitForResponse(uint16_t duration); private: - bool InitArgument(size_t argIndex, const char * argValue); + bool InitArgument(size_t argIndex, char * argValue); size_t AddArgument(const char * name, int64_t min, uint64_t max, void * out, ArgumentType type); size_t AddArgument(const char * name, int64_t min, uint64_t max, void * out); diff --git a/examples/chip-tool/commands/pairing/PairingCommand.cpp b/examples/chip-tool/commands/pairing/PairingCommand.cpp index 3d9f5a01b0a27e..ba12c5d0e88762 100644 --- a/examples/chip-tool/commands/pairing/PairingCommand.cpp +++ b/examples/chip-tool/commands/pairing/PairingCommand.cpp @@ -209,19 +209,16 @@ CHIP_ERROR PairingCommand::AddThreadNetwork() { Callback::Cancelable * successCallback = mOnAddThreadNetworkCallback->Cancel(); Callback::Cancelable * failureCallback = mOnFailureCallback->Cancel(); - ByteSpan operationalDataset = ByteSpan(Uint8::from_char(mOperationalDataset), strlen(mOperationalDataset)); - return mCluster.AddThreadNetwork(successCallback, failureCallback, operationalDataset, kBreadcrumb, kTimeoutMs); + return mCluster.AddThreadNetwork(successCallback, failureCallback, mOperationalDataset, kBreadcrumb, kTimeoutMs); } CHIP_ERROR PairingCommand::AddWiFiNetwork() { Callback::Cancelable * successCallback = mOnAddWiFiNetworkCallback->Cancel(); Callback::Cancelable * failureCallback = mOnFailureCallback->Cancel(); - ByteSpan ssid = ByteSpan(Uint8::from_char(mSSID), strlen(mSSID)); - ByteSpan credentials = ByteSpan(Uint8::from_char(mPassword), strlen(mPassword)); - return mCluster.AddWiFiNetwork(successCallback, failureCallback, ssid, credentials, kBreadcrumb, kTimeoutMs); + return mCluster.AddWiFiNetwork(successCallback, failureCallback, mSSID, mPassword, kBreadcrumb, kTimeoutMs); } CHIP_ERROR PairingCommand::EnableNetwork() @@ -232,7 +229,7 @@ CHIP_ERROR PairingCommand::EnableNetwork() ByteSpan networkId; if (mNetworkType == PairingNetworkType::WiFi) { - networkId = ByteSpan(Uint8::from_char(mSSID), strlen(mSSID)); + networkId = mSSID; } else { diff --git a/examples/chip-tool/commands/pairing/PairingCommand.h b/examples/chip-tool/commands/pairing/PairingCommand.h index d5e122263ab211..2d0b7abbbf1572 100644 --- a/examples/chip-tool/commands/pairing/PairingCommand.h +++ b/examples/chip-tool/commands/pairing/PairingCommand.h @@ -24,6 +24,7 @@ #include "gen/CHIPClusters.h" #include +#include enum class PairingMode { @@ -134,9 +135,9 @@ class PairingCommand : public Command, uint64_t mFabricId; uint16_t mDiscriminator; uint32_t mSetupPINCode; - char * mOperationalDataset; - char * mSSID; - char * mPassword; + chip::ByteSpan mOperationalDataset; + chip::ByteSpan mSSID; + chip::ByteSpan mPassword; chip::Callback::Callback * mOnAddThreadNetworkCallback; chip::Callback::Callback * mOnAddWiFiNetworkCallback; diff --git a/examples/chip-tool/templates/commands.zapt b/examples/chip-tool/templates/commands.zapt index a2c116735f2c88..096f716b150732 100644 --- a/examples/chip-tool/templates/commands.zapt +++ b/examples/chip-tool/templates/commands.zapt @@ -205,7 +205,7 @@ public: chip::Controller::{{asCamelCased parent.name false}}Cluster cluster; cluster.Associate(device, endpointId); - return cluster.{{asCamelCased name false}}(onSuccessCallback->Cancel(), onFailureCallback->Cancel(){{#chip_server_cluster_command_arguments}}, {{#if (isString type)}} chip::ByteSpan(chip::Uint8::from_char(m{{asCamelCased label false}}), strlen(m{{asCamelCased label false}})){{else}}m{{asCamelCased label false}}{{/if}}{{/chip_server_cluster_command_arguments}}); + return cluster.{{asCamelCased name false}}(onSuccessCallback->Cancel(), onFailureCallback->Cancel(){{#chip_server_cluster_command_arguments}}, {{#if (isCharString type)}} chip::ByteSpan(chip::Uint8::from_char(m{{asCamelCased label false}}), strlen(m{{asCamelCased label false}})){{else}}m{{asCamelCased label false}}{{/if}}{{/chip_server_cluster_command_arguments}}); } private: @@ -216,7 +216,7 @@ private: {{/if}} chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); {{#chip_server_cluster_command_arguments}} - {{#if (isString type)}} + {{#if (isCharString type)}} char * m{{asCamelCased label false}}; {{else}} {{chipType}} m{{asCamelCased label false}}; @@ -321,13 +321,13 @@ public: chip::Controller::{{asCamelCased parent.name false}}Cluster cluster; cluster.Associate(device, endpointId); - return cluster.WriteAttribute{{asCamelCased name false}}(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), {{#if (isString type)}} chip::ByteSpan(chip::Uint8::from_char(mValue), strlen(mValue)){{else}}mValue{{/if}}); + return cluster.WriteAttribute{{asCamelCased name false}}(onSuccessCallback->Cancel(), onFailureCallback->Cancel(), {{#if (isCharString type)}} chip::ByteSpan(chip::Uint8::from_char(mValue), strlen(mValue)){{else}}mValue{{/if}}); } private: chip::Callback::Callback * onSuccessCallback = new chip::Callback::Callback(OnDefaultSuccessResponse, this); chip::Callback::Callback * onFailureCallback = new chip::Callback::Callback(OnDefaultFailureResponse, this); - {{#if (isString type)}} + {{#if (isCharString type)}} char * mValue; {{else}} {{chipType}} mValue; diff --git a/src/app/zap-templates/README.md b/src/app/zap-templates/README.md index db73ddb027f8c3..f53b615e543ea9 100644 --- a/src/app/zap-templates/README.md +++ b/src/app/zap-templates/README.md @@ -61,7 +61,7 @@ To generate the application additional files, the command is: For example, to generate `chip-tool` additional files the command is: ``` -./scripts/tools/zap/generate.py examples/chip-tool/chip-tool.zap -t examples/chip-tool/templates/templates.json +./scripts/tools/zap/generate.py src/controller/data_model/controller-clusters.zap -t examples/chip-tool/templates/templates.json ``` For more information please see the documentation under `docs/` in diff --git a/src/lib/support/BytesToHex.h b/src/lib/support/BytesToHex.h index b4e0954a8891b6..38b6bb12484ad8 100644 --- a/src/lib/support/BytesToHex.h +++ b/src/lib/support/BytesToHex.h @@ -96,6 +96,23 @@ inline CHIP_ERROR BytesToLowercaseHexString(const uint8_t * src_bytes, size_t sr return BytesToHex(src_bytes, src_size, dest_hex_str, dest_size_max, HexFlags::kNullTerminate); } +/** + * Convert a buffer of hexadecimal characters to bytes. Supports both lowercase + * and uppercase (or a mix of cases) hexadecimal characters. Supported input is + * [0-9a-fA-F]. + * + * @param srcHex a pointer to the character buffer to convert. It is not + * assumed to be null-terminated. + * @param srcLen the length of the character buffer to convert. + * @param destBytes the buffer to fill with the decoded bytes. + * @param destMaxLen the total size of the buffer to be filled. + * + * @return 0 on errors: + * - destMaxLen not big enough. + * - srcLen not even. + * - Some character not in [0-9a-fA-F] is present in srcHex. + * number of bytes actually decoded from the string on success. + */ size_t HexToBytes(const char * srcHex, const size_t srcLen, uint8_t * destBytes, size_t destMaxLen); } // namespace Encoding