|
CDC Badge OS
Firmware for the CDC Badge v1.0 hardware security key
|
#include "mod_gpg/GpgStorage.h"#include "cdc_hal/ISecureElement.h"#include "cdc_core/Crypto.h"#include "cdc_log.h"#include <mbedtls/sha256.h>#include <mbedtls/hkdf.h>#include <mbedtls/md.h>#include <mbedtls/platform_util.h>#include <esp_random.h>#include <string.h>Go to the source code of this file.
Classes | |
| struct | cdc::mod_gpg::DecKeyStorage |
| struct | cdc::mod_gpg::AesKeyStorage |
Namespaces | |
| namespace | cdc |
| namespace | cdc::mod_gpg |
Functions | |
| static cdc::hal::ISecureElement * | get_se () |
| static bool | pin_to_hash (const char *pin, uint8_t *hash_out) |
| Computes SHA-256 over a PIN string. | |
| static bool | derive_storage_key (uint16_t slot_id, const uint8_t *pin_hash, uint8_t *key_out) |
| Derives a 32-byte storage key for a specific slot. | |
| static void | build_aad (uint16_t slot_id, const uint8_t magic[MAGIC_SIZE], uint8_t aad_out[6]) |
| Builds the 6-byte AAD for a slot: slot_id (BE) || magic (4). | |
| static uint16_t | resolve_slot (uint16_t rel_index) |
| Resolves an absolute R-Memory slot index relative to the module range. | |
| void | gpg_storage_set_slot_range (uint16_t eccStart, uint16_t eccEnd) |
| void | gpg_storage_set_rmem_range (uint16_t rmemStart, uint16_t rmemEnd) |
| bool | gpg_storage_ready (void) |
| uint8_t | gpg_storage_sig_slot (void) |
| uint8_t | gpg_storage_dec_slot (void) |
| uint8_t | gpg_storage_aut_slot (void) |
| static bool | save_slot_encrypted (uint16_t slot_id, const uint8_t magic[MAGIC_SIZE], const uint8_t *payload, size_t payload_len, uint8_t *record_buf, size_t record_buf_len, const char *pin) |
| Encrypts and writes an arbitrary payload to a slot. | |
| static bool | load_slot_decrypted (uint16_t slot_id, const uint8_t magic[MAGIC_SIZE], uint8_t *payload_out, size_t payload_len, const char *pin) |
| Reads and decrypts a payload from a slot. | |
| bool | gpg_storage_save_dec_privkey (const uint8_t *privkey, const char *pin) |
| Saves a DEC private key into R-Memory using PIN-bound AES-GCM. | |
| bool | gpg_storage_load_dec_privkey (uint8_t *privkey_out, const char *pin) |
| Loads and decrypts the DEC private key from R-Memory. | |
| bool | gpg_storage_has_dec_privkey (void) |
| Returns true if encrypted DEC private key record exists. | |
| bool | gpg_storage_delete_dec_privkey (void) |
| Deletes DEC private key record. | |
| bool | gpg_storage_save_aes_key (const uint8_t *key, size_t key_len, const char *pin) |
| Saves the symmetric AES key for PSO:DECIPHER (DO 0xD5). | |
| bool | gpg_storage_load_aes_key (uint8_t *key_out, size_t *key_len_out, const char *pin) |
| Loads the symmetric AES key from R-Memory. | |
| bool | gpg_storage_has_aes_key (void) |
| Returns true if a symmetric AES key record exists. | |
| bool | gpg_storage_delete_aes_key (void) |
| Deletes the symmetric AES key record. | |
| void | gpg_storage_set_session_pin (const char *pin) |
| Stores session PIN-derived key after successful PIN verification. | |
| bool | gpg_storage_get_session_key (uint8_t *key_out) |
| Returns current session key if session is active. | |
| void | gpg_storage_clear_session (void) |
| Clears the cached session key. | |
Variables | |
| static const char * | TAG = "GPGStorage" |
| Encrypted persistent storage for OpenPGP secret material. | |
| static constexpr uint16_t | RMEM_SLOT_DEC_KEY = 1 |
| R-Memory slot offset for the DEC private key payload (= ECC slot 2). | |
| static constexpr uint16_t | RMEM_SLOT_AES_KEY = 2 |
| R-Memory slot offset for the symmetric AES key payload (= ECC slot 3). | |
| static constexpr uint8_t | DEC_KEY_MAGIC [4] = {'E', 'C', 'D', 'H'} |
| Magic marker for encrypted DEC private key records. | |
| static constexpr uint8_t | AES_KEY_MAGIC [4] = {'A', 'E', 'S', '1'} |
| Magic marker for symmetric AES key records (DO 0xD5). | |
| static constexpr size_t | MAGIC_SIZE = 4 |
| static constexpr size_t | NONCE_SIZE = 12 |
| static constexpr size_t | TAG_SIZE = 16 |
| static constexpr size_t | PRIVKEY_SIZE = 32 |
| static constexpr size_t | AES_MAX_KEY_SIZE = 32 |
| static constexpr size_t | DEC_TOTAL_SIZE = MAGIC_SIZE + NONCE_SIZE + PRIVKEY_SIZE + TAG_SIZE |
| static constexpr size_t | AES_RECORD_PAYLOAD = 1 + AES_MAX_KEY_SIZE |
| static constexpr size_t | AES_TOTAL_SIZE = MAGIC_SIZE + NONCE_SIZE + AES_RECORD_PAYLOAD + TAG_SIZE |
| static constexpr char | HKDF_INFO [] = "GPG-STORAGE-V2" |
| HKDF info string for storage key derivation. | |
| static struct { ... } | s_storage |
|
static |
Builds the 6-byte AAD for a slot: slot_id (BE) || magic (4).
| slot_id | Absolute R-Memory slot identifier. |
| magic | 4-byte magic marker. |
| aad_out | 6-byte output buffer. |
Definition at line 171 of file GpgStorage.cpp.
References MAGIC_SIZE.
Referenced by load_slot_decrypted(), and save_slot_encrypted().
|
static |
Derives a 32-byte storage key for a specific slot.
| slot_id | Absolute R-Memory slot identifier (used in HKDF salt). |
| pin_hash | Optional 32-byte PIN hash; nullptr falls back to chip-bound key. |
| key_out | 32-byte output buffer. |
Definition at line 129 of file GpgStorage.cpp.
References get_se(), and HKDF_INFO.
Referenced by load_slot_decrypted(), and save_slot_encrypted().
|
static |
Definition at line 105 of file GpgStorage.cpp.
References cdc::hal::getSecureElementInstance().
Referenced by derive_storage_key(), gpg_storage_delete_aes_key(), gpg_storage_delete_dec_privkey(), gpg_storage_has_aes_key(), gpg_storage_has_dec_privkey(), load_slot_decrypted(), and save_slot_encrypted().
| uint8_t gpg_storage_aut_slot | ( | void | ) |
Definition at line 209 of file GpgStorage.cpp.
References s_storage.
Referenced by cmd_get_data(), cmd_internal_authenticate(), get_ecc_slot_for_key_ref(), gpg_generate_key(), openpgp_factory_reset(), and put_data_algo_attr().
| void gpg_storage_clear_session | ( | void | ) |
Clears the cached session key.
Definition at line 467 of file GpgStorage.cpp.
References s_storage.
Referenced by cmd_select(), gpg_storage_set_session_pin(), and openpgp_factory_reset().
| uint8_t gpg_storage_dec_slot | ( | void | ) |
Definition at line 208 of file GpgStorage.cpp.
References s_storage.
Referenced by cmd_get_data(), get_ecc_slot_for_key_ref(), and gpg_generate_key().
| bool gpg_storage_delete_aes_key | ( | void | ) |
Deletes the symmetric AES key record.
Definition at line 439 of file GpgStorage.cpp.
References get_se(), cdc::hal::OK, resolve_slot(), and RMEM_SLOT_AES_KEY.
| bool gpg_storage_delete_dec_privkey | ( | void | ) |
Deletes DEC private key record.
Definition at line 378 of file GpgStorage.cpp.
References get_se(), cdc::hal::OK, resolve_slot(), and RMEM_SLOT_DEC_KEY.
Referenced by openpgp_factory_reset().
| bool gpg_storage_get_session_key | ( | uint8_t * | key_out | ) |
Returns current session key if session is active.
| key_out | 32-byte output buffer. |
Definition at line 459 of file GpgStorage.cpp.
References s_storage.
| bool gpg_storage_has_aes_key | ( | void | ) |
Returns true if a symmetric AES key record exists.
Definition at line 427 of file GpgStorage.cpp.
References AES_KEY_MAGIC, get_se(), MAGIC_SIZE, cdc::hal::OK, resolve_slot(), and RMEM_SLOT_AES_KEY.
Referenced by cmd_pso_decipher_aes().
| bool gpg_storage_has_dec_privkey | ( | void | ) |
Returns true if encrypted DEC private key record exists.
Definition at line 366 of file GpgStorage.cpp.
References DEC_KEY_MAGIC, DEC_TOTAL_SIZE, get_se(), MAGIC_SIZE, cdc::hal::OK, resolve_slot(), and RMEM_SLOT_DEC_KEY.
Referenced by cmd_pso_decipher(), and read_public_key().
| bool gpg_storage_load_aes_key | ( | uint8_t * | key_out, |
| size_t * | key_len_out, | ||
| const char * | pin ) |
Loads the symmetric AES key from R-Memory.
| key_out | Output buffer (must hold at least 32 bytes). |
| key_len_out | Receives the stored key length (16 or 32). |
| pin | Session PIN; nullptr falls back to chip-bound key. |
Definition at line 408 of file GpgStorage.cpp.
References AES_KEY_MAGIC, AES_RECORD_PAYLOAD, load_slot_decrypted(), resolve_slot(), and RMEM_SLOT_AES_KEY.
Referenced by cmd_pso_decipher_aes().
| bool gpg_storage_load_dec_privkey | ( | uint8_t * | privkey_out, |
| const char * | pin ) |
Loads and decrypts the DEC private key from R-Memory.
| privkey_out | 32-byte output buffer. |
| pin | Session PIN; nullptr falls back to chip-bound key. |
Definition at line 360 of file GpgStorage.cpp.
References DEC_KEY_MAGIC, load_slot_decrypted(), PRIVKEY_SIZE, resolve_slot(), and RMEM_SLOT_DEC_KEY.
Referenced by cmd_pso_decipher(), and read_public_key().
| bool gpg_storage_ready | ( | void | ) |
Definition at line 206 of file GpgStorage.cpp.
References s_storage.
Referenced by gpg_export_pubkey_pem(), gpg_generate_key(), gpg_init(), gpg_reset(), and cdc::mod_gpg::GpgModule::init().
| bool gpg_storage_save_aes_key | ( | const uint8_t * | key, |
| size_t | key_len, | ||
| const char * | pin ) |
Saves the symmetric AES key for PSO:DECIPHER (DO 0xD5).
| key | AES key bytes (16 or 32). |
| key_len | Key length (16 or 32). |
| pin | Session PIN; nullptr falls back to chip-bound key. |
Definition at line 385 of file GpgStorage.cpp.
References AES_KEY_MAGIC, AES_MAX_KEY_SIZE, AES_RECORD_PAYLOAD, AES_TOTAL_SIZE, LOG_I, resolve_slot(), RMEM_SLOT_AES_KEY, save_slot_encrypted(), and TAG.
Referenced by cmd_put_data().
| bool gpg_storage_save_dec_privkey | ( | const uint8_t * | privkey, |
| const char * | pin ) |
Saves a DEC private key into R-Memory using PIN-bound AES-GCM.
| privkey | 32-byte P-256 private key scalar. |
| pin | Session PIN; nullptr falls back to chip-bound key. |
Definition at line 347 of file GpgStorage.cpp.
References DEC_KEY_MAGIC, DEC_TOTAL_SIZE, LOG_I, PRIVKEY_SIZE, resolve_slot(), RMEM_SLOT_DEC_KEY, save_slot_encrypted(), and TAG.
Referenced by cmd_put_data_odd(), generate_dec_key(), and gpg_generate_key().
| void gpg_storage_set_rmem_range | ( | uint16_t | rmemStart, |
| uint16_t | rmemEnd ) |
Definition at line 201 of file GpgStorage.cpp.
References s_storage.
Referenced by cdc::mod_gpg::GpgModule::init().
| void gpg_storage_set_session_pin | ( | const char * | pin | ) |
Stores session PIN-derived key after successful PIN verification.
| pin | Verified PIN string. |
Definition at line 446 of file GpgStorage.cpp.
References gpg_storage_clear_session(), pin_to_hash(), and s_storage.
Referenced by cmd_verify().
| void gpg_storage_set_slot_range | ( | uint16_t | eccStart, |
| uint16_t | eccEnd ) |
Definition at line 186 of file GpgStorage.cpp.
References s_storage.
Referenced by cdc::mod_gpg::GpgModule::init().
| uint8_t gpg_storage_sig_slot | ( | void | ) |
Definition at line 207 of file GpgStorage.cpp.
References s_storage.
Referenced by cmd_get_data(), cmd_pso_cds(), get_ecc_slot_for_key_ref(), gpg_alchemy_fingerprint(), gpg_export_pubkey_pem(), gpg_generate_key(), cdc::mod_gpg::gpgCrossSign(), openpgp_factory_reset(), and put_data_algo_attr().
|
static |
Reads and decrypts a payload from a slot.
| slot_id | Absolute R-Memory slot. |
| magic | 4-byte magic marker. |
| payload_out | Output buffer. |
| payload_len | Expected plaintext length. |
| pin | Session PIN; nullptr falls back to chip-bound key. |
Definition at line 290 of file GpgStorage.cpp.
References cdc::core::aesGcm256Open(), build_aad(), derive_storage_key(), get_se(), MAGIC_SIZE, NONCE_SIZE, cdc::hal::OK, pin_to_hash(), and TAG_SIZE.
Referenced by gpg_storage_load_aes_key(), and gpg_storage_load_dec_privkey().
|
static |
Computes SHA-256 over a PIN string.
| pin | PIN string; may be empty/nullptr. |
| hash_out | 32-byte output buffer. |
Definition at line 115 of file GpgStorage.cpp.
Referenced by gpg_storage_set_session_pin(), load_slot_decrypted(), and save_slot_encrypted().
|
static |
Resolves an absolute R-Memory slot index relative to the module range.
| rel_index | Offset within the module range. |
Definition at line 182 of file GpgStorage.cpp.
References s_storage.
Referenced by gpg_storage_delete_aes_key(), gpg_storage_delete_dec_privkey(), gpg_storage_has_aes_key(), gpg_storage_has_dec_privkey(), gpg_storage_load_aes_key(), gpg_storage_load_dec_privkey(), gpg_storage_save_aes_key(), and gpg_storage_save_dec_privkey().
|
static |
Encrypts and writes an arbitrary payload to a slot.
| slot_id | Absolute R-Memory slot. |
| magic | 4-byte magic marker. |
| payload | Plaintext bytes. |
| payload_len | Plaintext length. |
| record_buf | Scratch buffer; must hold MAGIC + NONCE + payload_len + TAG. |
| record_buf_len | Length of record_buf. |
| pin | Session PIN; nullptr falls back to chip-bound key. |
Definition at line 222 of file GpgStorage.cpp.
References cdc::core::aesGcm256Seal(), build_aad(), derive_storage_key(), get_se(), LOG_E, MAGIC_SIZE, NONCE_SIZE, cdc::hal::OK, pin_to_hash(), TAG, and TAG_SIZE.
Referenced by gpg_storage_save_aes_key(), and gpg_storage_save_dec_privkey().
|
staticconstexpr |
Magic marker for symmetric AES key records (DO 0xD5).
Definition at line 36 of file GpgStorage.cpp.
Referenced by gpg_storage_has_aes_key(), gpg_storage_load_aes_key(), and gpg_storage_save_aes_key().
|
staticconstexpr |
Definition at line 42 of file GpgStorage.cpp.
Referenced by gpg_storage_save_aes_key().
|
staticconstexpr |
Definition at line 44 of file GpgStorage.cpp.
Referenced by gpg_storage_load_aes_key(), and gpg_storage_save_aes_key().
|
staticconstexpr |
Definition at line 45 of file GpgStorage.cpp.
Referenced by gpg_storage_save_aes_key().
|
staticconstexpr |
Magic marker for encrypted DEC private key records.
Definition at line 33 of file GpgStorage.cpp.
Referenced by gpg_storage_has_dec_privkey(), gpg_storage_load_dec_privkey(), and gpg_storage_save_dec_privkey().
|
staticconstexpr |
Definition at line 43 of file GpgStorage.cpp.
Referenced by gpg_storage_has_dec_privkey(), and gpg_storage_save_dec_privkey().
|
staticconstexpr |
HKDF info string for storage key derivation.
Definition at line 48 of file GpgStorage.cpp.
Referenced by derive_storage_key().
|
staticconstexpr |
Definition at line 38 of file GpgStorage.cpp.
Referenced by build_aad(), gpg_storage_has_aes_key(), gpg_storage_has_dec_privkey(), load_slot_decrypted(), and save_slot_encrypted().
|
staticconstexpr |
Definition at line 39 of file GpgStorage.cpp.
Referenced by load_slot_decrypted(), and save_slot_encrypted().
|
staticconstexpr |
Definition at line 41 of file GpgStorage.cpp.
Referenced by gpg_storage_load_dec_privkey(), and gpg_storage_save_dec_privkey().
|
staticconstexpr |
R-Memory slot offset for the symmetric AES key payload (= ECC slot 3).
Definition at line 30 of file GpgStorage.cpp.
Referenced by gpg_storage_delete_aes_key(), gpg_storage_has_aes_key(), gpg_storage_load_aes_key(), and gpg_storage_save_aes_key().
|
staticconstexpr |
R-Memory slot offset for the DEC private key payload (= ECC slot 2).
Definition at line 27 of file GpgStorage.cpp.
Referenced by gpg_storage_delete_dec_privkey(), gpg_storage_has_dec_privkey(), gpg_storage_load_dec_privkey(), and gpg_storage_save_dec_privkey().
| struct { ... } s_storage |
|
static |
Encrypted persistent storage for OpenPGP secret material.
Encrypts payloads with AES-256-GCM. The wrapping key is derived via HKDF over (chip_id || pin_hash) with the slot index appended; AAD binds the record to (slot_id, magic) to defeat slot-shuffle attacks.
Definition at line 20 of file GpgStorage.cpp.
|
staticconstexpr |
Definition at line 40 of file GpgStorage.cpp.
Referenced by load_slot_decrypted(), and save_slot_encrypted().