2017-08-09 11:30:41 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 HAW-Hamburg
|
|
|
|
*
|
|
|
|
* 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 drivers_mtd_sdcard
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Driver for using sdcard_spi via mtd interface
|
|
|
|
*
|
|
|
|
* @author Michel Rottleuthner <michel.rottleuthner@haw-hamburg.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
#include "mtd.h"
|
|
|
|
#include "mtd_sdcard.h"
|
|
|
|
#include "sdcard_spi.h"
|
|
|
|
#include "sdcard_spi_internal.h"
|
2020-04-23 16:39:50 +02:00
|
|
|
#include "kernel_defines.h"
|
2017-08-09 11:30:41 +02:00
|
|
|
|
2018-08-01 23:06:00 +02:00
|
|
|
#include <inttypes.h>
|
2017-08-09 11:30:41 +02:00
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
static int mtd_sdcard_init(mtd_dev_t *dev)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_sdcard_init\n");
|
|
|
|
mtd_sdcard_t *mtd_sd = (mtd_sdcard_t*)dev;
|
|
|
|
if ((mtd_sd->sd_card->init_done == true) ||
|
|
|
|
(sdcard_spi_init(mtd_sd->sd_card, mtd_sd->params) == 0)) {
|
|
|
|
/* erasing whole sectors is handled internally by the card so you can
|
|
|
|
delete single blocks (i.e. pages) */
|
|
|
|
dev->pages_per_sector = 1;
|
|
|
|
dev->sector_count = sdcard_spi_get_sector_count(mtd_sd->sd_card);
|
|
|
|
|
|
|
|
/* sdcard_spi always uses the fixed block size of SD-HC cards */
|
|
|
|
dev->page_size = SD_HC_BLOCK_SIZE;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mtd_sdcard_read(mtd_dev_t *dev, void *buff, uint32_t addr,
|
|
|
|
uint32_t size)
|
|
|
|
{
|
2018-08-01 23:06:00 +02:00
|
|
|
DEBUG("mtd_sdcard_read: addr:%" PRIu32 " size:%" PRIu32 "\n", addr, size);
|
2017-08-09 11:30:41 +02:00
|
|
|
mtd_sdcard_t *mtd_sd = (mtd_sdcard_t*)dev;
|
|
|
|
sd_rw_response_t err;
|
2020-04-17 19:41:43 +02:00
|
|
|
sdcard_spi_read_blocks(mtd_sd->sd_card, addr / SD_HC_BLOCK_SIZE,
|
|
|
|
buff, SD_HC_BLOCK_SIZE,
|
|
|
|
size / SD_HC_BLOCK_SIZE, &err);
|
2017-08-09 11:30:41 +02:00
|
|
|
|
|
|
|
if (err == SD_RW_OK) {
|
2020-04-17 19:41:43 +02:00
|
|
|
return 0;
|
2017-08-09 11:30:41 +02:00
|
|
|
}
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2020-05-06 23:49:29 +02:00
|
|
|
static int mtd_sdcard_read_page(mtd_dev_t *dev, void *buff, uint32_t page,
|
|
|
|
uint32_t offset, uint32_t size)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_sdcard_read_page: page:%" PRIu32 " offset:%" PRIu32 " size:%" PRIu32 "\n",
|
|
|
|
page, offset, size);
|
|
|
|
|
|
|
|
if (offset) {
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
mtd_sdcard_t *mtd_sd = (mtd_sdcard_t*)dev;
|
|
|
|
sd_rw_response_t err;
|
|
|
|
int res = sdcard_spi_read_blocks(mtd_sd->sd_card, page,
|
|
|
|
buff, SD_HC_BLOCK_SIZE,
|
|
|
|
size / SD_HC_BLOCK_SIZE, &err);
|
|
|
|
|
|
|
|
if (err != SD_RW_OK) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
return res * SD_HC_BLOCK_SIZE;
|
|
|
|
}
|
|
|
|
|
2017-08-09 11:30:41 +02:00
|
|
|
static int mtd_sdcard_write(mtd_dev_t *dev, const void *buff, uint32_t addr,
|
|
|
|
uint32_t size)
|
|
|
|
{
|
2018-08-01 23:06:00 +02:00
|
|
|
DEBUG("mtd_sdcard_write: addr:%" PRIu32 " size:%" PRIu32 "\n", addr, size);
|
2017-08-09 11:30:41 +02:00
|
|
|
mtd_sdcard_t *mtd_sd = (mtd_sdcard_t*)dev;
|
|
|
|
sd_rw_response_t err;
|
2020-04-17 19:41:43 +02:00
|
|
|
sdcard_spi_write_blocks(mtd_sd->sd_card, addr / SD_HC_BLOCK_SIZE,
|
|
|
|
buff, SD_HC_BLOCK_SIZE,
|
|
|
|
size / SD_HC_BLOCK_SIZE, &err);
|
2017-08-09 11:30:41 +02:00
|
|
|
|
|
|
|
if (err == SD_RW_OK) {
|
2020-04-17 19:41:43 +02:00
|
|
|
return 0;
|
2017-08-09 11:30:41 +02:00
|
|
|
}
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2020-05-06 23:49:29 +02:00
|
|
|
static int mtd_sdcard_write_page(mtd_dev_t *dev, const void *buff, uint32_t page,
|
|
|
|
uint32_t offset, uint32_t size)
|
|
|
|
{
|
|
|
|
DEBUG("mtd_sdcard_write_page: page:%" PRIu32 " offset:%" PRIu32 " size:%" PRIu32 "\n",
|
|
|
|
page, offset, size);
|
|
|
|
|
|
|
|
if (offset) {
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
mtd_sdcard_t *mtd_sd = (mtd_sdcard_t*)dev;
|
|
|
|
sd_rw_response_t err;
|
|
|
|
int res = sdcard_spi_write_blocks(mtd_sd->sd_card, page,
|
|
|
|
buff, SD_HC_BLOCK_SIZE,
|
|
|
|
size / SD_HC_BLOCK_SIZE, &err);
|
|
|
|
|
|
|
|
if (err != SD_RW_OK) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
return res * SD_HC_BLOCK_SIZE;
|
|
|
|
}
|
|
|
|
|
2017-08-09 11:30:41 +02:00
|
|
|
static int mtd_sdcard_erase(mtd_dev_t *dev,
|
|
|
|
uint32_t addr,
|
|
|
|
uint32_t size)
|
|
|
|
{
|
2018-08-01 23:06:00 +02:00
|
|
|
DEBUG("mtd_sdcard_erase: addr:%" PRIu32 " size:%" PRIu32 "\n", addr, size);
|
2017-08-09 11:30:41 +02:00
|
|
|
(void)dev;
|
|
|
|
(void)addr;
|
|
|
|
(void)size;
|
|
|
|
|
2020-04-23 16:39:50 +02:00
|
|
|
if (!IS_ACTIVE(CONFIG_MTD_SDCARD_ERASE)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return -ENOTSUP; /* explicit erase currently not supported */
|
|
|
|
}
|
2017-08-09 11:30:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static int mtd_sdcard_power(mtd_dev_t *dev, enum mtd_power_state power)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
(void)power;
|
|
|
|
|
|
|
|
/* TODO: implement power down of sdcard in sdcard_spi
|
|
|
|
(make use of sdcard_spi_params_t.power pin) */
|
|
|
|
return -ENOTSUP; /* currently not supported */
|
|
|
|
}
|
2020-05-06 23:49:29 +02:00
|
|
|
|
|
|
|
const mtd_desc_t mtd_sdcard_driver = {
|
|
|
|
.init = mtd_sdcard_init,
|
|
|
|
.read = mtd_sdcard_read,
|
|
|
|
.read_page = mtd_sdcard_read_page,
|
|
|
|
.write = mtd_sdcard_write,
|
|
|
|
.write_page = mtd_sdcard_write_page,
|
|
|
|
.erase = mtd_sdcard_erase,
|
|
|
|
.power = mtd_sdcard_power,
|
|
|
|
};
|