1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/tests/mtd_mapper/main.c
Marian Buschsieweke 86fdbd7054
core/lib: Add macros/utils.h header
The macros CONCAT(), MIN(), and MAX() are defined over and over again in
RIOT's code base. This de-duplicates the code by moving the macros to a
common place.
2023-01-07 09:47:44 +01:00

390 lines
9.3 KiB
C

/*
* Copyright (C) 2020 Koen Zandberg <koen@bergzand.net>
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup tests
* @{
*
* @file
* @brief mtd_mapper module test
*
* @author Koen Zandberg <koen@bergzand.net>
*
* @}
*/
#include <stdint.h>
#include <errno.h>
#include <string.h>
#include "embUnit.h"
#include "macros/utils.h"
#include "mtd.h"
#include "mtd_mapper.h"
/* Test mock object implementing a simple RAM-based mtd */
#ifndef SECTOR_COUNT
#define SECTOR_COUNT 16
#endif
#ifndef PAGE_PER_SECTOR
#define PAGE_PER_SECTOR 4
#endif
#ifndef PAGE_SIZE
#define PAGE_SIZE 64
#endif
#ifndef WRITE_SIZE
#define WRITE_SIZE 4
#endif
#define MEMORY_PAGE_COUNT PAGE_PER_SECTOR * SECTOR_COUNT
#define MEMORY_SIZE PAGE_SIZE * MEMORY_PAGE_COUNT
#define REGION_FLASH_SIZE MEMORY_SIZE / 2
#define REGION_PAGE_COUNT MEMORY_PAGE_COUNT / 2
static uint8_t _dummy_memory[MEMORY_SIZE];
static uint8_t _buffer[PAGE_SIZE];
static int _init(mtd_dev_t *dev)
{
(void)dev;
return 0;
}
static int _read(mtd_dev_t *dev, void *buff, uint32_t addr, uint32_t size)
{
(void)dev;
if (addr + size > sizeof(_dummy_memory)) {
return -EOVERFLOW;
}
memcpy(buff, _dummy_memory + addr, size);
return 0;
}
static int _read_page(mtd_dev_t *dev, void *buff, uint32_t page, uint32_t offset, uint32_t size)
{
uint32_t addr = page * dev->page_size + offset;
if (page >= dev->sector_count * dev->pages_per_sector) {
return -EOVERFLOW;
}
if (offset > dev->page_size) {
return -EOVERFLOW;
}
size = MIN(dev->page_size - offset, size);
memcpy(buff, _dummy_memory + addr, size);
return size;
}
static int _write(mtd_dev_t *dev, const void *buff, uint32_t addr,
uint32_t size)
{
(void)dev;
if (addr + size > sizeof(_dummy_memory)) {
return -EOVERFLOW;
}
if (size > PAGE_SIZE) {
return -EOVERFLOW;
}
memcpy(_dummy_memory + addr, buff, size);
return 0;
}
static int _write_page(mtd_dev_t *dev, const void *buff, uint32_t page, uint32_t offset, uint32_t size)
{
uint32_t addr = page * dev->page_size + offset;
if (page >= dev->sector_count * dev->pages_per_sector) {
return -EOVERFLOW;
}
if (offset > dev->page_size) {
return -EOVERFLOW;
}
size = MIN(dev->page_size - offset, size);
memcpy(_dummy_memory + addr, buff, size);
return size;
}
static int _erase(mtd_dev_t *dev, uint32_t addr, uint32_t size)
{
(void)dev;
if (size % (PAGE_PER_SECTOR * PAGE_SIZE) != 0) {
return -EOVERFLOW;
}
if (addr % (PAGE_PER_SECTOR * PAGE_SIZE) != 0) {
return -EOVERFLOW;
}
if (addr + size > sizeof(_dummy_memory)) {
return -EOVERFLOW;
}
memset(_dummy_memory + addr, 0xff, size);
return 0;
}
static int _erase_sector(mtd_dev_t *dev, uint32_t sector, uint32_t count)
{
uint32_t addr = sector * dev->page_size * dev->pages_per_sector;
if (sector + count > dev->sector_count) {
return -EOVERFLOW;
}
memset(_dummy_memory + addr, 0xff,
count * dev->page_size * dev->pages_per_sector);
return 0;
}
static int _power(mtd_dev_t *dev, enum mtd_power_state power)
{
(void)dev;
(void)power;
return 0;
}
static const mtd_desc_t driver = {
.init = _init,
.read = _read,
.write = _write,
.erase = _erase,
.power = _power,
.read_page = _read_page,
.write_page = _write_page,
.erase_sector = _erase_sector,
};
static mtd_dev_t dev = {
.driver = &driver,
.sector_count = SECTOR_COUNT,
.pages_per_sector = PAGE_PER_SECTOR,
.page_size = PAGE_SIZE,
.write_size = WRITE_SIZE,
};
static mtd_mapper_parent_t _parent = MTD_PARENT_INIT(&dev);
static mtd_mapper_region_t _region_a = {
.mtd = {
.driver = &mtd_mapper_driver,
.sector_count = SECTOR_COUNT / 2,
},
.parent = &_parent,
.sector = 0,
};
static mtd_mapper_region_t _region_b = {
.mtd = {
.driver = &mtd_mapper_driver,
.sector_count = SECTOR_COUNT / 2,
},
.parent = &_parent,
.sector = SECTOR_COUNT / 2,
};
static mtd_dev_t *_dev_a = &_region_a.mtd;
static mtd_dev_t *_dev_b = &_region_b.mtd;
static void _test_mem(uint8_t *buffer, size_t len, uint8_t expected)
{
for (size_t i = 0; i < len; i++) {
TEST_ASSERT_EQUAL_INT(expected, buffer[i]);
}
}
static void test_mtd_init(void)
{
int ret = mtd_init(_dev_a);
TEST_ASSERT_EQUAL_INT(0, ret);
ret = mtd_init(_dev_b);
TEST_ASSERT_EQUAL_INT(0, ret);
}
static void test_mtd_erase(void)
{
/* Erase first region */
int ret = mtd_erase(_dev_a, 0, REGION_FLASH_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
/* Erase second region */
ret = mtd_erase(_dev_b, 0, REGION_FLASH_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
ret = mtd_erase(_dev_a, REGION_FLASH_SIZE, 1);
TEST_ASSERT_EQUAL_INT(-EOVERFLOW, ret);
ret = mtd_erase(_dev_b, REGION_FLASH_SIZE, 1);
TEST_ASSERT_EQUAL_INT(-EOVERFLOW, ret);
}
static void test_mtd_read(void)
{
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_read(_dev_a, _buffer, i, PAGE_SIZE);
_test_mem(_buffer, PAGE_SIZE, 0xff);
}
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_read(_dev_b, _buffer, i, PAGE_SIZE);
_test_mem(_buffer, PAGE_SIZE, 0xff);
}
}
static void test_mtd_read_page(void)
{
int ret;
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_read_page(_dev_a, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
_test_mem(_buffer, PAGE_SIZE, 0xff);
}
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_read_page(_dev_b, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
_test_mem(_buffer, PAGE_SIZE, 0xff);
}
ret = mtd_read_page(_dev_b, _buffer, REGION_PAGE_COUNT, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(-EOVERFLOW, ret);
}
static void test_mtd_write(void)
{
static const uint8_t test_val_a = 0xAA;
memset(_buffer, test_val_a, PAGE_SIZE);
/* Write first region */
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_write(_dev_a, _buffer, i, PAGE_SIZE);
}
/* Check second region, should still be 0xFF */
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_read(_dev_b, _buffer, i, PAGE_SIZE);
_test_mem(_buffer, PAGE_SIZE, 0xFF);
}
static const uint8_t test_val_b = 0xBB;
memset(_buffer, test_val_b, PAGE_SIZE);
/* Write second region */
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_write(_dev_b, _buffer, i, PAGE_SIZE);
}
/* Check second region after write, should now be 0xBB */
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_read(_dev_b, _buffer, i, PAGE_SIZE);
_test_mem(_buffer, PAGE_SIZE, 0xBB);
}
/* Check first region, should still be 0xAA */
for (uint32_t i = 0; i < REGION_FLASH_SIZE; i += PAGE_SIZE) {
mtd_read(_dev_a, _buffer, i, PAGE_SIZE);
_test_mem(_buffer, PAGE_SIZE, 0xAA);
}
}
static void test_mtd_write_page(void)
{
static const uint8_t test_val_a = 0xAA;
int ret;
memset(_buffer, test_val_a, PAGE_SIZE);
/* Write first region */
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_write_page(_dev_a, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
}
/* Check second region, should still be 0xFF */
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_read_page(_dev_b, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
_test_mem(_buffer, PAGE_SIZE, 0xFF);
}
static const uint8_t test_val_b = 0xBB;
memset(_buffer, test_val_b, PAGE_SIZE);
/* Write second region */
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_write_page(_dev_b, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
}
/* Check second region after write, should now be 0xBB */
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_read_page(_dev_b, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
_test_mem(_buffer, PAGE_SIZE, 0xBB);
}
/* Check first region, should still be 0xAA */
for (uint32_t i = 0; i < REGION_PAGE_COUNT; i += 1) {
ret = mtd_read_page(_dev_a, _buffer, i, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(0, ret);
_test_mem(_buffer, PAGE_SIZE, 0xAA);
}
ret = mtd_write_page(_dev_b, _buffer, REGION_PAGE_COUNT, 0, PAGE_SIZE);
TEST_ASSERT_EQUAL_INT(-EOVERFLOW, ret);
}
static void set_up(void)
{
memset(_dummy_memory, 0xff, sizeof(_dummy_memory));
}
Test *tests_mtd_mapper_tests(void)
{
EMB_UNIT_TESTFIXTURES(fixtures) {
new_TestFixture(test_mtd_init),
new_TestFixture(test_mtd_erase),
new_TestFixture(test_mtd_read),
new_TestFixture(test_mtd_read_page),
new_TestFixture(test_mtd_write),
new_TestFixture(test_mtd_write_page),
};
EMB_UNIT_TESTCALLER(mtd_flashpage_tests, set_up, NULL, fixtures);
return (Test *)&mtd_flashpage_tests;
}
int main(void)
{
TESTS_START();
TESTS_RUN(tests_mtd_mapper_tests());
TESTS_END();
return 0;
}