1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/notrynohigh-BabyOS

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
Это зеркальный репозиторий, синхронизируется ежедневно с исходного репозитория.
Клонировать/Скачать
b_drv_sd.c 14 КБ
Копировать Редактировать Web IDE Исходные данные Просмотреть построчно История
notrynohigh Отправлено 7 месяцев назад 3d29428
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
/**
*!
* \file b_drv_sd.c
* \version v0.0.1
* \date 2020/06/01
* \author Bean(notrynohigh@outlook.com)
*******************************************************************************
* @attention
*
* Copyright (c) 2020 Bean
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*******************************************************************************
*/
/*Includes ----------------------------------------------*/
#include "drivers/inc/b_drv_sd.h"
#include "utils/inc/b_util_log.h"
/**
* \addtogroup B_DRIVER
* \{
*/
/**
* \addtogroup SD
* \{
*/
/**
* \defgroup SD_Private_TypesDefinitions
* \{
*/
/**
* \}
*/
/**
* \defgroup SD_Private_Macros
* \{
*/
/**
* \}
*/
/**
* \defgroup SD_Private_Defines
* \{
*/
#define DRIVER_NAME SD
/**
* \}
*/
/**
* \defgroup SD_Private_Variables
* \{
*/
bDRIVER_HALIF_TABLE(bSD_HalIf_t, DRIVER_NAME);
/**
* \}
*/
/**
* \defgroup SD_Private_FunctionPrototypes
* \{
*/
/**
* \}
*/
/**
* \defgroup SD_Private_Functions
* \{
*/
/*****************************driver interface***************************/
static int _bSD_WaitReady(bDriverInterface_t *pdrv)
{
uint32_t tick = bHalGetSysTick();
uint8_t tmp = 0;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
for (;;)
{
tmp = bHalSpiTransfer(&_if->_if._spi, 0xff);
if (tmp == 0xff)
{
return 0;
}
if (TICK_DIFF_BIT32(tick, bHalGetSysTick()) > 500)
{
break;
}
}
return -1;
}
static void _bSD_SendDump(bDriverInterface_t *pdrv, uint8_t n)
{
uint8_t tmp = 0xff;
uint8_t i = 0;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
for (i = 0; i < n; i++)
{
bHalSpiSend(&_if->_if._spi, &tmp, 1);
}
}
static int _bSD_PowerON(bDriverInterface_t *pdrv)
{
uint8_t tmp = 0xff;
uint8_t cmd[6] = {CMD0, 0, 0, 0, 0, 0X95};
uint32_t cnt;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
bHalDelayMs(100);
bHalSpiSetSpeed(&_if->_if._spi, B_HAL_SPI_SLOW);
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 10);
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
bHalSpiSend(&_if->_if._spi, cmd, 6);
cnt = 0;
for (;;)
{
tmp = bHalSpiTransfer(&_if->_if._spi, 0xff);
if (tmp == 0x1 || cnt >= 0x1fff)
{
break;
}
cnt++;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
if (cnt >= 0x1fff)
{
return -1;
}
return 0;
}
static int _bSD_SendCmd(bDriverInterface_t *pdrv, uint8_t cmd, uint32_t param, uint8_t crc)
{
uint8_t cmd_table[6];
uint8_t tmp, cnt;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_bSD_WaitReady(pdrv) < 0)
{
return -1;
}
cmd_table[0] = cmd;
cmd_table[1] = (uint8_t)((param >> 24) & 0xff);
cmd_table[2] = (uint8_t)((param >> 16) & 0xff);
cmd_table[3] = (uint8_t)((param >> 8) & 0xff);
cmd_table[4] = (uint8_t)((param >> 0) & 0xff);
cmd_table[5] = crc;
bHalSpiSend(&_if->_if._spi, cmd_table, 6);
cnt = 0;
for (;;)
{
tmp = bHalSpiTransfer(&_if->_if._spi, 0xff);
if (tmp != 0xff || cnt > 200)
{
break;
}
cnt++;
}
if (cnt > 200)
{
b_log_e("cmd%d err..\r\n", cmd);
return -1;
}
return tmp;
}
static int _bSD_Init(bDriverInterface_t *pdrv)
{
int retval = -1;
uint32_t cnt = 0;
uint8_t ocr[4];
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_if->is_spi == 0)
{
return 0;
}
if (_bSD_PowerON(pdrv) < 0)
{
b_log_e("power on err\r\n");
return -1;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
retval = _bSD_SendCmd(pdrv, CMD0, 0, 0x95);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
if (retval == 1)
{
retval = _bSD_SendCmd(pdrv, CMD8, 0x1AA, 0x87);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
}
else
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
if (retval == 0x5)
{
pdrv->_private.v = CT_SD1;
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
cnt = 0;
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
do
{
retval = _bSD_SendCmd(pdrv, CMD55, 0, 0);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
retval = _bSD_SendCmd(pdrv, CMD41, 0, 0);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
cnt++;
} while ((retval != 0) && (cnt < 400));
if (cnt >= 400)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
bHalSpiSetSpeed(&_if->_if._spi, B_HAL_SPI_FAST);
_bSD_SendDump(pdrv, 1);
retval = _bSD_SendCmd(pdrv, CMD59, 0, 0x95);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
retval = _bSD_SendCmd(pdrv, CMD16, 512, 0x95);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
}
else if (retval == 0x1)
{
ocr[0] = bHalSpiTransfer(&_if->_if._spi, 0xff);
ocr[1] = bHalSpiTransfer(&_if->_if._spi, 0xff);
ocr[2] = bHalSpiTransfer(&_if->_if._spi, 0xff);
ocr[3] = bHalSpiTransfer(&_if->_if._spi, 0xff);
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
cnt = 0;
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
do
{
retval = _bSD_SendCmd(pdrv, CMD55, 0, 0);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
retval = _bSD_SendCmd(pdrv, CMD41, 0x40000000, 0);
if (retval < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
cnt++;
} while ((retval != 0) && (cnt < 400));
if (cnt >= 400)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
retval = _bSD_SendCmd(pdrv, CMD58, 0, 0);
if (retval != 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
ocr[0] = bHalSpiTransfer(&_if->_if._spi, 0xff);
ocr[1] = bHalSpiTransfer(&_if->_if._spi, 0xff);
ocr[2] = bHalSpiTransfer(&_if->_if._spi, 0xff);
ocr[3] = bHalSpiTransfer(&_if->_if._spi, 0xff);
if (ocr[0] & 0x40)
{
pdrv->_private.v = CT_SDHC;
}
else
{
pdrv->_private.v = CT_SD2;
}
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
bHalSpiSetSpeed(&_if->_if._spi, B_HAL_SPI_FAST);
b_log("sd type:%d\r\n", pdrv->_private.v);
return 0;
}
static int _bSD_WaitResponse(bDriverInterface_t *pdrv, uint8_t exp)
{
uint16_t cnt = 0;
uint8_t tmp;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
do
{
tmp = bHalSpiTransfer(&_if->_if._spi, 0xff);
cnt++;
} while (tmp != exp && cnt <= 0xfff);
if (tmp == exp)
{
return 0;
}
return -1;
}
static int _bSD_ReceiveData(bDriverInterface_t *pdrv, uint8_t *buff, uint16_t len)
{
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_bSD_WaitResponse(pdrv, 0xfe) < 0)
{
return -1;
}
while (len--)
{
*buff = bHalSpiTransfer(&_if->_if._spi, 0xff);
buff++;
}
bHalSpiTransfer(&_if->_if._spi, 0xff);
bHalSpiTransfer(&_if->_if._spi, 0xff);
return len;
}
static int _bSD_ReadSingleBlock(bDriverInterface_t *pdrv, uint32_t sector, uint8_t *pbuf)
{
int retval = 0;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_if->is_spi)
{
if (pdrv->_private.v != CT_SDHC)
{
sector = sector << 9;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
retval = _bSD_SendCmd(pdrv, CMD17, sector, 0);
if (retval != 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
if (_bSD_ReceiveData(pdrv, pbuf, 512) < 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
}
return 512;
}
static int _bSD_WriteSingleBlock(bDriverInterface_t *pdrv, uint32_t sector, uint8_t *pbuf)
{
int retval = 0;
uint16_t cnt;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_if->is_spi)
{
if (pdrv->_private.v != CT_SDHC)
{
sector = sector << 9;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
retval = _bSD_SendCmd(pdrv, CMD24, sector, 0);
if (retval != 0)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 0);
_bSD_SendDump(pdrv, 3);
bHalSpiTransfer(&_if->_if._spi, 0xfe);
bHalSpiSend(&_if->_if._spi, pbuf, 512);
_bSD_SendDump(pdrv, 2);
retval = bHalSpiTransfer(&_if->_if._spi, 0xff);
if ((retval & 0x1f) != 0x05)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
cnt = 0;
while (!bHalSpiTransfer(&_if->_if._spi, 0xff))
{
cnt++;
if (cnt >= 0xfffe)
{
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
return -1;
}
}
bHalGpioWritePin(_if->_if._spi.cs.port, _if->_if._spi.cs.pin, 1);
_bSD_SendDump(pdrv, 1);
}
return 0;
}
static int _bSD_Write(bDriverInterface_t *pdrv, uint32_t addr, uint8_t *pbuf, uint32_t len)
{
int i = 0;
int retval = -1;
uint32_t sector = addr / 512;
uint32_t count = len / 512;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_if->is_spi)
{
for (i = 0; i < count; i++)
{
retval = _bSD_WriteSingleBlock(pdrv, sector + i, pbuf + i * 512);
if (retval < 0)
{
break;
}
}
}
else
{
retval = bHalSDIOWriteBlocks(_if->_if._sdio, pbuf, sector, count);
}
if (retval < 0)
{
return -1;
}
return (count * 512);
}
static int _bSD_Read(bDriverInterface_t *pdrv, uint32_t addr, uint8_t *pbuf, uint32_t len)
{
int i = 0;
int retval = -1;
uint32_t sector = addr / 512;
uint32_t count = len / 512;
bDRIVER_GET_HALIF(_if, bSD_HalIf_t, pdrv);
if (_if->is_spi)
{
for (i = 0; i < count; i++)
{
retval = _bSD_ReadSingleBlock(pdrv, sector + i, pbuf + i * 512);
if (retval < 0)
{
break;
}
}
}
else
{
retval = bHalSDIOReadBlocks(_if->_if._sdio, pbuf, sector, count);
}
if (retval < 0)
{
return -1;
}
return (count * 512);
}
/**
* \}
*/
/**
* \addtogroup SD_Exported_Functions
* \{
*/
int bSD_Init(bDriverInterface_t *pdrv)
{
bDRIVER_STRUCT_INIT(pdrv, DRIVER_NAME, bSD_Init);
pdrv->read = _bSD_Read;
pdrv->write = _bSD_Write;
if (_bSD_Init(pdrv) < 0)
{
return -1;
}
return 0;
}
#ifdef BSECTION_NEED_PRAGMA
#pragma section driver_init
#endif
bDRIVER_REG_INIT(B_DRIVER_SD, bSD_Init);
#ifdef BSECTION_NEED_PRAGMA
#pragma section
#endif
/**
* \}
*/
/**
* \}
*/
/**
* \}
*/
/************************ Copyright (c) 2020 Bean *****END OF FILE****/

Комментарий ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://gitlife.ru/oschina-mirror/notrynohigh-BabyOS.git
git@gitlife.ru:oschina-mirror/notrynohigh-BabyOS.git
oschina-mirror
notrynohigh-BabyOS
notrynohigh-BabyOS
master