mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
sys/fido2: improve & simplify flash handling
This commit is contained in:
parent
91d587f8fa
commit
8fdb7a6f35
@ -9,10 +9,6 @@ ifneq (,$(filter fido2_ctap_%,$(USEMODULE)))
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
ifneq (,$(filter fido2_ctap,$(USEMODULE)))
|
ifneq (,$(filter fido2_ctap,$(USEMODULE)))
|
||||||
FEATURES_REQUIRED += periph_flashpage
|
|
||||||
ifeq (,$(filter native,$(CPU)))
|
|
||||||
FEATURES_REQUIRED += periph_flashpage_in_address_space
|
|
||||||
endif
|
|
||||||
FEATURES_REQUIRED += periph_gpio_irq
|
FEATURES_REQUIRED += periph_gpio_irq
|
||||||
|
|
||||||
USEPKG += tiny-asn1
|
USEPKG += tiny-asn1
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
* @}
|
* @}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include "fido2/ctap/ctap.h"
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
@ -35,7 +36,7 @@
|
|||||||
#include "fido2/ctap/transport/hid/ctap_hid.h"
|
#include "fido2/ctap/transport/hid/ctap_hid.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define ENABLE_DEBUG (0)
|
#define ENABLE_DEBUG (0)
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -424,10 +425,12 @@ static uint32_t get_id(void)
|
|||||||
|
|
||||||
static int _reset(void)
|
static int _reset(void)
|
||||||
{
|
{
|
||||||
int ret = fido2_ctap_mem_erase_flash();
|
if (_state.initialized_marker == CTAP_INITIALIZED_MARKER) {
|
||||||
|
int ret = fido2_ctap_mem_erase_flash(&_state);
|
||||||
|
|
||||||
if (ret != CTAP2_OK) {
|
if (ret != CTAP2_OK) {
|
||||||
return ret;
|
return ret;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_state.initialized_marker = CTAP_INITIALIZED_MARKER;
|
_state.initialized_marker = CTAP_INITIALIZED_MARKER;
|
||||||
@ -535,8 +538,7 @@ static int _make_credential(ctap_req_t *req_raw)
|
|||||||
uv = true;
|
uv = true;
|
||||||
}
|
}
|
||||||
/* CTAP specification (version 20190130) section 5.5.8.1 */
|
/* CTAP specification (version 20190130) section 5.5.8.1 */
|
||||||
else if (!fido2_ctap_pin_is_set() && req.pin_auth_present
|
else if (!fido2_ctap_pin_is_set() && req.pin_auth_present && req.pin_auth_len == 0) {
|
||||||
&& req.pin_auth_len == 0) {
|
|
||||||
if (!IS_ACTIVE(CONFIG_FIDO2_CTAP_DISABLE_UP)) {
|
if (!IS_ACTIVE(CONFIG_FIDO2_CTAP_DISABLE_UP)) {
|
||||||
fido2_ctap_utils_user_presence_test();
|
fido2_ctap_utils_user_presence_test();
|
||||||
}
|
}
|
||||||
@ -659,8 +661,7 @@ static int _get_assertion(ctap_req_t *req_raw)
|
|||||||
_assert_state.uv = true;
|
_assert_state.uv = true;
|
||||||
}
|
}
|
||||||
/* CTAP specification (version 20190130) section 5.5.8.2 */
|
/* CTAP specification (version 20190130) section 5.5.8.2 */
|
||||||
else if (!fido2_ctap_pin_is_set() && req.pin_auth_present
|
else if (!fido2_ctap_pin_is_set() && req.pin_auth_present && req.pin_auth_len == 0) {
|
||||||
&& req.pin_auth_len == 0) {
|
|
||||||
if (!IS_ACTIVE(CONFIG_FIDO2_CTAP_DISABLE_UP)) {
|
if (!IS_ACTIVE(CONFIG_FIDO2_CTAP_DISABLE_UP)) {
|
||||||
fido2_ctap_utils_user_presence_test();
|
fido2_ctap_utils_user_presence_test();
|
||||||
}
|
}
|
||||||
@ -1414,9 +1415,9 @@ static int _find_matching_rks(ctap_resident_key_t *rks, size_t rks_len,
|
|||||||
}
|
}
|
||||||
|
|
||||||
ctap_resident_key_t rk = { 0 };
|
ctap_resident_key_t rk = { 0 };
|
||||||
uint32_t addr = 0x0;
|
uint32_t off = 0x0;
|
||||||
|
|
||||||
while (fido2_ctap_mem_read_rk_from_flash(&rk, rp_id_hash, &addr) == CTAP2_OK) {
|
while (fido2_ctap_mem_read_rk_from_flash(&rk, rp_id_hash, &off) == CTAP2_OK) {
|
||||||
if (allow_list_len == 0) {
|
if (allow_list_len == 0) {
|
||||||
memcpy(&rks[index], &rk, sizeof(rk));
|
memcpy(&rks[index], &rk, sizeof(rk));
|
||||||
index++;
|
index++;
|
||||||
|
@ -11,75 +11,47 @@
|
|||||||
* @{
|
* @{
|
||||||
* @file
|
* @file
|
||||||
*
|
*
|
||||||
* @author Nils Ollrogge <nils.ollrogge@fu-berlin.de>
|
* @author Nils Ollrogge <nils-ollrogge@outlook.de>
|
||||||
* @}
|
* @}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#include "bitarithm.h"
|
|
||||||
|
|
||||||
#include "mtd.h"
|
|
||||||
#include "mtd_flashpage.h"
|
#include "mtd_flashpage.h"
|
||||||
|
|
||||||
#include "fido2/ctap/ctap_mem.h"
|
#include "fido2/ctap/ctap_mem.h"
|
||||||
#include "fido2/ctap/ctap_utils.h"
|
#include "fido2/ctap/ctap_utils.h"
|
||||||
|
|
||||||
#define ENABLE_DEBUG (0)
|
#define ENABLE_DEBUG (0)
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
|
static mtd_dev_t *_mtd_dev;
|
||||||
|
|
||||||
#ifdef BOARD_NATIVE
|
#ifdef BOARD_NATIVE
|
||||||
#include "mtd_default.h"
|
#include "mtd_default.h"
|
||||||
/* native mtd is file backed => Start address of flash is 0. */
|
|
||||||
char *_backing_memory = NULL;
|
|
||||||
static mtd_dev_t *_mtd_dev = NULL;
|
|
||||||
#else
|
|
||||||
/**
|
|
||||||
* @brief Reserve flash memory to store CTAP data
|
|
||||||
*/
|
|
||||||
FLASH_WRITABLE_INIT(_backing_memory, CONFIG_FIDO2_CTAP_NUM_FLASHPAGES);
|
|
||||||
/**
|
|
||||||
* @brief MTD device descriptor initialized with flash-page driver
|
|
||||||
*/
|
|
||||||
static mtd_flashpage_t _mtd_flash_dev = MTD_FLASHPAGE_INIT_VAL(CTAP_FLASH_PAGES_PER_SECTOR);
|
|
||||||
static mtd_dev_t *_mtd_dev = &_mtd_flash_dev.base;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Check if flash region is erased
|
|
||||||
*/
|
|
||||||
static bool _flash_is_erased(uint32_t addr, size_t len);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Get available amount of flashpages to store resident keys
|
|
||||||
*/
|
|
||||||
static unsigned _amount_flashpages_rk(void);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Write to flash memory
|
* @brief Write to flash memory
|
||||||
*/
|
*/
|
||||||
static ctap_status_code_t _flash_write(const void *buf, uint32_t addr, size_t len);
|
static ctap_status_code_t _flash_write(const void *buf, uint32_t page, uint32_t off, size_t len);
|
||||||
|
|
||||||
ctap_status_code_t fido2_ctap_mem_init(void)
|
ctap_status_code_t fido2_ctap_mem_init(void)
|
||||||
{
|
{
|
||||||
#ifdef BOARD_NATIVE
|
#ifdef BOARD_NATIVE
|
||||||
_mtd_dev = mtd_default_get_dev(0);
|
_mtd_dev = mtd_default_get_dev(0);
|
||||||
|
#else
|
||||||
|
_mtd_dev = mtd_aux;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
int ret = mtd_init(_mtd_dev);
|
int ret = mtd_init(_mtd_dev);
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
DEBUG("%s, %d: mtd_init failed\n", RIOT_FILE_RELATIVE,
|
||||||
|
__LINE__);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
return CTAP2_OK;
|
return CTAP2_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned _amount_flashpages_rk(void)
|
|
||||||
{
|
|
||||||
return _mtd_dev->sector_count * _mtd_dev->pages_per_sector;
|
|
||||||
}
|
|
||||||
|
|
||||||
ctap_status_code_t fido2_ctap_mem_read(void *buf, uint32_t page, uint32_t offset, uint32_t len)
|
ctap_status_code_t fido2_ctap_mem_read(void *buf, uint32_t page, uint32_t offset, uint32_t len)
|
||||||
{
|
{
|
||||||
assert(buf);
|
assert(buf);
|
||||||
@ -94,59 +66,32 @@ ctap_status_code_t fido2_ctap_mem_read(void *buf, uint32_t page, uint32_t offset
|
|||||||
return CTAP2_OK;
|
return CTAP2_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ctap_status_code_t _flash_write(const void *buf, uint32_t addr, size_t len)
|
static ctap_status_code_t _flash_write(const void *buf, uint32_t page, uint32_t off, size_t len)
|
||||||
{
|
{
|
||||||
assert(buf);
|
assert(buf);
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (!_flash_is_erased(addr, len)) {
|
ret = mtd_write_page(_mtd_dev, buf, page, off, len);
|
||||||
/* page size is always a power of two */
|
|
||||||
const uint32_t page_shift = bitarithm_msb(_mtd_dev->page_size);
|
|
||||||
const uint32_t page_mask = _mtd_dev->page_size - 1;
|
|
||||||
|
|
||||||
ret = mtd_write_page(_mtd_dev, buf, addr >> page_shift, addr & page_mask, len);
|
if (ret < 0) {
|
||||||
|
DEBUG("%s, %d: mtd_write_page failed\n", RIOT_FILE_RELATIVE,
|
||||||
if (ret < 0) {
|
__LINE__);
|
||||||
return CTAP1_ERR_OTHER;
|
return CTAP1_ERR_OTHER;
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
ret = mtd_write(_mtd_dev, buf, addr, len);
|
|
||||||
|
|
||||||
if (ret < 0) {
|
|
||||||
return CTAP1_ERR_OTHER;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return CTAP2_OK;
|
return CTAP2_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool _flash_is_erased(uint32_t addr, size_t len)
|
ctap_status_code_t fido2_ctap_mem_erase_flash(ctap_state_t *state)
|
||||||
{
|
{
|
||||||
#ifdef BOARD_NATIVE
|
/* Calculate total pages needed, rounding up */
|
||||||
return true;
|
uint32_t used_pages = (CTAP_FLASH_STATE_SZ + state->rk_amount_stored * CTAP_FLASH_RK_SZ +
|
||||||
#else
|
_mtd_dev->page_size - 1) / _mtd_dev->page_size;
|
||||||
for (size_t i = 0; i < len; i++) {
|
/* Calculate the number of sectors needed, rounding up */
|
||||||
if (*(uint32_t *)(addr + i) != FLASHPAGE_ERASE_STATE) {
|
uint32_t sector_cnt = (used_pages + _mtd_dev->pages_per_sector - 1) /
|
||||||
return false;
|
_mtd_dev->pages_per_sector;
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
int ret = mtd_erase_sector(_mtd_dev, 0, sector_cnt);
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
static uint32_t _flash_start_addr(void)
|
|
||||||
{
|
|
||||||
return (uint32_t)_backing_memory;
|
|
||||||
}
|
|
||||||
|
|
||||||
ctap_status_code_t fido2_ctap_mem_erase_flash(void)
|
|
||||||
{
|
|
||||||
unsigned addr = _flash_start_addr();
|
|
||||||
unsigned sector_size = _mtd_dev->pages_per_sector * _mtd_dev->page_size;
|
|
||||||
|
|
||||||
int ret = mtd_erase(_mtd_dev, addr, sector_size * CONFIG_FIDO2_CTAP_NUM_FLASHPAGES);
|
|
||||||
|
|
||||||
return ret == 0 ? CTAP2_OK : CTAP1_ERR_OTHER;
|
return ret == 0 ? CTAP2_OK : CTAP1_ERR_OTHER;
|
||||||
}
|
}
|
||||||
@ -157,9 +102,7 @@ ctap_status_code_t fido2_ctap_mem_erase_flash(void)
|
|||||||
*/
|
*/
|
||||||
ctap_status_code_t fido2_ctap_mem_read_state_from_flash(ctap_state_t *state)
|
ctap_status_code_t fido2_ctap_mem_read_state_from_flash(ctap_state_t *state)
|
||||||
{
|
{
|
||||||
uint32_t addr = _flash_start_addr();
|
int ret = mtd_read_page(_mtd_dev, state, 0, 0, sizeof(ctap_state_t));
|
||||||
|
|
||||||
int ret = mtd_read(_mtd_dev, state, addr, sizeof(ctap_state_t));
|
|
||||||
|
|
||||||
return ret == 0 ? CTAP2_OK : CTAP1_ERR_OTHER;
|
return ret == 0 ? CTAP2_OK : CTAP1_ERR_OTHER;
|
||||||
}
|
}
|
||||||
@ -174,16 +117,20 @@ ctap_status_code_t fido2_ctap_mem_read_state_from_flash(ctap_state_t *state)
|
|||||||
ctap_status_code_t fido2_ctap_mem_write_rk_to_flash(ctap_resident_key_t *rk)
|
ctap_status_code_t fido2_ctap_mem_write_rk_to_flash(ctap_resident_key_t *rk)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
uint32_t addr = _flash_start_addr() + FLASHPAGE_SIZE;
|
|
||||||
uint16_t amt_stored = fido2_ctap_get_state()->rk_amount_stored;
|
uint16_t amt_stored = fido2_ctap_get_state()->rk_amount_stored;
|
||||||
ctap_resident_key_t tmp = { 0 };
|
ctap_resident_key_t tmp = { 0 };
|
||||||
bool equal = false;
|
bool equal = false;
|
||||||
|
|
||||||
|
/* skip ctap_state_t struct */
|
||||||
|
uint32_t off = CTAP_FLASH_STATE_SZ;
|
||||||
|
uint32_t page = off / _mtd_dev->page_size;
|
||||||
|
uint32_t page_off = off % _mtd_dev->page_size;
|
||||||
|
|
||||||
for (uint16_t i = 0; i < amt_stored; i++) {
|
for (uint16_t i = 0; i < amt_stored; i++) {
|
||||||
ret = mtd_read(_mtd_dev, &tmp, addr, sizeof(ctap_resident_key_t));
|
ret = mtd_read_page(_mtd_dev, &tmp, page, page_off, sizeof(ctap_resident_key_t));
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
DEBUG("%s, %d: mtd_read failed", RIOT_FILE_RELATIVE,
|
DEBUG("%s, %d: mtd_read failed\n", RIOT_FILE_RELATIVE,
|
||||||
__LINE__);
|
__LINE__);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -193,7 +140,9 @@ ctap_status_code_t fido2_ctap_mem_write_rk_to_flash(ctap_resident_key_t *rk)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
addr += CTAP_FLASH_RK_SZ;
|
off += CTAP_FLASH_RK_SZ;
|
||||||
|
page = off / _mtd_dev->page_size;
|
||||||
|
page_off = off % _mtd_dev->page_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!equal) {
|
if (!equal) {
|
||||||
@ -210,27 +159,43 @@ ctap_status_code_t fido2_ctap_mem_write_rk_to_flash(ctap_resident_key_t *rk)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return _flash_write(rk, addr, CTAP_FLASH_RK_SZ);
|
return _flash_write(rk, page, page_off, CTAP_FLASH_RK_SZ);
|
||||||
}
|
}
|
||||||
|
|
||||||
ctap_status_code_t fido2_ctap_mem_write_state_to_flash(ctap_state_t *state)
|
ctap_status_code_t fido2_ctap_mem_write_state_to_flash(ctap_state_t *state)
|
||||||
{
|
{
|
||||||
return _flash_write(state, _flash_start_addr(), CTAP_FLASH_STATE_SZ);
|
return _flash_write(state, 0, 0, CTAP_FLASH_STATE_SZ);
|
||||||
}
|
}
|
||||||
|
|
||||||
ctap_status_code_t fido2_ctap_mem_read_rk_from_flash(ctap_resident_key_t *key, uint8_t *rp_id_hash,
|
ctap_status_code_t fido2_ctap_mem_read_rk_from_flash(ctap_resident_key_t *key,
|
||||||
uint32_t *addr)
|
const uint8_t *rp_id_hash,
|
||||||
|
uint32_t *off)
|
||||||
{
|
{
|
||||||
uint16_t end;
|
uint16_t end;
|
||||||
uint16_t amt_stored = fido2_ctap_get_state()->rk_amount_stored;
|
uint16_t amt_stored = fido2_ctap_get_state()->rk_amount_stored;
|
||||||
|
|
||||||
if (*addr == 0x0) {
|
/* some error checks for weird offsets that indicate something went wrong */
|
||||||
|
if (*off != 0x0 && *off < CTAP_FLASH_STATE_SZ) {
|
||||||
|
DEBUG("%s, %d: Incorrect offset detected\n", RIOT_FILE_RELATIVE,
|
||||||
|
__LINE__);
|
||||||
|
|
||||||
|
return CTAP1_ERR_OTHER;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*off > CTAP_FLASH_STATE_SZ && (*off - CTAP_FLASH_STATE_SZ) % CTAP_FLASH_RK_SZ != 0) {
|
||||||
|
DEBUG("%s, %d: Incorrect offset detected\n", RIOT_FILE_RELATIVE,
|
||||||
|
__LINE__);
|
||||||
|
|
||||||
|
return CTAP1_ERR_OTHER;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*off == 0x0) {
|
||||||
end = amt_stored;
|
end = amt_stored;
|
||||||
*addr = _flash_start_addr() + FLASHPAGE_SIZE;
|
/* skip ctap_state_t struct */
|
||||||
|
*off = CTAP_FLASH_STATE_SZ;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
uint32_t start_addr = _flash_start_addr() + FLASHPAGE_SIZE;
|
uint16_t rks_read = (*off - CTAP_FLASH_STATE_SZ) / CTAP_FLASH_RK_SZ;
|
||||||
uint16_t rks_read = (*addr - start_addr) / CTAP_FLASH_RK_SZ;
|
|
||||||
|
|
||||||
if (rks_read > amt_stored) {
|
if (rks_read > amt_stored) {
|
||||||
return CTAP1_ERR_OTHER;
|
return CTAP1_ERR_OTHER;
|
||||||
@ -240,7 +205,10 @@ ctap_status_code_t fido2_ctap_mem_read_rk_from_flash(ctap_resident_key_t *key, u
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (uint16_t i = 0; i < end; i++) {
|
for (uint16_t i = 0; i < end; i++) {
|
||||||
int ret = mtd_read(_mtd_dev, key, *addr, sizeof(ctap_resident_key_t));
|
uint32_t page = *off / _mtd_dev->page_size;
|
||||||
|
uint32_t page_off = *off % _mtd_dev->page_size;
|
||||||
|
|
||||||
|
int ret = mtd_read_page(_mtd_dev, key, page, page_off, sizeof(ctap_resident_key_t));
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
DEBUG("%s, %d: mtd_read failed", RIOT_FILE_RELATIVE,
|
DEBUG("%s, %d: mtd_read failed", RIOT_FILE_RELATIVE,
|
||||||
@ -248,7 +216,7 @@ ctap_status_code_t fido2_ctap_mem_read_rk_from_flash(ctap_resident_key_t *key, u
|
|||||||
return CTAP1_ERR_OTHER;
|
return CTAP1_ERR_OTHER;
|
||||||
}
|
}
|
||||||
|
|
||||||
*addr += CTAP_FLASH_RK_SZ;
|
*off += CTAP_FLASH_RK_SZ;
|
||||||
|
|
||||||
if (memcmp(key->rp_id_hash, rp_id_hash, SHA256_DIGEST_LENGTH) == 0) {
|
if (memcmp(key->rp_id_hash, rp_id_hash, SHA256_DIGEST_LENGTH) == 0) {
|
||||||
return CTAP2_OK;
|
return CTAP2_OK;
|
||||||
|
@ -26,6 +26,7 @@
|
|||||||
#define FIDO2_CTAP_H
|
#define FIDO2_CTAP_H
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
|
@ -22,10 +22,11 @@
|
|||||||
#ifndef FIDO2_CTAP_CTAP_MEM_H
|
#ifndef FIDO2_CTAP_CTAP_MEM_H
|
||||||
#define FIDO2_CTAP_CTAP_MEM_H
|
#define FIDO2_CTAP_CTAP_MEM_H
|
||||||
|
|
||||||
|
#include "assert.h"
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
#include "fido2/ctap/ctap.h"
|
#include "fido2/ctap/ctap.h"
|
||||||
#include "periph/flashpage.h"
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
@ -41,23 +42,20 @@ extern "C" {
|
|||||||
/** @} */
|
/** @} */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Default amount of flashpages to use
|
* @brief Ensure CONFIG_SLOT_AUX_LEN is defined
|
||||||
*/
|
*/
|
||||||
#ifndef CONFIG_FIDO2_CTAP_NUM_FLASHPAGES
|
#ifndef CONFIG_SLOT_AUX_LEN
|
||||||
#define CONFIG_FIDO2_CTAP_NUM_FLASHPAGES 4
|
#define CONFIG_SLOT_AUX_LEN 0
|
||||||
#endif
|
|
||||||
|
|
||||||
#if CONFIG_FIDO2_CTAP_NUM_FLASHPAGES < 2
|
|
||||||
#error "ctap_mem.h: Configured number of flashpages is invalid"
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Calculate padding needed to align struct size for saving to flash
|
* @brief Calculate padding needed to align struct size for saving to flash
|
||||||
*/
|
*/
|
||||||
#define CTAP_FLASH_ALIGN_PAD(x) (sizeof(x) % FLASHPAGE_WRITE_BLOCK_SIZE == \
|
#define CTAP_FLASH_ALIGN_PAD(x) (sizeof(x) % FLASHPAGE_WRITE_BLOCK_ALIGNMENT == \
|
||||||
0 ? \
|
0 ? \
|
||||||
0 : FLASHPAGE_WRITE_BLOCK_SIZE - \
|
0 : \
|
||||||
sizeof(x) % FLASHPAGE_WRITE_BLOCK_SIZE)
|
FLASHPAGE_WRITE_BLOCK_ALIGNMENT - \
|
||||||
|
sizeof(x) % FLASHPAGE_WRITE_BLOCK_ALIGNMENT)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Resident key size with alignment padding
|
* @brief Resident key size with alignment padding
|
||||||
@ -70,32 +68,15 @@ extern "C" {
|
|||||||
*/
|
*/
|
||||||
#define CTAP_FLASH_STATE_SZ (sizeof(ctap_state_t) + \
|
#define CTAP_FLASH_STATE_SZ (sizeof(ctap_state_t) + \
|
||||||
CTAP_FLASH_ALIGN_PAD(ctap_state_t))
|
CTAP_FLASH_ALIGN_PAD(ctap_state_t))
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Max amount of resident keys that can be stored on device
|
* @brief Max amount of resident keys that can be stored on device
|
||||||
*/
|
*/
|
||||||
#define CTAP_FLASH_MAX_NUM_RKS ((CONFIG_FIDO2_CTAP_NUM_FLASHPAGES - 1) * \
|
#define CTAP_FLASH_MAX_NUM_RKS ((CONFIG_SLOT_AUX_LEN - CTAP_FLASH_STATE_SZ) / \
|
||||||
FLASHPAGE_SIZE / CTAP_FLASH_RK_SZ)
|
CTAP_FLASH_RK_SZ)
|
||||||
|
|
||||||
/**
|
/* SLOT_AUX_LEN must be large enough to store at least one resident key and the CTAP state */
|
||||||
* @brief Minimum flash sector size needed to hold CTAP related data
|
static_assert(CONFIG_SLOT_AUX_LEN >= (CTAP_FLASH_RK_SZ + CTAP_FLASH_STATE_SZ),
|
||||||
*
|
"SLOT_AUX_LEN is too small or not configured");
|
||||||
* This is needed to ensure that the MTD work_area buffer is big enough
|
|
||||||
*/
|
|
||||||
#define CTAP_FLASH_MIN_SECTOR_SZ _MAX(CTAP_FLASH_STATE_SZ, CTAP_FLASH_RK_SZ)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Pages per sector needed
|
|
||||||
*/
|
|
||||||
#define CTAP_FLASH_PAGES_PER_SECTOR ((CTAP_FLASH_MIN_SECTOR_SZ / FLASHPAGE_SIZE) + 1)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Offset of flashpage for storing resident keys
|
|
||||||
*
|
|
||||||
* The offset is in units of flashpages from the beginning of the flash memory
|
|
||||||
* area dedicated for storing CTAP data.
|
|
||||||
*/
|
|
||||||
#define CTAP_FLASH_RK_OFF 0x1
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Initialize memory helper
|
* @brief Initialize memory helper
|
||||||
@ -119,9 +100,11 @@ ctap_status_code_t fido2_ctap_mem_read(void *buf, uint32_t page, uint32_t offset
|
|||||||
/**
|
/**
|
||||||
* @brief Erase all flashpages containing CTAP data
|
* @brief Erase all flashpages containing CTAP data
|
||||||
*
|
*
|
||||||
|
* @param[in] state pointer to authenticator state
|
||||||
|
*
|
||||||
* @return @ref ctap_status_code_t
|
* @return @ref ctap_status_code_t
|
||||||
*/
|
*/
|
||||||
ctap_status_code_t fido2_ctap_mem_erase_flash(void);
|
ctap_status_code_t fido2_ctap_mem_erase_flash(ctap_state_t *state);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Read authenticator state from flash
|
* @brief Read authenticator state from flash
|
||||||
@ -142,21 +125,28 @@ ctap_status_code_t fido2_ctap_mem_read_state_from_flash(ctap_state_t *state);
|
|||||||
ctap_status_code_t fido2_ctap_mem_write_state_to_flash(ctap_state_t *state);
|
ctap_status_code_t fido2_ctap_mem_write_state_to_flash(ctap_state_t *state);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Find resident credential for @p rp_id_hash in flash
|
* @brief Find resident credential for @p rp_id_hash in flash memory
|
||||||
*
|
*
|
||||||
* The function stores the flash address of the next credential in @p addr.
|
* This function searches for a resident credential associated with the
|
||||||
* This allows for consecutive calls of the function in order to find all
|
* relying party identifier hash ( @p rip_id_hash ) in the flash memory region
|
||||||
* stored credentials stored for the relying party identified by
|
* used by the FIDO2 implementation. The function updates the @p absolute_offset
|
||||||
* @p rp_id_hash.
|
* parameter to indicate the total offset from the beginning of the flash
|
||||||
|
* memory region where the next credential is stored. This allows for consecutive
|
||||||
|
* calls of the function in order to find all credentials stored for the
|
||||||
|
* relying party.
|
||||||
*
|
*
|
||||||
* @param[in] key pointer to authenticator state
|
* @param[in] key pointer to authenticator state
|
||||||
* @param[in] rp_id_hash pointer to hash of rp domain string
|
* @param[in] rp_id_hash pointer to hash of rp domain string
|
||||||
* @param[in] addr pointer to address where to read from
|
* @param[in] absolute_offset pointer to a variable holding the total offset from the
|
||||||
|
* start of the flash memory region used by the FIDO2
|
||||||
|
* implementation. Updated by the function to indicate
|
||||||
|
* the location of the next credential.
|
||||||
*
|
*
|
||||||
* @return @ref ctap_status_code_t
|
* @return @ref ctap_status_code_t
|
||||||
*/
|
*/
|
||||||
ctap_status_code_t fido2_ctap_mem_read_rk_from_flash(ctap_resident_key_t *key, uint8_t *rp_id_hash,
|
ctap_status_code_t fido2_ctap_mem_read_rk_from_flash(ctap_resident_key_t *key,
|
||||||
uint32_t *addr);
|
const uint8_t *rp_id_hash,
|
||||||
|
uint32_t *absolute_offset);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Write resident credential to flash
|
* @brief Write resident credential to flash
|
||||||
|
@ -15,6 +15,9 @@ USEMODULE += embunit
|
|||||||
|
|
||||||
USEMODULE += fido2_ctap
|
USEMODULE += fido2_ctap
|
||||||
|
|
||||||
|
# Size of flash memory used to store persistent FIDO2-related data
|
||||||
|
SLOT_AUX_LEN := 0x1000
|
||||||
|
|
||||||
# Disable user presence tests
|
# Disable user presence tests
|
||||||
CFLAGS += -DCONFIG_FIDO2_CTAP_DISABLE_UP=1
|
CFLAGS += -DCONFIG_FIDO2_CTAP_DISABLE_UP=1
|
||||||
|
|
||||||
|
@ -20,6 +20,9 @@ CFLAGS += -DCONFIG_FIDO2_CTAP_DISABLE_UP=1
|
|||||||
# Disable user LED animation
|
# Disable user LED animation
|
||||||
CFLAGS += -DCONFIG_FIDO2_CTAP_DISABLE_LED=1
|
CFLAGS += -DCONFIG_FIDO2_CTAP_DISABLE_LED=1
|
||||||
|
|
||||||
|
# Size of flash memory used to store persistent FIDO2-related data
|
||||||
|
SLOT_AUX_LEN := 0x1000
|
||||||
|
|
||||||
# FIDO2 tests except for the ones requiring user presence
|
# FIDO2 tests except for the ones requiring user presence
|
||||||
#
|
#
|
||||||
# Use env -i because fido2-test has a depedency (pyscard) that needs to be
|
# Use env -i because fido2-test has a depedency (pyscard) that needs to be
|
||||||
|
Loading…
Reference in New Issue
Block a user