Skip to content

Commit cb9f02f

Browse files
shubhamdprestyled-commits
authored andcommitted
[ESP32] Fix few attributes with fixed quality in DeviceInfoProvider (#32893)
* [ESP32] Fix few attributes with fixed quality in DeviceInfoProvider Fixed labels, supported locales, supported calendar types were being read from the nvs(flash) and during OTA its a hassle if one wants to upgrade these values. Added few APIs to set the data for these attributes in ESP32DeviceInfoProvider. * Restyled by clang-format * Restyled by prettier-markdown * fix the lint errors * Add back the original Device info provider which reads from the nvs Add StaticESP32DeviceInfoProvider along with APIs to set data Remove changes from example and add a guide along with usage --------- Co-authored-by: Restyled.io <commits@restyled.io>
1 parent 28ea1c4 commit cb9f02f

7 files changed

+346
-129
lines changed

docs/guides/esp32/README.md

+1
Original file line numberDiff line numberDiff line change
@@ -18,3 +18,4 @@ example on ESP32 series of SoCs
1818
- [Matter OTA](ota.md)
1919
- [Generating and Using ESP Secure Cert Partition](secure_cert_partition.md)
2020
- [BLE Settings](ble_settings.md)
21+
- [Providers](providers.md)

docs/guides/esp32/factory_data.md

+3-7
Original file line numberDiff line numberDiff line change
@@ -30,13 +30,9 @@ Following data can be added to the manufacturing partition using
3030
- Serial Number
3131
- Unique identifier
3232

33-
- Device information
34-
- Fixed Labels
35-
- Supported locales
36-
- Supported calendar types
37-
- Supported modes
38-
- Note: As per spec at max size of label should be 64 and `\0` will be
39-
added at the end.
33+
- Supported modes
34+
- Note: As per spec at max size of label should be 64 and `\0` will be
35+
added at the end.
4036

4137
### Configuration Options
4238

docs/guides/esp32/providers.md

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
## Providers Implemented for ESP32 Platform
2+
3+
The ESP32 platform has implemented several providers that can be used with data
4+
stored in the factory or by setting fixed data.
5+
6+
Below are the providers that have been implemented:
7+
8+
- [Commissionable Data Provider](https://github.com/project-chip/connectedhomeip/blob/master/src/platform/ESP32/ESP32FactoryDataProvider.h#L47)
9+
This provider reads the discriminator and setup pincode related parameters
10+
from the factory partition.
11+
- [Device Attestation Credentials Provider](https://github.com/project-chip/connectedhomeip/blob/master/src/platform/ESP32/ESP32FactoryDataProvider.h#L56)
12+
This provider manages the attestation data.
13+
- [Device Instance Info Provider](https://github.com/project-chip/connectedhomeip/blob/master/src/platform/ESP32/ESP32FactoryDataProvider.h#L86)
14+
This provider reads basic device information from the factory partition.
15+
- [Device Info Provider](https://github.com/project-chip/connectedhomeip/blob/master/src/platform/ESP32/ESP32DeviceInfoProvider.h#L31)
16+
This provider provides fixed labels, supported calendar types, and supported
17+
locales from the factory partition.
18+
- [Supported Modes](https://github.com/project-chip/connectedhomeip/blob/master/examples/platform/esp32/mode-support/static-supported-modes-manager.h#L28)
19+
This provider offers the supported modes for the mode-select cluster.
20+
21+
More information can be found in the [factory data guide](factory_data.md).
22+
23+
### Device Info Provider
24+
25+
Currently, there are two implementations for this provider:
26+
27+
1. [Reads data stored in the factory partition](https://github.com/project-chip/connectedhomeip/blob/master/src/platform/ESP32/ESP32FactoryDataProvider.h#L56)
28+
_(This will be deprecated in the future)_
29+
2. [Provides APIs to set fixed data that gets read later](https://github.com/project-chip/connectedhomeip/blob/master/src/platform/ESP32/StaticESP32DeviceInfoProvider.h)
30+
31+
- New products should use the `StaticESP32DeviceInfoProvider`. Utilize the
32+
`Set...()` APIs to set the fixed data.
33+
- Existing products using the first implementation can continue to use it if
34+
they do not wish to change the data.
35+
- For products using the first implementation and wanting to change the fixed
36+
data via OTA, they should switch to the second implementation in the OTA
37+
image and use the `Set...()` APIs to set the fixed data.
38+
39+
#### Example:
40+
41+
```cpp
42+
#include <platform/ESP32/StaticESP32FactoryDataProvider.h>
43+
44+
DeviceLayer::StaticESP32DeviceInfoProvider deviceInfoProvider;
45+
46+
// Define array for Supported Calendar Types
47+
using namespace chip::app::Clusters::TimeFormatLocalization::CalendarTypeEnum;
48+
CalendarTypeEnum supportedCalendarTypes[] = {
49+
CalendarTypeEnum::kGregorian, CalendarTypeEnum::kCoptic,
50+
CalendarTypeEnum::kEthiopian, CalendarTypeEnum::kChinese,
51+
};
52+
53+
// Define array for Supported Locales
54+
const char* supportedLocales[] = {
55+
"en-US",
56+
"en-EU",
57+
};
58+
59+
// Define array for Fixed labels { EndpointId, Label, Value }
60+
struct StaticESP32DeviceInfoProvider::FixedLabelEntry fixedLabels[] = {
61+
{ 0, "Room", "Bedroom 2" },
62+
{ 0, "Orientation", "North" },
63+
{ 0, "Direction", "Up" },
64+
};
65+
66+
Span<CalendarTypeEnum> sSupportedCalendarTypes(supportedCalendarTypes);
67+
Span<const char*> sSupportedLocales(supportedLocales);
68+
Span<StaticESP32DeviceInfoProvider::FixedLabelEntry> sFixedLabels(fixedLabels);
69+
70+
{
71+
deviceInfoProvider.SetSupportedLocales(sSupportedLocales);
72+
deviceInfoProvider.SetSupportedCalendarTypes(sSupportedCalendarTypes);
73+
deviceInfoProvider.SetFixedLabels(sFixedLabels);
74+
DeviceLayer::SetDeviceInfoProvider(&deviceInfoProvider);
75+
}
76+
```

scripts/tools/generate_esp32_chip_factory_bin.py

+1-122
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,6 @@
1818

1919
import argparse
2020
import base64
21-
import enum
2221
import logging
2322
import os
2423
import sys
@@ -27,6 +26,7 @@
2726
import cryptography.x509
2827
from bitarray import bitarray
2928
from bitarray.util import ba2int
29+
from esp_secure_cert.tlv_format import generate_partition_ds, generate_partition_no_ds, tlv_priv_key_t, tlv_priv_key_type_t
3030

3131
CHIP_TOPDIR = os.path.dirname(os.path.realpath(__file__))[:-len(os.path.join('scripts', 'tools'))]
3232
sys.path.insert(0, os.path.join(CHIP_TOPDIR, 'scripts', 'tools', 'spake2p'))
@@ -148,84 +148,16 @@
148148
'encoding': 'hex2bin',
149149
'value': None,
150150
},
151-
# DeviceInfoProvider
152-
'cal-types': {
153-
'type': 'data',
154-
'encoding': 'u32',
155-
'value': None,
156-
},
157-
'locale-sz': {
158-
'type': 'data',
159-
'encoding': 'u32',
160-
'value': None,
161-
},
162-
163-
# Other device info provider keys are dynamically generated
164-
# in the respective functions.
165151
}
166152

167153

168-
class CalendarTypes(enum.Enum):
169-
Buddhist = 0
170-
Chinese = 1
171-
Coptic = 2
172-
Ethiopian = 3
173-
Gregorian = 4
174-
Hebrew = 5
175-
Indian = 6
176-
Islamic = 7
177-
Japanese = 8
178-
Korean = 9
179-
Persian = 10
180-
Taiwanese = 11
181-
182-
183-
# Supported Calendar types is stored as a bit array in one uint32_t.
184-
def calendar_types_to_uint32(calendar_types):
185-
result = bitarray(32, endian='little')
186-
result.setall(0)
187-
for calendar_type in calendar_types:
188-
try:
189-
result[CalendarTypes[calendar_type].value] = 1
190-
except KeyError:
191-
logging.error('Unknown calendar type: %s', calendar_type)
192-
logging.error('Supported calendar types: %s', ', '.join(CalendarTypes.__members__))
193-
sys.exit(1)
194-
return ba2int(result)
195-
196-
197154
def ishex(s):
198155
try:
199156
_ = int(s, 16)
200157
return True
201158
except ValueError:
202159
return False
203160

204-
# get_fixed_label_dict() converts the list of strings to per endpoint dictionaries.
205-
# example input : ['0/orientation/up', '1/orientation/down', '2/orientation/down']
206-
# example output : {'0': [{'orientation': 'up'}], '1': [{'orientation': 'down'}], '2': [{'orientation': 'down'}]}
207-
208-
209-
def get_fixed_label_dict(fixed_labels):
210-
fl_dict = {}
211-
for fl in fixed_labels:
212-
_l = fl.split('/')
213-
214-
if len(_l) != 3:
215-
logging.error('Invalid fixed label: %s', fl)
216-
sys.exit(1)
217-
218-
if not (ishex(_l[0]) and (len(_l[1]) > 0 and len(_l[1]) < 16) and (len(_l[2]) > 0 and len(_l[2]) < 16)):
219-
logging.error('Invalid fixed label: %s', fl)
220-
sys.exit(1)
221-
222-
if _l[0] not in fl_dict.keys():
223-
fl_dict[_l[0]] = list()
224-
225-
fl_dict[_l[0]].append({_l[1]: _l[2]})
226-
227-
return fl_dict
228-
229161
# get_supported_modes_dict() converts the list of strings to per endpoint dictionaries.
230162
# example with semantic tags
231163
# input : ['0/label1/1/"1\0x8000, 2\0x8000" 1/label2/1/"1\0x8000, 2\0x8000"']
@@ -345,52 +277,6 @@ def populate_factory_data(args, spake2p_params):
345277
if args.hw_ver_str:
346278
FACTORY_DATA['hw-ver-str']['value'] = args.hw_ver_str
347279

348-
if args.calendar_types:
349-
FACTORY_DATA['cal-types']['value'] = calendar_types_to_uint32(args.calendar_types)
350-
351-
# Supported locale is stored as multiple entries, key format: "locale/<index>, example key: "locale/0"
352-
if args.locales:
353-
FACTORY_DATA['locale-sz']['value'] = len(args.locales)
354-
355-
for i in range(len(args.locales)):
356-
_locale = {
357-
'type': 'data',
358-
'encoding': 'string',
359-
'value': args.locales[i]
360-
}
361-
FACTORY_DATA.update({'locale/{:x}'.format(i): _locale})
362-
363-
# Each endpoint can contains the fixed lables
364-
# - fl-sz/<index> : number of fixed labels for the endpoint
365-
# - fl-k/<ep>/<index> : fixed label key for the endpoint and index
366-
# - fl-v/<ep>/<index> : fixed label value for the endpoint and index
367-
if args.fixed_labels:
368-
dict = get_fixed_label_dict(args.fixed_labels)
369-
for key in dict.keys():
370-
_sz = {
371-
'type': 'data',
372-
'encoding': 'u32',
373-
'value': len(dict[key])
374-
}
375-
FACTORY_DATA.update({'fl-sz/{:x}'.format(int(key)): _sz})
376-
377-
for i in range(len(dict[key])):
378-
entry = dict[key][i]
379-
380-
_label_key = {
381-
'type': 'data',
382-
'encoding': 'string',
383-
'value': list(entry.keys())[0]
384-
}
385-
_label_value = {
386-
'type': 'data',
387-
'encoding': 'string',
388-
'value': list(entry.values())[0]
389-
}
390-
391-
FACTORY_DATA.update({'fl-k/{:x}/{:x}'.format(int(key), i): _label_key})
392-
FACTORY_DATA.update({'fl-v/{:x}/{:x}'.format(int(key), i): _label_value})
393-
394280
# SupportedModes are stored as multiple entries
395281
# - sm-sz/<ep> : number of supported modes for the endpoint
396282
# - sm-label/<ep>/<index> : supported modes label key for the endpoint and index
@@ -547,13 +433,6 @@ def any_base_int(s): return int(s, 0)
547433
help=('128-bit unique identifier for generating rotating device identifier, '
548434
'provide 32-byte hex string, e.g. "1234567890abcdef1234567890abcdef"'))
549435

550-
# These will be used by DeviceInfoProvider
551-
parser.add_argument('--calendar-types', nargs='+',
552-
help=('List of supported calendar types.\nSupported Calendar Types: Buddhist, Chinese, Coptic, Ethiopian, '
553-
'Gregorian, Hebrew, Indian, Islamic, Japanese, Korean, Persian, Taiwanese'))
554-
parser.add_argument('--locales', nargs='+', help='List of supported locales, Language Tag as defined by BCP47, eg. en-US en-GB')
555-
parser.add_argument('--fixed-labels', nargs='+',
556-
help='List of fixed labels, eg: "0/orientation/up" "1/orientation/down" "2/orientation/down"')
557436
parser.add_argument('--supported-modes', type=str, nargs='+', required=False,
558437
help='List of supported modes, eg: mode1/label1/ep/"tagValue1\\mfgCode, tagValue2\\mfgCode" mode2/label2/ep/"tagValue1\\mfgCode, tagValue2\\mfgCode" mode3/label3/ep/"tagValue1\\mfgCode, tagValue2\\mfgCode"')
559438

src/platform/ESP32/BUILD.gn

+2
Original file line numberDiff line numberDiff line change
@@ -182,6 +182,8 @@ static_library("ESP32") {
182182
sources += [
183183
"ESP32DeviceInfoProvider.cpp",
184184
"ESP32DeviceInfoProvider.h",
185+
"StaticESP32DeviceInfoProvider.cpp",
186+
"StaticESP32DeviceInfoProvider.h",
185187
]
186188
}
187189

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,122 @@
1+
/*
2+
3+
* Copyright (c) 2024 Project CHIP Authors
4+
*
5+
* Licensed under the Apache License, Version 2.0 (the "License");
6+
* you may not use this file except in compliance with the License.
7+
* You may obtain a copy of the License at
8+
*
9+
* http://www.apache.org/licenses/LICENSE-2.0
10+
*
11+
* Unless required by applicable law or agreed to in writing, software
12+
* distributed under the License is distributed on an "AS IS" BASIS,
13+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
* See the License for the specific language governing permissions and
15+
* limitations under the License.
16+
*/
17+
#include <lib/support/CodeUtils.h>
18+
#include <platform/ESP32/StaticESP32DeviceInfoProvider.h>
19+
20+
namespace chip {
21+
namespace DeviceLayer {
22+
23+
StaticESP32DeviceInfoProvider & StaticESP32DeviceInfoProvider::GetDefaultInstance(void)
24+
{
25+
static StaticESP32DeviceInfoProvider sInstance;
26+
return sInstance;
27+
}
28+
29+
DeviceInfoProvider::FixedLabelIterator * StaticESP32DeviceInfoProvider::IterateFixedLabel(EndpointId endpoint)
30+
{
31+
return chip::Platform::New<StaticFixedLabelIteratorImpl>(endpoint, mFixedLabels);
32+
}
33+
34+
StaticESP32DeviceInfoProvider::StaticFixedLabelIteratorImpl::StaticFixedLabelIteratorImpl(EndpointId endpoint,
35+
const Span<FixedLabelEntry> & labels)
36+
{
37+
mEndpoint = endpoint;
38+
mLabels = labels;
39+
mIndex = 0;
40+
}
41+
42+
size_t StaticESP32DeviceInfoProvider::StaticFixedLabelIteratorImpl::Count()
43+
{
44+
size_t count = 0;
45+
for (size_t i = 0; i < mLabels.size(); i++)
46+
{
47+
const FixedLabelEntry & entry = mLabels.data()[i];
48+
49+
if (entry.endpointId == mEndpoint)
50+
{
51+
count++;
52+
}
53+
}
54+
return count;
55+
}
56+
57+
bool StaticESP32DeviceInfoProvider::StaticFixedLabelIteratorImpl::Next(FixedLabelType & output)
58+
{
59+
ChipLogDetail(DeviceLayer, "Get the fixed label with index:%u at endpoint:%d", static_cast<unsigned>(mIndex), mEndpoint);
60+
61+
while (mIndex < mLabels.size())
62+
{
63+
const FixedLabelEntry & entry = mLabels.data()[mIndex++];
64+
if (entry.endpointId == mEndpoint)
65+
{
66+
output.label = entry.label;
67+
output.value = entry.value;
68+
return true;
69+
}
70+
}
71+
72+
return false;
73+
}
74+
75+
DeviceInfoProvider::SupportedLocalesIterator * StaticESP32DeviceInfoProvider::IterateSupportedLocales()
76+
{
77+
return chip::Platform::New<StaticSupportedLocalesIteratorImpl>(mSupportedLocales);
78+
}
79+
80+
StaticESP32DeviceInfoProvider::StaticSupportedLocalesIteratorImpl::StaticSupportedLocalesIteratorImpl(
81+
const Span<CharSpan> & locales)
82+
{
83+
mLocales = locales;
84+
}
85+
86+
size_t StaticESP32DeviceInfoProvider::StaticSupportedLocalesIteratorImpl::Count()
87+
{
88+
return mLocales.empty() ? 0 : mLocales.size();
89+
}
90+
91+
bool StaticESP32DeviceInfoProvider::StaticSupportedLocalesIteratorImpl::Next(CharSpan & output)
92+
{
93+
VerifyOrReturnValue(mIndex < mLocales.size(), false);
94+
output = mLocales.data()[mIndex++];
95+
return true;
96+
}
97+
98+
DeviceInfoProvider::SupportedCalendarTypesIterator * StaticESP32DeviceInfoProvider::IterateSupportedCalendarTypes()
99+
{
100+
return chip::Platform::New<StaticSupportedCalendarTypesIteratorImpl>(mSupportedCalendarTypes);
101+
}
102+
103+
StaticESP32DeviceInfoProvider::StaticSupportedCalendarTypesIteratorImpl::StaticSupportedCalendarTypesIteratorImpl(
104+
const Span<CalendarType> & calendarTypes)
105+
{
106+
mCalendarTypes = calendarTypes;
107+
}
108+
109+
size_t StaticESP32DeviceInfoProvider::StaticSupportedCalendarTypesIteratorImpl::Count()
110+
{
111+
return mCalendarTypes.empty() ? 0 : mCalendarTypes.size();
112+
}
113+
114+
bool StaticESP32DeviceInfoProvider::StaticSupportedCalendarTypesIteratorImpl::Next(CalendarType & output)
115+
{
116+
VerifyOrReturnValue(mIndex < mCalendarTypes.size(), false);
117+
output = mCalendarTypes.data()[mIndex++];
118+
return true;
119+
}
120+
121+
} // namespace DeviceLayer
122+
} // namespace chip

0 commit comments

Comments
 (0)