123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731 |
- /*
- * File : msd.c
- * SPI mode SD Card Driver
- * This file is part of RT-Thread RTOS
- * COPYRIGHT (C) 2006 - 2018, RT-Thread Development Team
- *
- * The license and distribution terms for this file may be
- * found in the file LICENSE in this distribution or at
- * http://www.rt-thread.org/license/LICENSE
- *
- * Change Logs:
- * Date Author Notes
- * 2009-04-17 Bernard first version.
- * 2010-07-15 aozima Modify read/write according new block driver interface.
- * 2012-02-01 aozima use new RT-Thread SPI drivers.
- * 2012-04-11 aozima get max. data transfer rate from CSD[TRAN_SPEED].
- * 2012-05-21 aozima update MMC card support.
- * 2018-03-09 aozima fixed CSD Version 2.0 sector count calc.
- */
- #include <string.h>
- #include "spi_msd.h"
- //#define MSD_TRACE
- #ifdef MSD_TRACE
- #define MSD_DEBUG(...) rt_kprintf("[MSD] %d ", rt_tick_get()); rt_kprintf(__VA_ARGS__);
- #else
- #define MSD_DEBUG(...)
- #endif /* #ifdef MSD_TRACE */
- #define DUMMY 0xFF
- #define CARD_NCR_MAX 9
- #define CARD_NRC 1
- #define CARD_NCR 1
- static struct msd_device _msd_device;
- /* function define */
- static rt_bool_t rt_tick_timeout(rt_tick_t tick_start, rt_tick_t tick_long);
- static rt_err_t MSD_take_owner(struct rt_spi_device *spi_device);
- static void MSD_take_cs(struct rt_spi_device *device);
- static void MSD_release_cs(struct rt_spi_device *device);
- static rt_err_t _wait_token(struct rt_spi_device *device, uint8_t token);
- static rt_err_t _wait_ready(struct rt_spi_device *device);
- static rt_err_t rt_msd_init(rt_device_t dev);
- static rt_err_t rt_msd_open(rt_device_t dev, rt_uint16_t oflag);
- static rt_err_t rt_msd_close(rt_device_t dev);
- static rt_size_t rt_msd_write(rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size);
- static rt_size_t rt_msd_read(rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size);
- static rt_size_t rt_msd_sdhc_read(rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size);
- static rt_size_t rt_msd_sdhc_write(rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size);
- static rt_err_t rt_msd_control(rt_device_t dev, int cmd, void *args);
- static rt_err_t MSD_take_owner(struct rt_spi_device *spi_device)
- {
- rt_err_t result;
- result = rt_mutex_take(&(spi_device->bus->lock), RT_WAITING_FOREVER);
- if (result == RT_EOK)
- {
- if (spi_device->bus->owner != spi_device)
- {
- /* not the same owner as current, re-configure SPI bus */
- result = spi_device->bus->ops->configure(spi_device, &spi_device->config);
- if (result == RT_EOK)
- {
- /* set SPI bus owner */
- spi_device->bus->owner = spi_device;
- }
- }
- }
- return result;
- }
- static void MSD_take_cs(struct rt_spi_device *device)
- {
- struct rt_spi_message message;
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = RT_NULL;
- message.length = 0;
- message.cs_take = 1;
- message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- }
- static void MSD_release_cs(struct rt_spi_device *device)
- {
- struct rt_spi_message message;
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = RT_NULL;
- message.length = 0;
- message.cs_take = 0;
- message.cs_release = 1;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- }
- static rt_bool_t rt_tick_timeout(rt_tick_t tick_start, rt_tick_t tick_long)
- {
- rt_tick_t tick_end = tick_start + tick_long;
- rt_tick_t tick_now = rt_tick_get();
- rt_bool_t result = RT_FALSE;
- if (tick_end >= tick_start)
- {
- if (tick_now >= tick_end)
- {
- result = RT_TRUE;
- }
- else
- {
- result = RT_FALSE;
- }
- }
- else
- {
- if ((tick_now < tick_start) && (tick_now >= tick_end))
- {
- result = RT_TRUE;
- }
- else
- {
- result = RT_FALSE;
- }
- }
- return result;
- }
- static uint8_t crc7(const uint8_t *buf, int len)
- {
- unsigned char i, j, crc, ch, ch2, ch3;
- crc = 0;
- for (i = 0; i < len; i ++)
- {
- ch = buf[i];
- for (j = 0; j < 8; j ++, ch <<= 1)
- {
- ch2 = (crc & 0x40) ? 1 : 0;
- ch3 = (ch & 0x80) ? 1 : 0;
- if (ch2 ^ ch3)
- {
- crc ^= 0x04;
- crc <<= 1;
- crc |= 0x01;
- }
- else
- {
- crc <<= 1;
- }
- }
- }
- return crc;
- }
- static rt_err_t _send_cmd(
- struct rt_spi_device *device,
- uint8_t cmd,
- uint32_t arg,
- uint8_t crc,
- response_type type,
- uint8_t *response
- )
- {
- struct rt_spi_message message;
- uint8_t cmd_buffer[8];
- uint8_t recv_buffer[sizeof(cmd_buffer)];
- uint32_t i;
- cmd_buffer[0] = DUMMY;
- cmd_buffer[1] = (cmd | 0x40);
- cmd_buffer[2] = (uint8_t)(arg >> 24);
- cmd_buffer[3] = (uint8_t)(arg >> 16);
- cmd_buffer[4] = (uint8_t)(arg >> 8);
- cmd_buffer[5] = (uint8_t)(arg);
- if (crc == 0x00)
- {
- crc = crc7(&cmd_buffer[1], 5);
- crc = (crc << 1) | 0x01;
- }
- cmd_buffer[6] = (crc);
- cmd_buffer[7] = DUMMY;
- /* initial message */
- message.send_buf = cmd_buffer;
- message.recv_buf = recv_buffer;
- message.length = sizeof(cmd_buffer);
- message.cs_take = message.cs_release = 0;
- _wait_ready(device);
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- for (i = CARD_NCR; i < (CARD_NCR_MAX + 1); i++)
- {
- uint8_t send = DUMMY;
- /* initial message */
- message.send_buf = &send;
- message.recv_buf = response;
- message.length = 1;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- if (0 == (response[0] & 0x80))
- {
- break;
- }
- } /* wait response */
- if ((CARD_NCR_MAX + 1) == i)
- {
- return RT_ERROR;//fail
- }
- //recieve other byte
- if (type == response_r1)
- {
- return RT_EOK;
- }
- else if (type == response_r1b)
- {
- rt_tick_t tick_start = rt_tick_get();
- uint8_t recv;
- while (1)
- {
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = &recv;
- message.length = 1;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- if (recv == DUMMY)
- {
- return RT_EOK;
- }
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(2000)))
- {
- return RT_ETIMEOUT;
- }
- }
- }
- else if (type == response_r2)
- {
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = response + 1;
- message.length = 1;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- }
- else if ((type == response_r3) || (type == response_r7))
- {
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = response + 1;
- message.length = 4;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- }
- else
- {
- return RT_ERROR; // unknow type?
- }
- return RT_EOK;
- }
- static rt_err_t _wait_token(struct rt_spi_device *device, uint8_t token)
- {
- struct rt_spi_message message;
- rt_tick_t tick_start;
- uint8_t send, recv;
- tick_start = rt_tick_get();
- /* wati token */
- /* initial message */
- send = DUMMY;
- message.send_buf = &send;
- message.recv_buf = &recv;
- message.length = 1;
- message.cs_take = message.cs_release = 0;
- while (1)
- {
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- if (recv == token)
- {
- return RT_EOK;
- }
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(CARD_WAIT_TOKEN_TIMES)))
- {
- MSD_DEBUG("[err] wait data start token timeout!\r\n");
- return RT_ETIMEOUT;
- }
- } /* wati token */
- }
- static rt_err_t _wait_ready(struct rt_spi_device *device)
- {
- struct rt_spi_message message;
- rt_tick_t tick_start;
- uint8_t send, recv;
- tick_start = rt_tick_get();
- send = DUMMY;
- /* initial message */
- message.send_buf = &send;
- message.recv_buf = &recv;
- message.length = 1;
- message.cs_take = message.cs_release = 0;
- while (1)
- {
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- if (recv == DUMMY)
- {
- return RT_EOK;
- }
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(1000)))
- {
- MSD_DEBUG("[err] wait ready timeout!\r\n");
- return RT_ETIMEOUT;
- }
- }
- }
- static rt_err_t _read_block(struct rt_spi_device *device, void *buffer, uint32_t block_size)
- {
- struct rt_spi_message message;
- rt_err_t result;
- /* wati token */
- result = _wait_token(device, MSD_TOKEN_READ_START);
- if (result != RT_EOK)
- {
- return result;
- }
- /* read data */
- {
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = buffer;
- message.length = block_size;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- } /* read data */
- /* get crc */
- {
- uint8_t recv_buffer[2];
- /* initial message */
- message.send_buf = RT_NULL;
- message.recv_buf = recv_buffer;
- message.length = 2;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- } /* get crc */
- return RT_EOK;
- }
- static rt_err_t _write_block(struct rt_spi_device *device, const void *buffer, uint32_t block_size, uint8_t token)
- {
- struct rt_spi_message message;
- uint8_t send_buffer[16];
- rt_memset(send_buffer, DUMMY, sizeof(send_buffer));
- send_buffer[sizeof(send_buffer) - 1] = token;
- /* send start block token */
- {
- /* initial message */
- message.send_buf = send_buffer;
- message.recv_buf = RT_NULL;
- message.length = sizeof(send_buffer);
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- }
- /* send data */
- {
- /* initial message */
- message.send_buf = buffer;
- message.recv_buf = RT_NULL;
- message.length = block_size;
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- }
- /* put crc and get data response */
- {
- uint8_t recv_buffer[3];
- uint8_t response;
- /* initial message */
- message.send_buf = send_buffer;
- message.recv_buf = recv_buffer;
- message.length = sizeof(recv_buffer);
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- device->bus->ops->xfer(device, &message);
- // response = 0x0E & recv_buffer[2];
- response = MSD_GET_DATA_RESPONSE(recv_buffer[2]);
- if (response != MSD_DATA_OK)
- {
- MSD_DEBUG("[err] write block fail! data response : 0x%02X\r\n", response);
- return RT_ERROR;
- }
- }
- /* wati ready */
- return _wait_ready(device);
- }
- #ifdef RT_USING_DEVICE_OPS
- const static struct rt_device_ops msd_ops =
- {
- rt_msd_init,
- rt_msd_open,
- rt_msd_close,
- rt_msd_read,
- rt_msd_write,
- rt_msd_control
- };
- const static struct rt_device_ops msd_sdhc_ops =
- {
- rt_msd_init,
- rt_msd_open,
- rt_msd_close,
- rt_msd_sdhc_read,
- rt_msd_sdhc_write,
- rt_msd_control
- };
- #endif
- /* RT-Thread Device Driver Interface */
- static rt_err_t rt_msd_init(rt_device_t dev)
- {
- struct msd_device *msd = (struct msd_device *)dev;
- uint8_t response[MSD_RESPONSE_MAX_LEN];
- rt_err_t result = RT_EOK;
- rt_tick_t tick_start;
- uint32_t OCR;
- if (msd->spi_device == RT_NULL)
- {
- MSD_DEBUG("[err] the SPI SD device has no SPI!\r\n");
- return RT_EIO;
- }
- /* config spi */
- {
- struct rt_spi_configuration cfg;
- cfg.data_width = 8;
- cfg.mode = RT_SPI_MODE_0 | RT_SPI_MSB; /* SPI Compatible Modes 0 */
- cfg.max_hz = 1000 * 400; /* 400kbit/s */
- rt_spi_configure(msd->spi_device, &cfg);
- } /* config spi */
- /* init SD card */
- {
- struct rt_spi_message message;
- result = MSD_take_owner(msd->spi_device);
- if (result != RT_EOK)
- {
- goto _exit;
- }
- MSD_release_cs(msd->spi_device);
- /* The host shall supply power to the card so that the voltage is reached to Vdd_min within 250ms and
- start to supply at least 74 SD clocks to the SD card with keeping CMD line to high.
- In case of SPI mode, CS shall be held to high during 74 clock cycles. */
- {
- uint8_t send_buffer[100]; /* 100byte > 74 clock */
- /* initial message */
- memset(send_buffer, DUMMY, sizeof(send_buffer));
- message.send_buf = send_buffer;
- message.recv_buf = RT_NULL;
- message.length = sizeof(send_buffer);
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- msd->spi_device->bus->ops->xfer(msd->spi_device, &message);
- } /* send 74 clock */
- /* Send CMD0 (GO_IDLE_STATE) to put MSD in SPI mode */
- {
- tick_start = rt_tick_get();
- while (1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, GO_IDLE_STATE, 0x00, 0x95, response_r1, response);
- MSD_release_cs(msd->spi_device);
- if ((result == RT_EOK) && (response[0] == MSD_IN_IDLE_STATE))
- {
- break;
- }
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(CARD_TRY_TIMES)))
- {
- MSD_DEBUG("[err] SD card goto IDLE mode timeout!\r\n");
- result = RT_ETIMEOUT;
- goto _exit;
- }
- }
- MSD_DEBUG("[info] SD card goto IDLE mode OK!\r\n");
- } /* Send CMD0 (GO_IDLE_STATE) to put MSD in SPI mode */
- /* CMD8 */
- {
- tick_start = rt_tick_get();
- do
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, SEND_IF_COND, 0x01AA, 0x87, response_r7, response);
- MSD_release_cs(msd->spi_device);
- if (result == RT_EOK)
- {
- MSD_DEBUG("[info] CMD8 response : 0x%02X 0x%02X 0x%02X 0x%02X 0x%02X\r\n",
- response[0], response[1], response[2], response[3], response[4]);
- if (response[0] & (1 << 2))
- {
- /* illegal command, SD V1.x or MMC card */
- MSD_DEBUG("[info] CMD8 is illegal command.\r\n");
- MSD_DEBUG("[info] maybe Ver1.X SD Memory Card or MMC card!\r\n");
- msd->card_type = MSD_CARD_TYPE_SD_V1_X;
- break;
- }
- else
- {
- /* SD V2.0 or later or SDHC or SDXC memory card! */
- MSD_DEBUG("[info] Ver2.00 or later or SDHC or SDXC memory card!\r\n");
- msd->card_type = MSD_CARD_TYPE_SD_V2_X;
- }
- if ((0xAA == response[4]) && (0x00 == response[3]))
- {
- /* SD2.0 not support current voltage */
- MSD_DEBUG("[err] VCA = 0, SD2.0 not surpport current operation voltage range\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- }
- else
- {
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(200)))
- {
- MSD_DEBUG("[err] CMD8 SEND_IF_COND timeout!\r\n");
- result = RT_ETIMEOUT;
- goto _exit;
- }
- }
- }
- while (0xAA != response[4]);
- } /* CMD8 */
- /* Ver1.X SD Memory Card or MMC card */
- if (msd->card_type == MSD_CARD_TYPE_SD_V1_X)
- {
- rt_bool_t is_sd_v1_x = RT_FALSE;
- rt_tick_t tick_start;
- /* try SD Ver1.x */
- while (1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_OCR, 0x00, 0x00, response_r3, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[info] It maybe SD1.x or MMC But it is Not response to CMD58!\r\n");
- goto _exit;
- }
- if (0 != (response[0] & 0xFE))
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[info] It look CMD58 as illegal command so it is not SD card!\r\n");
- break;
- }
- MSD_release_cs(msd->spi_device);
- OCR = response[1];
- OCR = (OCR << 8) + response[2];
- OCR = (OCR << 8) + response[3];
- OCR = (OCR << 8) + response[4];
- MSD_DEBUG("[info] OCR is 0x%08X\r\n", OCR);
- if (0 == (OCR & (0x1 << 15)))
- {
- MSD_DEBUG(("[err] SD 1.x But not surpport current voltage\r\n"));
- result = RT_ERROR;
- goto _exit;
- }
- /* --Send ACMD41 to make card ready */
- tick_start = rt_tick_get();
- /* try CMD55 + ACMD41 */
- while (1)
- {
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(CARD_TRY_TIMES_ACMD41)))
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[info] try CMD55 + ACMD41 timeout! mabey MMC card!\r\n");
- break;
- }
- MSD_take_cs(msd->spi_device);
- /* CMD55 APP_CMD */
- result = _send_cmd(msd->spi_device, APP_CMD, 0x00, 0x00, response_r1, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- continue;
- }
- if (0 != (response[0] & 0xFE))
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[info] Not SD card2 , may be MMC\r\n");
- break;
- }
- /* ACMD41 SD_SEND_OP_COND */
- result = _send_cmd(msd->spi_device, SD_SEND_OP_COND, 0x00, 0x00, response_r1, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- continue;
- }
- if (0 != (response[0] & 0xFE))
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[info] Not SD card4 , may be MMC\r\n");
- break;
- }
- if (0 == (response[0] & 0xFF))
- {
- MSD_release_cs(msd->spi_device);
- is_sd_v1_x = RT_TRUE;
- MSD_DEBUG("[info] It is Ver1.X SD Memory Card!!!\r\n");
- break;
- }
- } /* try CMD55 + ACMD41 */
- break;
- } /* try SD Ver1.x */
- /* try MMC */
- if (is_sd_v1_x != RT_TRUE)
- {
- uint32_t i;
- MSD_DEBUG("[info] try MMC card!\r\n");
- MSD_release_cs(msd->spi_device);
- /* send dummy clock */
- {
- uint8_t send_buffer[100];
- /* initial message */
- memset(send_buffer, DUMMY, sizeof(send_buffer));
- message.send_buf = send_buffer;
- message.recv_buf = RT_NULL;
- message.length = sizeof(send_buffer);
- message.cs_take = message.cs_release = 0;
- for (i = 0; i < 10; i++)
- {
- /* transfer message */
- msd->spi_device->bus->ops->xfer(msd->spi_device, &message);
- }
- } /* send dummy clock */
- /* send CMD0 goto IDLE state */
- tick_start = rt_tick_get();
- while (1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, GO_IDLE_STATE, 0x00, 0x95, response_r1, response);
- MSD_release_cs(msd->spi_device);
- if ((result == RT_EOK) && (response[0] == MSD_IN_IDLE_STATE))
- {
- break;
- }
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(CARD_TRY_TIMES)))
- {
- MSD_DEBUG("[err] SD card goto IDLE mode timeout!\r\n");
- result = RT_ETIMEOUT;
- goto _exit;
- }
- } /* send CMD0 goto IDLE stat */
- /* send CMD1 */
- tick_start = rt_tick_get();
- while (1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, SEND_OP_COND, 0x00, 0x00, response_r1, response);
- MSD_release_cs(msd->spi_device);
- if ((result == RT_EOK) && (response[0] == MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[info] It is MMC card!!!\r\n");
- msd->card_type = MSD_CARD_TYPE_MMC;
- break;
- }
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(CARD_TRY_TIMES)))
- {
- MSD_DEBUG("[err] SD card goto IDLE mode timeout!\r\n");
- result = RT_ETIMEOUT;
- goto _exit;
- }
- } /* send CMD1 */
- } /* try MMC */
- }
- else if (msd->card_type == MSD_CARD_TYPE_SD_V2_X)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_OCR, 0x00, 0x00, response_r3, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] It maybe SD2.0 But it is Not response to CMD58!\r\n");
- goto _exit;
- }
- if ((response[0] & 0xFE) != 0)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] It look CMD58 as illegal command so it is not SD card!\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- MSD_release_cs(msd->spi_device);
- OCR = response[1];
- OCR = (OCR << 8) + response[2];
- OCR = (OCR << 8) + response[3];
- OCR = (OCR << 8) + response[4];
- MSD_DEBUG("[info] OCR is 0x%08X\r\n", OCR);
- if (0 == (OCR & (0x1 << 15)))
- {
- MSD_DEBUG(("[err] SD 1.x But not surpport current voltage\r\n"));
- result = RT_ERROR;
- goto _exit;
- }
- /* --Send ACMD41 to make card ready */
- tick_start = rt_tick_get();
- /* try CMD55 + ACMD41 */
- do
- {
- MSD_take_cs(msd->spi_device);
- if (rt_tick_timeout(tick_start, rt_tick_from_millisecond(CARD_TRY_TIMES_ACMD41)))
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] SD Ver2.x or later try CMD55 + ACMD41 timeout!\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- /* CMD55 APP_CMD */
- result = _send_cmd(msd->spi_device, APP_CMD, 0x00, 0x65, response_r1, response);
- // if((result != RT_EOK) || (response[0] == 0x01))
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- continue;
- }
- if ((response[0] & 0xFE) != 0)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] Not SD ready!\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- /* ACMD41 SD_SEND_OP_COND */
- result = _send_cmd(msd->spi_device, SD_SEND_OP_COND, 0x40000000, 0x77, response_r1, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] ACMD41 fail!\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- if ((response[0] & 0xFE) != 0)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[info] Not SD card4 , response : 0x%02X\r\n", response[0]);
- // break;
- }
- }
- while (response[0] != MSD_RESPONSE_NO_ERROR);
- MSD_release_cs(msd->spi_device);
- /* try CMD55 + ACMD41 */
- /* --Read OCR again */
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_OCR, 0x00, 0x00, response_r3, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] It maybe SD2.0 But it is Not response to 2nd CMD58!\r\n");
- goto _exit;
- }
- if ((response[0] & 0xFE) != 0)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] It look 2nd CMD58 as illegal command so it is not SD card!\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- MSD_release_cs(msd->spi_device);
- OCR = response[1];
- OCR = (OCR << 8) + response[2];
- OCR = (OCR << 8) + response[3];
- OCR = (OCR << 8) + response[4];
- MSD_DEBUG("[info] OCR 2nd read is 0x%08X\r\n", OCR);
- if ((OCR & 0x40000000) != 0)
- {
- MSD_DEBUG("[info] It is SD2.0 SDHC Card!!!\r\n");
- msd->card_type = MSD_CARD_TYPE_SD_SDHC;
- }
- else
- {
- MSD_DEBUG("[info] It is SD2.0 standard capacity Card!!!\r\n");
- }
- } /* MSD_CARD_TYPE_SD_V2_X */
- else
- {
- MSD_DEBUG("[err] SD card type unkonw!\r\n");
- result = RT_ERROR;
- goto _exit;
- }
- } /* init SD card */
- if (msd->card_type == MSD_CARD_TYPE_SD_SDHC)
- {
- #ifdef RT_USING_DEVICE_OPS
- dev->ops = &msd_sdhc_ops;
- #else
- dev->read = rt_msd_sdhc_read;
- dev->write = rt_msd_sdhc_write;
- #endif
- }
- else
- {
- #ifdef RT_USING_DEVICE_OPS
- dev->ops = &msd_ops;
- #else
- dev->read = rt_msd_read;
- dev->write = rt_msd_write;
- #endif
- }
- /* set CRC */
- {
- MSD_release_cs(msd->spi_device);
- MSD_take_cs(msd->spi_device);
- #ifdef MSD_USE_CRC
- result = _send_cmd(msd->spi_device, CRC_ON_OFF, 0x01, 0x83, response_r1, response);
- #else
- result = _send_cmd(msd->spi_device, CRC_ON_OFF, 0x00, 0x91, response_r1, response);
- #endif
- MSD_release_cs(msd->spi_device);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD59 CRC_ON_OFF fail! response : 0x%02X\r\n", response[0]);
- result = RT_ERROR;
- goto _exit;
- }
- } /* set CRC */
- /* CMD16 SET_BLOCKLEN */
- {
- MSD_release_cs(msd->spi_device);
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, SET_BLOCKLEN, SECTOR_SIZE, 0x00, response_r1, response);
- MSD_release_cs(msd->spi_device);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD16 SET_BLOCKLEN fail! response : 0x%02X\r\n", response[0]);
- result = RT_ERROR;
- goto _exit;
- }
- msd->geometry.block_size = SECTOR_SIZE;
- msd->geometry.bytes_per_sector = SECTOR_SIZE;
- }
- /* read CSD */
- {
- uint8_t CSD_buffer[MSD_CSD_LEN];
- MSD_take_cs(msd->spi_device);
- // result = _send_cmd(msd->spi_device, SEND_CSD, 0x00, 0xAF, response_r1, response);
- result = _send_cmd(msd->spi_device, SEND_CSD, 0x00, 0x00, response_r1, response);
- if (result != RT_EOK)
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] CMD9 SEND_CSD timeout!\r\n");
- goto _exit;
- }
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_release_cs(msd->spi_device);
- MSD_DEBUG("[err] CMD9 SEND_CSD fail! response : 0x%02X\r\n", response[0]);
- result = RT_ERROR;
- goto _exit;
- }
- result = _read_block(msd->spi_device, CSD_buffer, MSD_CSD_LEN);
- MSD_release_cs(msd->spi_device);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read CSD fail!\r\n");
- goto _exit;
- }
- /* Analyze CSD */
- {
- uint8_t CSD_STRUCTURE;
- uint32_t C_SIZE;
- uint32_t card_capacity;
- uint8_t tmp8;
- uint16_t tmp16;
- uint32_t tmp32;
- /* get CSD_STRUCTURE */
- tmp8 = CSD_buffer[0] & 0xC0; /* 0b11000000 */
- CSD_STRUCTURE = tmp8 >> 6;
- /* MMC CSD Analyze. */
- if (msd->card_type == MSD_CARD_TYPE_MMC)
- {
- uint8_t C_SIZE_MULT;
- uint8_t READ_BL_LEN;
- if (CSD_STRUCTURE > 2)
- {
- MSD_DEBUG("[err] bad CSD Version : %d\r\n", CSD_STRUCTURE);
- result = RT_ERROR;
- goto _exit;
- }
- if (CSD_STRUCTURE == 0)
- {
- MSD_DEBUG("[info] CSD version No. 1.0\r\n");
- }
- else if (CSD_STRUCTURE == 1)
- {
- MSD_DEBUG("[info] CSD version No. 1.1\r\n");
- }
- else if (CSD_STRUCTURE == 2)
- {
- MSD_DEBUG("[info] CSD version No. 1.2\r\n");
- }
- /* get TRAN_SPEED 8bit [103:96] */
- tmp8 = CSD_buffer[3];
- tmp8 &= 0x03; /* [2:0] transfer rate unit.*/
- if (tmp8 == 0)
- {
- msd->max_clock = 100 * 1000; /* 0=100kbit/s. */
- }
- else if (tmp8 == 1)
- {
- msd->max_clock = 1 * 1000 * 1000; /* 1=1Mbit/s. */
- }
- else if (tmp8 == 2)
- {
- msd->max_clock = 10 * 1000 * 1000; /* 2=10Mbit/s. */
- }
- else if (tmp8 == 3)
- {
- msd->max_clock = 100 * 1000 * 1000; /* 3=100Mbit/s. */
- }
- if (tmp8 == 0)
- {
- MSD_DEBUG("[info] TRAN_SPEED: 0x%02X, %dkbit/s.\r\n", tmp8, msd->max_clock / 1000);
- }
- else
- {
- MSD_DEBUG("[info] TRAN_SPEED: 0x%02X, %dMbit/s.\r\n", tmp8, msd->max_clock / 1000 / 1000);
- }
- /* get READ_BL_LEN 4bit [83:80] */
- tmp8 = CSD_buffer[5] & 0x0F; /* 0b00001111; */
- READ_BL_LEN = tmp8; /* 4 bit */
- MSD_DEBUG("[info] CSD : READ_BL_LEN : %d %dbyte\r\n", READ_BL_LEN, (1 << READ_BL_LEN));
- /* get C_SIZE 12bit [73:62] */
- tmp16 = CSD_buffer[6] & 0x03; /* get [73:72] 0b00000011 */
- tmp16 = tmp16 << 8;
- tmp16 += CSD_buffer[7]; /* get [71:64] */
- tmp16 = tmp16 << 2;
- tmp8 = CSD_buffer[8] & 0xC0; /* get [63:62] 0b11000000 */
- tmp8 = tmp8 >> 6;
- tmp16 = tmp16 + tmp8;
- C_SIZE = tmp16; //12 bit
- MSD_DEBUG("[info] CSD : C_SIZE : %d\r\n", C_SIZE);
- /* get C_SIZE_MULT 3bit [49:47] */
- tmp8 = CSD_buffer[9] & 0x03;//0b00000011;
- tmp8 = tmp8 << 1;
- tmp8 = tmp8 + ((CSD_buffer[10] & 0x80/*0b10000000*/) >> 7);
- C_SIZE_MULT = tmp8; // 3 bit
- MSD_DEBUG("[info] CSD : C_SIZE_MULT : %d\r\n", C_SIZE_MULT);
- /* memory capacity = BLOCKNR * BLOCK_LEN */
- /* BLOCKNR = (C_SIZE+1) * MULT */
- /* MULT = 2^(C_SIZE_MULT+2) */
- /* BLOCK_LEN = 2^READ_BL_LEN */
- card_capacity = (1 << READ_BL_LEN) * ((C_SIZE + 1) * (1 << (C_SIZE_MULT + 2)));
- msd->geometry.sector_count = card_capacity / msd->geometry.bytes_per_sector;
- MSD_DEBUG("[info] card capacity : %d Mbyte\r\n", card_capacity / (1024 * 1024));
- }
- else /* SD CSD Analyze. */
- {
- if (CSD_STRUCTURE == 0)
- {
- uint8_t C_SIZE_MULT;
- uint8_t READ_BL_LEN;
- MSD_DEBUG("[info] CSD Version 1.0\r\n");
- /* get TRAN_SPEED 8bit [103:96] */
- tmp8 = CSD_buffer[3];
- if (tmp8 == 0x32)
- {
- msd->max_clock = 1000 * 1000 * 10; /* 10Mbit/s. */
- }
- else if (tmp8 == 0x5A)
- {
- msd->max_clock = 1000 * 1000 * 50; /* 50Mbit/s. */
- }
- else
- {
- msd->max_clock = 1000 * 1000 * 1; /* 1Mbit/s default. */
- }
- MSD_DEBUG("[info] TRAN_SPEED: 0x%02X, %dMbit/s.\r\n", tmp8, msd->max_clock / 1000 / 1000);
- /* get READ_BL_LEN 4bit [83:80] */
- tmp8 = CSD_buffer[5] & 0x0F; /* 0b00001111; */
- READ_BL_LEN = tmp8; /* 4 bit */
- MSD_DEBUG("[info] CSD : READ_BL_LEN : %d %dbyte\r\n", READ_BL_LEN, (1 << READ_BL_LEN));
- /* get C_SIZE 12bit [73:62] */
- tmp16 = CSD_buffer[6] & 0x03; /* get [73:72] 0b00000011 */
- tmp16 = tmp16 << 8;
- tmp16 += CSD_buffer[7]; /* get [71:64] */
- tmp16 = tmp16 << 2;
- tmp8 = CSD_buffer[8] & 0xC0; /* get [63:62] 0b11000000 */
- tmp8 = tmp8 >> 6;
- tmp16 = tmp16 + tmp8;
- C_SIZE = tmp16; //12 bit
- MSD_DEBUG("[info] CSD : C_SIZE : %d\r\n", C_SIZE);
- /* get C_SIZE_MULT 3bit [49:47] */
- tmp8 = CSD_buffer[9] & 0x03;//0b00000011;
- tmp8 = tmp8 << 1;
- tmp8 = tmp8 + ((CSD_buffer[10] & 0x80/*0b10000000*/) >> 7);
- C_SIZE_MULT = tmp8; // 3 bit
- MSD_DEBUG("[info] CSD : C_SIZE_MULT : %d\r\n", C_SIZE_MULT);
- /* memory capacity = BLOCKNR * BLOCK_LEN */
- /* BLOCKNR = (C_SIZE+1) * MULT */
- /* MULT = 2^(C_SIZE_MULT+2) */
- /* BLOCK_LEN = 2^READ_BL_LEN */
- card_capacity = (1 << READ_BL_LEN) * ((C_SIZE + 1) * (1 << (C_SIZE_MULT + 2)));
- msd->geometry.sector_count = card_capacity / msd->geometry.bytes_per_sector;
- MSD_DEBUG("[info] card capacity : %d Mbyte\r\n", card_capacity / (1024 * 1024));
- }
- else if (CSD_STRUCTURE == 1)
- {
- MSD_DEBUG("[info] CSD Version 2.0\r\n");
- /* get TRAN_SPEED 8bit [103:96] */
- tmp8 = CSD_buffer[3];
- if (tmp8 == 0x32)
- {
- msd->max_clock = 1000 * 1000 * 10; /* 10Mbit/s. */
- }
- else if (tmp8 == 0x5A)
- {
- msd->max_clock = 1000 * 1000 * 50; /* 50Mbit/s. */
- }
- else if (tmp8 == 0x0B)
- {
- msd->max_clock = 1000 * 1000 * 100; /* 100Mbit/s. */
- /* UHS50 Card sets TRAN_SPEED to 0Bh (100Mbit/sec), */
- /* for both SDR50 and DDR50 modes. */
- }
- else if (tmp8 == 0x2B)
- {
- msd->max_clock = 1000 * 1000 * 200; /* 200Mbit/s. */
- /* UHS104 Card sets TRAN_SPEED to 2Bh (200Mbit/sec). */
- }
- else
- {
- msd->max_clock = 1000 * 1000 * 1; /* 1Mbit/s default. */
- }
- MSD_DEBUG("[info] TRAN_SPEED: 0x%02X, %dMbit/s.\r\n", tmp8, msd->max_clock / 1000 / 1000);
- /* get C_SIZE 22bit [69:48] */
- tmp32 = CSD_buffer[7] & 0x3F; /* 0b00111111 */
- tmp32 = tmp32 << 8;
- tmp32 += CSD_buffer[8];
- tmp32 = tmp32 << 8;
- tmp32 += CSD_buffer[9];
- C_SIZE = tmp32;
- MSD_DEBUG("[info] CSD : C_SIZE : %d\r\n", C_SIZE);
- /* memory capacity = (C_SIZE+1) * 512K byte */
- card_capacity = (C_SIZE + 1) / 2; /* unit : Mbyte */
- msd->geometry.sector_count = (C_SIZE + 1) * 1024; /* 512KB = 1024sector */
- MSD_DEBUG("[info] card capacity : %d.%d Gbyte\r\n", card_capacity / 1024, (card_capacity % 1024) * 100 / 1024);
- MSD_DEBUG("[info] sector_count : %d\r\n", msd->geometry.sector_count);
- }
- else
- {
- MSD_DEBUG("[err] bad CSD Version : %d\r\n", CSD_STRUCTURE);
- result = RT_ERROR;
- goto _exit;
- }
- } /* SD CSD Analyze. */
- } /* Analyze CSD */
- } /* read CSD */
- /* config spi to high speed */
- {
- struct rt_spi_configuration cfg;
- cfg.data_width = 8;
- cfg.mode = RT_SPI_MODE_0 | RT_SPI_MSB; /* SPI Compatible Modes 0 */
- cfg.max_hz = msd->max_clock;
- rt_spi_configure(msd->spi_device, &cfg);
- } /* config spi */
- _exit:
- MSD_release_cs(msd->spi_device);
- rt_mutex_release(&(msd->spi_device->bus->lock));
- return result;
- }
- static rt_err_t rt_msd_open(rt_device_t dev, rt_uint16_t oflag)
- {
- // struct msd_device * msd = (struct msd_device *)dev;
- return RT_EOK;
- }
- static rt_err_t rt_msd_close(rt_device_t dev)
- {
- // struct msd_device * msd = (struct msd_device *)dev;
- return RT_EOK;
- }
- static rt_size_t rt_msd_read(rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size)
- {
- struct msd_device *msd = (struct msd_device *)dev;
- uint8_t response[MSD_RESPONSE_MAX_LEN];
- rt_err_t result = RT_EOK;
- result = MSD_take_owner(msd->spi_device);
- if (result != RT_EOK)
- {
- goto _exit;
- }
- /* config spi to high speed */
- {
- struct rt_spi_configuration cfg;
- cfg.data_width = 8;
- cfg.mode = RT_SPI_MODE_0 | RT_SPI_MSB; /* SPI Compatible Modes 0 */
- cfg.max_hz = msd->max_clock;
- rt_spi_configure(msd->spi_device, &cfg);
- } /* config spi */
- /* SINGLE_BLOCK? */
- if (size == 1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_SINGLE_BLOCK, pos * msd->geometry.bytes_per_sector, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] read SINGLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- goto _exit;
- }
- result = _read_block(msd->spi_device, buffer, msd->geometry.bytes_per_sector);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read SINGLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- }
- }
- else if (size > 1)
- {
- uint32_t i;
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_MULTIPLE_BLOCK, pos * msd->geometry.bytes_per_sector, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] read READ_MULTIPLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- goto _exit;
- }
- for (i = 0; i < size; i++)
- {
- result = _read_block(msd->spi_device,
- (uint8_t *)buffer + msd->geometry.bytes_per_sector * i,
- msd->geometry.bytes_per_sector);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read READ_MULTIPLE_BLOCK #%d fail!\r\n", pos);
- size = i;
- break;
- }
- }
- /* send CMD12 stop transfer */
- result = _send_cmd(msd->spi_device, STOP_TRANSMISSION, 0x00, 0x00, response_r1b, response);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read READ_MULTIPLE_BLOCK, send stop token fail!\r\n");
- }
- } /* READ_MULTIPLE_BLOCK */
- _exit:
- /* release and exit */
- MSD_release_cs(msd->spi_device);
- rt_mutex_release(&(msd->spi_device->bus->lock));
- return size;
- }
- static rt_size_t rt_msd_sdhc_read(rt_device_t dev, rt_off_t pos, void *buffer, rt_size_t size)
- {
- struct msd_device *msd = (struct msd_device *)dev;
- uint8_t response[MSD_RESPONSE_MAX_LEN];
- rt_err_t result = RT_EOK;
- result = MSD_take_owner(msd->spi_device);
- if (result != RT_EOK)
- {
- goto _exit;
- }
- /* config spi to high speed */
- {
- struct rt_spi_configuration cfg;
- cfg.data_width = 8;
- cfg.mode = RT_SPI_MODE_0 | RT_SPI_MSB; /* SPI Compatible Modes 0 */
- cfg.max_hz = msd->max_clock;
- rt_spi_configure(msd->spi_device, &cfg);
- } /* config spi */
- /* SINGLE_BLOCK? */
- if (size == 1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_SINGLE_BLOCK, pos, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] read SINGLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- goto _exit;
- }
- result = _read_block(msd->spi_device, buffer, msd->geometry.bytes_per_sector);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read SINGLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- }
- }
- else if (size > 1)
- {
- uint32_t i;
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, READ_MULTIPLE_BLOCK, pos, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] read READ_MULTIPLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- goto _exit;
- }
- for (i = 0; i < size; i++)
- {
- result = _read_block(msd->spi_device,
- (uint8_t *)buffer + msd->geometry.bytes_per_sector * i,
- msd->geometry.bytes_per_sector);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read READ_MULTIPLE_BLOCK #%d fail!\r\n", pos);
- size = i;
- break;
- }
- }
- /* send CMD12 stop transfer */
- result = _send_cmd(msd->spi_device, STOP_TRANSMISSION, 0x00, 0x00, response_r1b, response);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] read READ_MULTIPLE_BLOCK, send stop token fail!\r\n");
- }
- } /* READ_MULTIPLE_BLOCK */
- _exit:
- /* release and exit */
- MSD_release_cs(msd->spi_device);
- rt_mutex_release(&(msd->spi_device->bus->lock));
- return size;
- }
- static rt_size_t rt_msd_write(rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size)
- {
- struct msd_device *msd = (struct msd_device *)dev;
- uint8_t response[MSD_RESPONSE_MAX_LEN];
- rt_err_t result;
- result = MSD_take_owner(msd->spi_device);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] get SPI owner fail!\r\n");
- goto _exit;
- }
- /* config spi to high speed */
- {
- struct rt_spi_configuration cfg;
- cfg.data_width = 8;
- cfg.mode = RT_SPI_MODE_0 | RT_SPI_MSB; /* SPI Compatible Modes 0 */
- cfg.max_hz = msd->max_clock;
- rt_spi_configure(msd->spi_device, &cfg);
- } /* config spi */
- /* SINGLE_BLOCK? */
- if (size == 1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, WRITE_BLOCK, pos * msd->geometry.bytes_per_sector, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD WRITE_BLOCK fail!\r\n");
- size = 0;
- goto _exit;
- }
- result = _write_block(msd->spi_device, buffer, msd->geometry.bytes_per_sector, MSD_TOKEN_WRITE_SINGLE_START);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] write SINGLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- }
- }
- else if (size > 1)
- {
- struct rt_spi_message message;
- uint32_t i;
- MSD_take_cs(msd->spi_device);
- #ifdef MSD_USE_PRE_ERASED
- if (msd->card_type != MSD_CARD_TYPE_MMC)
- {
- /* CMD55 APP_CMD */
- result = _send_cmd(msd->spi_device, APP_CMD, 0x00, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD55 APP_CMD fail!\r\n");
- size = 0;
- goto _exit;
- }
- /* ACMD23 Pre-erased */
- result = _send_cmd(msd->spi_device, SET_WR_BLK_ERASE_COUNT, size, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] ACMD23 SET_BLOCK_COUNT fail!\r\n");
- size = 0;
- goto _exit;
- }
- }
- #endif
- result = _send_cmd(msd->spi_device, WRITE_MULTIPLE_BLOCK, pos * msd->geometry.bytes_per_sector, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD WRITE_MULTIPLE_BLOCK fail!\r\n");
- size = 0;
- goto _exit;
- }
- /* write all block */
- for (i = 0; i < size; i++)
- {
- result = _write_block(msd->spi_device,
- (const uint8_t *)buffer + msd->geometry.bytes_per_sector * i,
- msd->geometry.bytes_per_sector,
- MSD_TOKEN_WRITE_MULTIPLE_START);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] write SINGLE_BLOCK #%d fail!\r\n", pos);
- size = i;
- break;
- }
- } /* write all block */
- /* send stop token */
- {
- uint8_t send_buffer[18];
- rt_memset(send_buffer, DUMMY, sizeof(send_buffer));
- send_buffer[sizeof(send_buffer) - 1] = MSD_TOKEN_WRITE_MULTIPLE_STOP;
- /* initial message */
- message.send_buf = send_buffer;
- message.recv_buf = RT_NULL;
- message.length = sizeof(send_buffer);
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- msd->spi_device->bus->ops->xfer(msd->spi_device, &message);
- }
- /* wait ready */
- result = _wait_ready(msd->spi_device);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[warning] wait WRITE_MULTIPLE_BLOCK stop token ready timeout!\r\n");
- }
- } /* size > 1 */
- _exit:
- /* release and exit */
- MSD_release_cs(msd->spi_device);
- rt_mutex_release(&(msd->spi_device->bus->lock));
- return size;
- }
- static rt_size_t rt_msd_sdhc_write(rt_device_t dev, rt_off_t pos, const void *buffer, rt_size_t size)
- {
- struct msd_device *msd = (struct msd_device *)dev;
- uint8_t response[MSD_RESPONSE_MAX_LEN];
- rt_err_t result;
- result = MSD_take_owner(msd->spi_device);
- if (result != RT_EOK)
- {
- goto _exit;
- }
- /* config spi to high speed */
- {
- struct rt_spi_configuration cfg;
- cfg.data_width = 8;
- cfg.mode = RT_SPI_MODE_0 | RT_SPI_MSB; /* SPI Compatible Modes 0 */
- cfg.max_hz = msd->max_clock;
- rt_spi_configure(msd->spi_device, &cfg);
- } /* config spi */
- /* SINGLE_BLOCK? */
- if (size == 1)
- {
- MSD_take_cs(msd->spi_device);
- result = _send_cmd(msd->spi_device, WRITE_BLOCK, pos, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD WRITE_BLOCK fail!\r\n");
- size = 0;
- goto _exit;
- }
- result = _write_block(msd->spi_device, buffer, msd->geometry.bytes_per_sector, MSD_TOKEN_WRITE_SINGLE_START);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] write SINGLE_BLOCK #%d fail!\r\n", pos);
- size = 0;
- }
- }
- else if (size > 1)
- {
- struct rt_spi_message message;
- uint32_t i;
- MSD_take_cs(msd->spi_device);
- #ifdef MSD_USE_PRE_ERASED
- /* CMD55 APP_CMD */
- result = _send_cmd(msd->spi_device, APP_CMD, 0x00, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD55 APP_CMD fail!\r\n");
- size = 0;
- goto _exit;
- }
- /* ACMD23 Pre-erased */
- result = _send_cmd(msd->spi_device, SET_WR_BLK_ERASE_COUNT, size, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] ACMD23 SET_BLOCK_COUNT fail!\r\n");
- size = 0;
- goto _exit;
- }
- #endif
- result = _send_cmd(msd->spi_device, WRITE_MULTIPLE_BLOCK, pos, 0x00, response_r1, response);
- if ((result != RT_EOK) || (response[0] != MSD_RESPONSE_NO_ERROR))
- {
- MSD_DEBUG("[err] CMD WRITE_MULTIPLE_BLOCK fail!\r\n");
- size = 0;
- goto _exit;
- }
- /* write all block */
- for (i = 0; i < size; i++)
- {
- result = _write_block(msd->spi_device,
- (const uint8_t *)buffer + msd->geometry.bytes_per_sector * i,
- msd->geometry.bytes_per_sector,
- MSD_TOKEN_WRITE_MULTIPLE_START);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[err] write MULTIPLE_BLOCK #%d fail!\r\n", pos);
- size = i;
- break;
- }
- } /* write all block */
- /* send stop token */
- {
- uint8_t send_buffer[18];
- rt_memset(send_buffer, DUMMY, sizeof(send_buffer));
- send_buffer[sizeof(send_buffer) - 1] = MSD_TOKEN_WRITE_MULTIPLE_STOP;
- /* initial message */
- message.send_buf = send_buffer;
- message.recv_buf = RT_NULL;
- message.length = sizeof(send_buffer);
- message.cs_take = message.cs_release = 0;
- /* transfer message */
- msd->spi_device->bus->ops->xfer(msd->spi_device, &message);
- }
- result = _wait_ready(msd->spi_device);
- if (result != RT_EOK)
- {
- MSD_DEBUG("[warning] wait WRITE_MULTIPLE_BLOCK stop token ready timeout!\r\n");
- }
- } /* size > 1 */
- _exit:
- /* release and exit */
- MSD_release_cs(msd->spi_device);
- rt_mutex_release(&(msd->spi_device->bus->lock));
- return size;
- }
- static rt_err_t rt_msd_control(rt_device_t dev, int cmd, void *args)
- {
- struct msd_device *msd = (struct msd_device *)dev;
- RT_ASSERT(dev != RT_NULL);
- if (cmd == RT_DEVICE_CTRL_BLK_GETGEOME)
- {
- struct rt_device_blk_geometry *geometry;
- geometry = (struct rt_device_blk_geometry *)args;
- if (geometry == RT_NULL) return -RT_ERROR;
- geometry->bytes_per_sector = msd->geometry.bytes_per_sector;
- geometry->block_size = msd->geometry.block_size;
- geometry->sector_count = msd->geometry.sector_count;
- }
- return RT_EOK;
- }
- rt_err_t msd_init(const char *sd_device_name, const char *spi_device_name)
- {
- rt_err_t result = RT_EOK;
- struct rt_spi_device *spi_device;
- spi_device = (struct rt_spi_device *)rt_device_find(spi_device_name);
- if (spi_device == RT_NULL)
- {
- MSD_DEBUG("spi device %s not found!\r\n", spi_device_name);
- return -RT_ENOSYS;
- }
- rt_memset(&_msd_device, 0, sizeof(_msd_device));
- _msd_device.spi_device = spi_device;
- /* register sdcard device */
- _msd_device.parent.type = RT_Device_Class_Block;
- _msd_device.geometry.bytes_per_sector = 0;
- _msd_device.geometry.sector_count = 0;
- _msd_device.geometry.block_size = 0;
- #ifdef RT_USING_DEVICE_OPS
- _msd_device.parent.ops = &msd_ops;
- #else
- _msd_device.parent.init = rt_msd_init;
- _msd_device.parent.open = rt_msd_open;
- _msd_device.parent.close = rt_msd_close;
- _msd_device.parent.read = RT_NULL;
- _msd_device.parent.write = RT_NULL;
- _msd_device.parent.control = rt_msd_control;
- #endif
- /* no private, no callback */
- _msd_device.parent.user_data = RT_NULL;
- _msd_device.parent.rx_indicate = RT_NULL;
- _msd_device.parent.tx_complete = RT_NULL;
- result = rt_device_register(&_msd_device.parent, sd_device_name,
- RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_REMOVABLE | RT_DEVICE_FLAG_STANDALONE);
- return result;
- }
|