2017-07-05 14:54:42 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Michel Rottleuthner <michel.rottleuthner@haw-hamburg.de>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2018-06-01 13:18:00 +02:00
|
|
|
* @ingroup pkg_fatfs
|
2017-07-05 14:54:42 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of fatfs diskio interface that supports mtd driver
|
|
|
|
* based on low level disk I/O module example for FatFs by ChaN, 2016
|
|
|
|
*
|
|
|
|
* @author Michel Rottleuthner <michel.rottleuthner@haw-hamburg.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
2020-10-21 15:58:21 +02:00
|
|
|
|
2017-07-05 14:54:42 +02:00
|
|
|
#include "fatfs_diskio_mtd.h"
|
2020-11-10 17:59:03 +01:00
|
|
|
#include "ffconf.h"
|
2017-07-05 14:54:42 +02:00
|
|
|
#include "mtd.h"
|
2020-10-22 11:35:04 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2017-07-05 14:54:42 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
#if FATFS_FFCONF_OPT_FS_NORTC == 0
|
|
|
|
#include "periph/rtc.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* mtd devices for use by FatFs should be provided by the application */
|
2018-01-23 19:32:09 +01:00
|
|
|
extern mtd_dev_t *fatfs_mtd_devs[FF_VOLUMES];
|
2017-07-05 14:54:42 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief returns the status of the disk
|
|
|
|
*
|
|
|
|
* @param[in] pdrv drive number to identify the drive
|
|
|
|
*
|
|
|
|
* @return STA_NODISK if no disk exists with the given id
|
|
|
|
* @return 0 if disk is initialized
|
|
|
|
* @return STA_NOINIT if disk id exists, but disk isn't initialized
|
|
|
|
*/
|
|
|
|
DSTATUS disk_status(BYTE pdrv)
|
|
|
|
{
|
|
|
|
DEBUG("disk_status %d\n", pdrv);
|
2018-01-23 19:32:09 +01:00
|
|
|
if (pdrv >= FF_VOLUMES) {
|
2017-07-05 14:54:42 +02:00
|
|
|
return STA_NODISK;
|
|
|
|
} else if (fatfs_mtd_devs[pdrv]->driver == NULL){
|
|
|
|
return STA_NOINIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FATFS_DISKIO_DSTASTUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief initializes the disk
|
|
|
|
*
|
|
|
|
* @param[in] pdrv drive number to identify the drive
|
|
|
|
*
|
|
|
|
* @return STA_NODISK if no disk exists with the given id
|
2019-10-23 21:18:07 +02:00
|
|
|
* @return 0 if disk was initialized successfully
|
2017-07-05 14:54:42 +02:00
|
|
|
* @return STA_NOINIT if disk id exists, but couldn't be initialized
|
|
|
|
*/
|
|
|
|
DSTATUS disk_initialize(BYTE pdrv)
|
|
|
|
{
|
|
|
|
DEBUG("disk_initialize %d\n", pdrv);
|
2018-01-23 19:32:09 +01:00
|
|
|
if (pdrv >= FF_VOLUMES) {
|
2017-07-05 14:54:42 +02:00
|
|
|
return STA_NODISK;
|
|
|
|
} else if (fatfs_mtd_devs[pdrv]->driver == NULL){
|
|
|
|
return STA_NOINIT;
|
|
|
|
}
|
|
|
|
|
2022-02-09 12:10:17 +01:00
|
|
|
if (mtd_init(fatfs_mtd_devs[pdrv])) {
|
|
|
|
return STA_NOINIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t sector_size = fatfs_mtd_devs[pdrv]->page_size
|
|
|
|
* fatfs_mtd_devs[pdrv]->pages_per_sector;
|
|
|
|
if (sector_size > FF_MAX_SS) {
|
|
|
|
assert(0);
|
|
|
|
return STA_NOINIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FATFS_DISKIO_DSTASTUS_OK;
|
2017-07-05 14:54:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief reads sectors from disk
|
|
|
|
*
|
|
|
|
* @param[in] pdrv drive number to identify the drive
|
|
|
|
* @param[out] buff Data buffer to store read data
|
|
|
|
* @param[in] sector Start sector in LBA
|
|
|
|
* @param[in] count Number of sectors to read
|
|
|
|
*
|
|
|
|
* @return RES_OK if no error occurred
|
|
|
|
* @return RES_ERROR if data wasn't read completely
|
|
|
|
* @return RES_PARERR if prdv is invalid or has no mtd-driver set
|
|
|
|
*/
|
|
|
|
DRESULT disk_read(BYTE pdrv, BYTE *buff, DWORD sector, UINT count)
|
|
|
|
{
|
2020-11-10 17:59:03 +01:00
|
|
|
DEBUG("disk_read: %d, %lu, %d\n", pdrv, (long unsigned)sector, count);
|
2018-01-23 19:32:09 +01:00
|
|
|
if ((pdrv >= FF_VOLUMES) || (fatfs_mtd_devs[pdrv]->driver == NULL)) {
|
2017-07-05 14:54:42 +02:00
|
|
|
return RES_PARERR;
|
|
|
|
}
|
|
|
|
|
2020-05-06 23:52:16 +02:00
|
|
|
uint32_t sector_size = fatfs_mtd_devs[pdrv]->page_size
|
|
|
|
* fatfs_mtd_devs[pdrv]->pages_per_sector;
|
|
|
|
|
|
|
|
int res = mtd_read_page(fatfs_mtd_devs[pdrv], buff,
|
|
|
|
sector, 0, count * sector_size);
|
2017-07-05 14:54:42 +02:00
|
|
|
|
2020-04-17 19:41:43 +02:00
|
|
|
if (res != 0) {
|
|
|
|
return RES_ERROR;
|
2017-07-05 14:54:42 +02:00
|
|
|
}
|
2020-04-17 19:41:43 +02:00
|
|
|
return RES_OK;
|
2017-07-05 14:54:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief writes sectors to disk
|
|
|
|
*
|
|
|
|
* @param[in] pdrv Physical drive nmuber to identify the drive
|
|
|
|
* @param[in] buff Data to be written
|
|
|
|
* @param[in] sector Start sector in LBA
|
|
|
|
* @param[in] count Number of sectors to write
|
|
|
|
*
|
|
|
|
* @return RES_OK if no error occurred
|
|
|
|
* @return RES_ERROR if data wasn't written completely
|
|
|
|
* @return RES_PARERR if prdv is invalid or has no mtd-driver set
|
|
|
|
*/
|
|
|
|
DRESULT disk_write(BYTE pdrv, const BYTE *buff, DWORD sector, UINT count)
|
|
|
|
{
|
2023-12-13 22:20:50 +01:00
|
|
|
mtd_dev_t *mtd = fatfs_mtd_devs[pdrv];
|
2020-11-10 17:59:03 +01:00
|
|
|
DEBUG("disk_write: %d, %lu, %d\n", pdrv, (long unsigned)sector, count);
|
2023-12-13 22:20:50 +01:00
|
|
|
if ((pdrv >= FF_VOLUMES) || (mtd->driver == NULL)) {
|
2017-07-05 14:54:42 +02:00
|
|
|
return RES_PARERR;
|
|
|
|
}
|
|
|
|
|
2023-12-13 22:20:50 +01:00
|
|
|
int res = mtd_write_sector(mtd, buff, sector, count);
|
2020-04-17 19:41:43 +02:00
|
|
|
if (res != 0) {
|
|
|
|
return RES_ERROR;
|
2017-07-05 14:54:42 +02:00
|
|
|
}
|
2020-04-17 19:41:43 +02:00
|
|
|
return RES_OK;
|
2017-07-05 14:54:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief perform miscellaneous low-level control functions
|
|
|
|
*
|
2018-01-23 19:32:09 +01:00
|
|
|
* @param[in] pdrv Physical drive nmuber (0..FF_VOLUMES-1)
|
2017-07-05 14:54:42 +02:00
|
|
|
* @param[in out] cmd Control code
|
|
|
|
* @param[in] sector Buffer to send/receive control data
|
|
|
|
*
|
|
|
|
* @return RES_OK if no error occurred
|
|
|
|
* @return RES_PARERR if cmd is unknown or
|
|
|
|
* prdv is either invalid or has no mtd-driver set
|
|
|
|
*/
|
|
|
|
DRESULT disk_ioctl(BYTE pdrv, BYTE cmd, void *buff)
|
|
|
|
{
|
2020-07-02 19:11:05 +02:00
|
|
|
(void)buff;
|
|
|
|
|
2018-01-23 19:32:09 +01:00
|
|
|
if ((pdrv >= FF_VOLUMES) || (fatfs_mtd_devs[pdrv]->driver == NULL)) {
|
2017-07-05 14:54:42 +02:00
|
|
|
return RES_PARERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (cmd) {
|
2018-01-23 19:32:09 +01:00
|
|
|
#if (FF_FS_READONLY == 0)
|
2017-07-05 14:54:42 +02:00
|
|
|
case CTRL_SYNC:
|
|
|
|
/* r/w is always finished within r/w-functions of mtd */
|
|
|
|
return RES_OK;
|
|
|
|
#endif
|
|
|
|
|
2018-01-23 19:32:09 +01:00
|
|
|
#if (FF_USE_MKFS == 1)
|
2017-07-05 14:54:42 +02:00
|
|
|
case GET_SECTOR_COUNT:
|
|
|
|
*(DWORD *)buff = fatfs_mtd_devs[pdrv]->sector_count;
|
|
|
|
return RES_OK;
|
|
|
|
|
|
|
|
/*Erase block size in number of sectors (1 to 32768 in power of 2).
|
|
|
|
Return 1 if the erase block size is unknown. */
|
|
|
|
case GET_BLOCK_SIZE:
|
|
|
|
*(DWORD *)buff = fatfs_mtd_devs[pdrv]->pages_per_sector;
|
|
|
|
return RES_OK;
|
|
|
|
#endif
|
|
|
|
|
2018-01-23 19:32:09 +01:00
|
|
|
#if (FF_MAX_SS != FF_MIN_SS)
|
2017-07-05 14:54:42 +02:00
|
|
|
case GET_SECTOR_SIZE:
|
|
|
|
*(DWORD *)buff = fatfs_mtd_devs[pdrv]->page_size;
|
|
|
|
return RES_OK;
|
|
|
|
#endif
|
|
|
|
|
2018-01-23 19:32:09 +01:00
|
|
|
#if (FF_USE_TRIM == 1)
|
2017-07-05 14:54:42 +02:00
|
|
|
case CTRL_TRIM:
|
|
|
|
return RES_OK;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return RES_PARERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if FATFS_FFCONF_OPT_FS_NORTC == 0
|
|
|
|
/**
|
|
|
|
* @brief store local time in a 32-bit bitfiled
|
|
|
|
*
|
|
|
|
* @return timestamp as bitfield
|
|
|
|
*/
|
|
|
|
DWORD get_fattime(void)
|
|
|
|
{
|
|
|
|
struct tm time;
|
|
|
|
|
|
|
|
rtc_get_time(&time);
|
|
|
|
|
|
|
|
/* bit 31:25 Year origin from 1980 (0..127) */
|
|
|
|
uint8_t year = time.tm_year + RTC_YEAR_OFFSET - FATFS_YEAR_OFFSET;
|
|
|
|
uint8_t month = time.tm_mon + 1; /* bit 24:21 month (1..12) */
|
2020-04-03 11:27:10 +02:00
|
|
|
uint8_t day_of_month = time.tm_mday; /* bit 20:16 day (1..31) */
|
2017-07-05 14:54:42 +02:00
|
|
|
uint8_t hour = time.tm_hour; /* bit 15:11 hour (0..23) */
|
|
|
|
uint8_t minute = time.tm_min; /* bit 10:5 minute (0..59) */
|
|
|
|
uint8_t second = (time.tm_sec / 2); /* bit 4:0 second/2 (0..29) */
|
|
|
|
|
2022-01-06 12:32:16 +01:00
|
|
|
return (DWORD)year << FATFS_DISKIO_FATTIME_YEAR_OFFS |
|
|
|
|
(DWORD)month << FATFS_DISKIO_FATTIME_MON_OFFS |
|
|
|
|
(DWORD)day_of_month << FATFS_DISKIO_FATTIME_DAY_OFFS |
|
2017-07-05 14:54:42 +02:00
|
|
|
hour << FATFS_DISKIO_FATTIME_HH_OFFS |
|
|
|
|
minute << FATFS_DISKIO_FATTIME_MM_OFFS |
|
|
|
|
second;
|
|
|
|
}
|
|
|
|
#endif
|