You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

360 lines
8.1 KiB
C

/**
* \file
*
* \brief CTRL_ACCESS interface for common SD/MMC stack
*
* Copyright (c) 2014-2018 Microchip Technology Inc. and its subsidiaries.
*
* \asf_license_start
*
* \page License
*
* Subject to your compliance with these terms, you may use Microchip
* software and any derivatives exclusively with Microchip products.
* It is your responsibility to comply with third party license terms applicable
* to your use of third party software (including open source software) that
* may accompany Microchip software.
*
* THIS SOFTWARE IS SUPPLIED BY MICROCHIP "AS IS". NO WARRANTIES,
* WHETHER EXPRESS, IMPLIED OR STATUTORY, APPLY TO THIS SOFTWARE,
* INCLUDING ANY IMPLIED WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY,
* AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT WILL MICROCHIP BE
* LIABLE FOR ANY INDIRECT, SPECIAL, PUNITIVE, INCIDENTAL OR CONSEQUENTIAL
* LOSS, DAMAGE, COST OR EXPENSE OF ANY KIND WHATSOEVER RELATED TO THE
* SOFTWARE, HOWEVER CAUSED, EVEN IF MICROCHIP HAS BEEN ADVISED OF THE
* POSSIBILITY OR THE DAMAGES ARE FORESEEABLE. TO THE FULLEST EXTENT
* ALLOWED BY LAW, MICROCHIP'S TOTAL LIABILITY ON ALL CLAIMS IN ANY WAY
* RELATED TO THIS SOFTWARE WILL NOT EXCEED THE AMOUNT OF FEES, IF ANY,
* THAT YOU HAVE PAID DIRECTLY TO MICROCHIP FOR THIS SOFTWARE.
*
* \asf_license_stop
*
*/
/*
* Support and FAQ: visit <a href="https://www.microchip.com/support/">Microchip Support</a>
*/
#include "conf_access.h"
#if (SD_MMC_0_MEM == ENABLE) || (SD_MMC_1_MEM == ENABLE)
#include "conf_sd_mmc.h"
#include "sd_mmc.h"
#include "sd_mmc_mem.h"
/**
* \ingroup sd_mmc_stack_mem
* \defgroup sd_mmc_stack_mem_internal Implementation of SD/MMC Memory
* @{
*/
/**
* \name Control Interface
* @{
*/
static bool sd_mmc_ejected[2] = {false, false};
Ctrl_status sd_mmc_test_unit_ready(uint8_t slot)
{
switch (sd_mmc_check(slot))
{
case SD_MMC_OK:
if (sd_mmc_ejected[slot]) {
return CTRL_NO_PRESENT;
}
if (sd_mmc_get_type(slot) & (CARD_TYPE_SD | CARD_TYPE_MMC)) {
return CTRL_GOOD;
}
// It is not a memory card
return CTRL_NO_PRESENT;
case SD_MMC_INIT_ONGOING:
return CTRL_BUSY;
case SD_MMC_ERR_NO_CARD:
sd_mmc_ejected[slot] = false;
return CTRL_NO_PRESENT;
default:
return CTRL_FAIL;
}
}
Ctrl_status sd_mmc_test_unit_ready_0(void)
{
return sd_mmc_test_unit_ready(0);
}
Ctrl_status sd_mmc_test_unit_ready_1(void)
{
return sd_mmc_test_unit_ready(1);
}
Ctrl_status sd_mmc_read_capacity(uint8_t slot, uint32_t *nb_sector)
{
// Return last sector address (-1)
*nb_sector = (sd_mmc_get_capacity(slot) * 2) - 1;
return sd_mmc_test_unit_ready(slot);
}
Ctrl_status sd_mmc_read_capacity_0(uint32_t *nb_sector)
{
return sd_mmc_read_capacity(0, nb_sector);
}
Ctrl_status sd_mmc_read_capacity_1(uint32_t *nb_sector)
{
return sd_mmc_read_capacity(1, nb_sector);
}
bool sd_mmc_unload(uint8_t slot, bool unload)
{
sd_mmc_ejected[slot] = unload;
return true;
}
bool sd_mmc_unload_0(bool unload)
{
return sd_mmc_unload(0, unload);
}
bool sd_mmc_unload_1(bool unload)
{
return sd_mmc_unload(1, unload);
}
bool sd_mmc_wr_protect(uint8_t slot)
{
return sd_mmc_is_write_protected(slot);
}
bool sd_mmc_wr_protect_0(void)
{
return sd_mmc_wr_protect(0);
}
bool sd_mmc_wr_protect_1(void)
{
return sd_mmc_wr_protect(1);
}
bool sd_mmc_removal(uint8_t slot)
{
UNUSED(slot);
return true;
}
bool sd_mmc_removal_0(void)
{
return sd_mmc_removal(0);
}
bool sd_mmc_removal_1(void)
{
return sd_mmc_removal(1);
}
//! @}
#if ACCESS_USB == true
/**
* \name MEM <-> USB Interface
* @{
*/
#include "udi_msc.h"
COMPILER_WORD_ALIGNED
uint8_t sector_buf_0[SD_MMC_BLOCK_SIZE];
COMPILER_WORD_ALIGNED
uint8_t sector_buf_1[SD_MMC_BLOCK_SIZE];
Ctrl_status sd_mmc_usb_read_10(uint8_t slot, uint32_t addr, uint16_t nb_sector)
{
bool b_first_step = true;
uint16_t nb_step;
switch (sd_mmc_init_read_blocks(slot, addr, nb_sector)) {
case SD_MMC_OK:
break;
case SD_MMC_ERR_NO_CARD:
return CTRL_NO_PRESENT;
default:
return CTRL_FAIL;
}
// Pipeline the 2 transfer in order to speed-up the performances
nb_step = nb_sector + 1;
while (nb_step--) {
if (nb_step) { // Skip last step
// MCI -> RAM
if (SD_MMC_OK != sd_mmc_start_read_blocks(((nb_step % 2) == 0) ?
sector_buf_0 : sector_buf_1, 1)) {
return CTRL_FAIL;
}
}
if (!b_first_step) { // Skip first step
// RAM -> USB
if (!udi_msc_trans_block(true,
((nb_step % 2) == 0) ?
sector_buf_1 : sector_buf_0,
SD_MMC_BLOCK_SIZE,
NULL)) {
if (!b_first_step) {
sd_mmc_wait_end_of_read_blocks(true);
}
return CTRL_FAIL;
}
} else {
b_first_step = false;
}
if (nb_step) { // Skip last step
if (SD_MMC_OK != sd_mmc_wait_end_of_read_blocks(false)) {
return CTRL_FAIL;
}
}
b_first_step = false;
}
return CTRL_GOOD;
}
Ctrl_status sd_mmc_usb_read_10_0(uint32_t addr, uint16_t nb_sector)
{
return sd_mmc_usb_read_10(0, addr, nb_sector);
}
Ctrl_status sd_mmc_usb_read_10_1(uint32_t addr, uint16_t nb_sector)
{
return sd_mmc_usb_read_10(1, addr, nb_sector);
}
Ctrl_status sd_mmc_usb_write_10(uint8_t slot, uint32_t addr, uint16_t nb_sector)
{
bool b_first_step = true;
uint16_t nb_step;
switch (sd_mmc_init_write_blocks(slot, addr, nb_sector)) {
case SD_MMC_OK:
break;
case SD_MMC_ERR_NO_CARD:
return CTRL_NO_PRESENT;
default:
return CTRL_FAIL;
}
// Pipeline the 2 transfer in order to speed-up the performances
nb_step = nb_sector + 1;
while (nb_step--) {
if (!b_first_step) { // Skip first step
// RAM -> MCI
if (SD_MMC_OK != sd_mmc_start_write_blocks(((nb_step % 2) == 0) ?
sector_buf_0 : sector_buf_1, 1)) {
return CTRL_FAIL;
}
}
if (nb_step) { // Skip last step
// USB -> RAM
if (!udi_msc_trans_block(false,
((nb_step % 2) == 0) ?
sector_buf_1 : sector_buf_0,
SD_MMC_BLOCK_SIZE,
NULL)) {
if (!b_first_step) {
sd_mmc_wait_end_of_write_blocks(true);
}
return CTRL_FAIL;
}
}
if (!b_first_step) { // Skip first step
if (SD_MMC_OK != sd_mmc_wait_end_of_write_blocks(false)) {
return CTRL_FAIL;
}
} else {
b_first_step = false;
}
}
return CTRL_GOOD;
}
Ctrl_status sd_mmc_usb_write_10_0(uint32_t addr, uint16_t nb_sector)
{
return sd_mmc_usb_write_10(0, addr, nb_sector);
}
Ctrl_status sd_mmc_usb_write_10_1(uint32_t addr, uint16_t nb_sector)
{
return sd_mmc_usb_write_10(1, addr, nb_sector);
}
//! @}
#endif // ACCESS_USB == true
#if ACCESS_MEM_TO_RAM == true
/**
* \name MEM <-> RAM Interface
* @{
*/
Ctrl_status sd_mmc_mem_2_ram(uint8_t slot, uint32_t addr, void *ram)
{
switch (sd_mmc_init_read_blocks(slot, addr, 1)) {
case SD_MMC_OK:
break;
case SD_MMC_ERR_NO_CARD:
return CTRL_NO_PRESENT;
default:
return CTRL_FAIL;
}
if (SD_MMC_OK != sd_mmc_start_read_blocks(ram, 1)) {
return CTRL_FAIL;
}
if (SD_MMC_OK != sd_mmc_wait_end_of_read_blocks(false)) {
return CTRL_FAIL;
}
return CTRL_GOOD;
}
Ctrl_status sd_mmc_mem_2_ram_0(uint32_t addr, void *ram)
{
return sd_mmc_mem_2_ram(0, addr, ram);
}
Ctrl_status sd_mmc_mem_2_ram_1(uint32_t addr, void *ram)
{
return sd_mmc_mem_2_ram(1, addr, ram);
}
Ctrl_status sd_mmc_ram_2_mem(uint8_t slot, uint32_t addr, const void *ram)
{
switch (sd_mmc_init_write_blocks(slot, addr, 1)) {
case SD_MMC_OK:
break;
case SD_MMC_ERR_NO_CARD:
return CTRL_NO_PRESENT;
default:
return CTRL_FAIL;
}
if (SD_MMC_OK != sd_mmc_start_write_blocks(ram, 1)) {
return CTRL_FAIL;
}
if (SD_MMC_OK != sd_mmc_wait_end_of_write_blocks(false)) {
return CTRL_FAIL;
}
return CTRL_GOOD;
}
Ctrl_status sd_mmc_ram_2_mem_0(uint32_t addr, const void *ram)
{
return sd_mmc_ram_2_mem(0, addr, ram);
}
Ctrl_status sd_mmc_ram_2_mem_1(uint32_t addr, const void *ram)
{
return sd_mmc_ram_2_mem(1, addr, ram);
}
//! @}
//! @}
#endif // ACCESS_MEM_TO_RAM == true
#endif // SD_MMC_0_MEM == ENABLE || SD_MMC_1_MEM == ENABLE