/* * * Copyright (c) 2022 Project CHIP Authors * * 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 * Open IoT SDK key-value storage base on flash TDBStore. */ #pragma once #include #include #include #include #include #include #include namespace chip { namespace DeviceLayer { namespace Internal { using namespace chip::Platform::PersistedStorage; // PSA Protected Storage interface provides 64-bit ID number, // For Matter we reserved region ranges from 0x00010000 to 0x0001FFFF // e.g. key = 0x010214 // '01' = Matter region // '02' = the sub region group base offset (Factory, Config, Counter or KVS) // '14' = the id offset inside the group. inline constexpr Key kMatterPsaPaKeyRegion = 0x010000U; constexpr inline Key GetPsaPaKey(Key group, uint8_t id) { return kMatterPsaPaKeyRegion | (group) << 8 | id; } #define KVS_MAX_ENTRIES 30 #define KVS_KEY_MAP_UPDATE_DELAY_SEC 5 /** * This class provides access to configuration information for Open IoT SDK platform * stored in PSA Protected Storage. */ class KVPsaPsStore { public: using Key = chip::Platform::PersistedStorage::Key; // Persistent config values set at manufacturing time. Retained during factory reset. static const Key kMatterFactory_KeyOffset; // Persistent config values set at runtime. Cleared during factory reset. static const Key kMatterConfig_KeyOffset; // Persistent counter values set at runtime. Retained during factory reset. static const Key kMatterCounter_KeyOffset; // Persistent config values set at runtime. Cleared during factory reset. static const Key kMatterKvs_KeyOffset; // Key definitions for well-known keys. // Factory keys static const Key kConfigKey_SerialNum; static const Key kConfigKey_MfrDeviceId; static const Key kConfigKey_MfrDeviceCert; static const Key kConfigKey_MfrDeviceICACerts; static const Key kConfigKey_MfrDevicePrivateKey; static const Key kConfigKey_HardwareVersion; static const Key kConfigKey_ManufacturingDate; static const Key kConfigKey_SetupPinCode; static const Key kConfigKey_SetupDiscriminator; static const Key kConfigKey_Spake2pIterationCount; static const Key kConfigKey_Spake2pSalt; static const Key kConfigKey_Spake2pVerifier; static const Key kConfigKey_VendorId; static const Key kConfigKey_ProductId; // Config Keys static const Key kConfigKey_ServiceConfig; static const Key kConfigKey_PairedAccountId; static const Key kConfigKey_ServiceId; static const Key kConfigKey_LastUsedEpochKeyId; static const Key kConfigKey_FailSafeArmed; static const Key kConfigKey_WiFiStationSecType; static const Key kConfigKey_RegulatoryLocation; static const Key kConfigKey_CountryCode; static const Key kConfigKey_LocationCapability; static const Key kConfigKey_UniqueId; // Counter Keys static const Key kCounterKey_RebootCount; static const Key kCounterKey_UpTime; static const Key kCounterKey_TotalOperationalHours; static const Key kCounterKey_BootReason; // KVS storage Keys static const Key kConfigKey_KvsStringKeyMap; static const Key kConfigKey_KvsFirstKeySlot; static const Key kConfigKey_KvsLastKeySlot; // NVS helper variables static const Key kMinConfigKey_MatterConfig; static const Key kMaxConfigKey_MatterConfig; static const Key kMinConfigKey_MatterCounter; static const Key kMaxConfigKey_MatterCounter; static const Key kMinMatterPsaPaKeyRegion; static const Key kMaxMatterPsaPaKeyRegion; static CHIP_ERROR Init(void); static CHIP_ERROR Shutdown(void); // Config value accessors. static CHIP_ERROR ReadConfigValue(Key key, bool & val); static CHIP_ERROR ReadConfigValue(Key key, uint32_t & val); static CHIP_ERROR ReadConfigValue(Key key, uint64_t & val); static CHIP_ERROR ReadConfigValueStr(Key key, char * buf, size_t bufSize, size_t & outLen); static CHIP_ERROR ReadConfigValueBin(Key key, uint8_t * buf, size_t bufSize, size_t & outLen); static CHIP_ERROR ReadConfigValueBin(Key key, uint8_t * buf, size_t bufSize, size_t & outLen, size_t offset); static CHIP_ERROR ReadConfigValueCounter(Key counterId, uint32_t & val); static CHIP_ERROR WriteConfigValue(Key key, bool val); static CHIP_ERROR WriteConfigValue(Key key, uint32_t val); static CHIP_ERROR WriteConfigValue(Key key, uint64_t val); static CHIP_ERROR WriteConfigValueStr(Key key, const char * str); static CHIP_ERROR WriteConfigValueStr(Key key, const char * str, size_t strLen); static CHIP_ERROR WriteConfigValueBin(Key key, const uint8_t * data, size_t dataLen); static CHIP_ERROR WriteConfigValueCounter(Key counterId, uint32_t val); static CHIP_ERROR ClearConfigValue(Key key); // Additional functions static CHIP_ERROR FactoryResetConfig(void); static void RunConfigUnitTest(void); static bool ConfigValueExists(Key key); static bool ValidKvsKey(Key key); static void KVSKeyMapUpdate(void); static void ScheduleKVSKeyMapUpdate(void); protected: using ForEachObjectFunct = std::function; static CHIP_ERROR ForEachObject(Key firstKey, Key lastKey, bool addNewRecord, ForEachObjectFunct funct); private: // NVS helper functions. static bool ValidConfigKey(Key key); static bool ConfigValueExists(Key key, size_t & size); static CHIP_ERROR PsaStatus2ChipError(psa_status_t status); static void OnScheduleKVSKeyMapUpdate(System::Layer * systemLayer, void * appState); static bool initialized; }; class KVPsaPsStoreKeyBuilder { public: KVPsaPsStoreKeyBuilder(const char * key); ~KVPsaPsStoreKeyBuilder(); void AddKey(void); void RemoveKey(void); KVPsaPsStore::Key GetKey() const; private: KVPsaPsStore::Key keyValue; char buffer[PersistentStorageDelegate::kKeyLengthMax + 1]; bool valid; bool existing; }; inline KVPsaPsStore::Key KVPsaPsStoreKeyBuilder::GetKey() const { return valid ? keyValue : 0; } } // namespace Internal } // namespace DeviceLayer } // namespace chip using KVStoreConfig = chip::DeviceLayer::Internal::KVPsaPsStore; using KVStoreKeyBuilder = chip::DeviceLayer::Internal::KVPsaPsStoreKeyBuilder;