|
-
- #include <Arduino.h>
- #include <SPI.h>
- #include "Sd2Card.h"
-
- #ifdef SPI_HAS_TRANSACTION
- static SPISettings settings;
- #endif
-
-
-
- #if defined(__MK20DX128__) || defined(__MK20DX256__) || defined(__MK64FX512__) || defined(__MK66FX1M0__)
- #define USE_TEENSY3_SPI
-
-
- #include <kinetis.h>
-
- #define SPI_INITIAL_FIFO_DEPTH 3
-
- #ifndef SPI_SR_RXCTR
- #define SPI_SR_RXCTR 0XF0
- #endif
- #ifndef SPI_PUSHR_CONT
- #define SPI_PUSHR_CONT 0X80000000
- #endif
- #ifndef SPI_PUSHR_CTAS
- #define SPI_PUSHR_CTAS(n) (((n) & 7) << 28)
- #endif
-
- static void spiBegin() {
- SIM_SCGC6 |= SIM_SCGC6_SPI0;
- }
-
- static void spiInit(uint8_t spiRate) {
- switch (spiRate) {
-
- case 0: settings = SPISettings(24000000, MSBFIRST, SPI_MODE0); break;
- case 1: settings = SPISettings(24000000, MSBFIRST, SPI_MODE0); break;
- case 2: settings = SPISettings(8000000, MSBFIRST, SPI_MODE0); break;
- case 3: settings = SPISettings(4000000, MSBFIRST, SPI_MODE0); break;
- case 4: settings = SPISettings(3000000, MSBFIRST, SPI_MODE0); break;
- case 5: settings = SPISettings(2000000, MSBFIRST, SPI_MODE0); break;
- default: settings = SPISettings(400000, MSBFIRST, SPI_MODE0);
- }
- SPI.begin();
- }
-
-
- static uint8_t spiRec() {
- SPI0_MCR |= SPI_MCR_CLR_RXF;
- SPI0_SR = SPI_SR_TCF;
- SPI0_PUSHR = 0xFF;
- while (!(SPI0_SR & SPI_SR_TCF)) {}
- return SPI0_POPR;
- }
-
- static uint8_t spiRec(uint8_t* buf, size_t len) {
-
- SPI0_MCR = SPI_MCR_MSTR | SPI_MCR_CLR_RXF | SPI_MCR_PCSIS(0x1F);
-
-
- if (len & 1) {
- *buf++ = spiRec();
- len--;
- }
-
- int nf = len/2 < SPI_INITIAL_FIFO_DEPTH ? len/2 : SPI_INITIAL_FIFO_DEPTH;
- for (int i = 0; i < nf; i++) {
- SPI0_PUSHR = SPI_PUSHR_CONT | SPI_PUSHR_CTAS(1) | 0XFFFF;
- }
- uint8_t* limit = buf + len - 2*nf;
- while (buf < limit) {
- while (!(SPI0_SR & SPI_SR_RXCTR)) {}
- SPI0_PUSHR = SPI_PUSHR_CONT | SPI_PUSHR_CTAS(1) | 0XFFFF;
- uint16_t w = SPI0_POPR;
- *buf++ = w >> 8;
- *buf++ = w & 0XFF;
- }
-
- limit += 2*nf;
- while (buf < limit) {
- while (!(SPI0_SR & SPI_SR_RXCTR)) {}
- uint16_t w = SPI0_POPR;
- *buf++ = w >> 8;
- *buf++ = w & 0XFF;
- }
- return 0;
- }
- static void spiRecIgnore(size_t len) {
-
- SPI0_MCR = SPI_MCR_MSTR | SPI_MCR_CLR_RXF | SPI_MCR_PCSIS(0x1F);
-
-
- if (len & 1) {
- spiRec();
- len--;
- }
-
- int nf = len/2 < SPI_INITIAL_FIFO_DEPTH ? len/2 : SPI_INITIAL_FIFO_DEPTH;
- for (int i = 0; i < nf; i++) {
- SPI0_PUSHR = SPI_PUSHR_CONT | SPI_PUSHR_CTAS(1) | 0XFFFF;
- len -= 2;
- }
-
-
- while (len > 0) {
- while (!(SPI0_SR & SPI_SR_RXCTR)) {}
- SPI0_PUSHR = SPI_PUSHR_CONT | SPI_PUSHR_CTAS(1) | 0XFFFF;
- SPI0_POPR;
- len -= 2;
- }
-
- while (nf > 0) {
- while (!(SPI0_SR & SPI_SR_RXCTR)) {}
- SPI0_POPR;
- nf--;
- }
- }
-
- static void spiSend(uint8_t b) {
- SPI0_MCR |= SPI_MCR_CLR_RXF;
- SPI0_SR = SPI_SR_TCF;
- SPI0_PUSHR = b;
- while (!(SPI0_SR & SPI_SR_TCF)) {}
- }
-
-
- #elif defined(__IMXRT1052__) || defined(__IMXRT1062__)
- #define USE_TEENSY4_SPI
-
- static void spiInit(uint8_t spiRate) {
- switch (spiRate) {
-
- case 0: settings = SPISettings(25200000, MSBFIRST, SPI_MODE0); break;
- case 1: settings = SPISettings(24000000, MSBFIRST, SPI_MODE0); break;
- case 2: settings = SPISettings(8000000, MSBFIRST, SPI_MODE0); break;
- case 3: settings = SPISettings(4000000, MSBFIRST, SPI_MODE0); break;
- case 4: settings = SPISettings(3000000, MSBFIRST, SPI_MODE0); break;
- case 5: settings = SPISettings(2000000, MSBFIRST, SPI_MODE0); break;
- default: settings = SPISettings(400000, MSBFIRST, SPI_MODE0);
- }
- SPI.begin();
- }
-
- static void spiSend(uint8_t b) {
- SPI.transfer(b);
- }
-
- static uint8_t spiRec(void) {
- return SPI.transfer(0xff);
- }
-
- static void spiRec(uint8_t* buf, size_t len) {
- SPI.transfer(buf, len);
- }
-
- static void spiRecIgnore(size_t len) {
- for (size_t i=0; i < len; i++)
- SPI.transfer(0xff);
- }
-
-
- #else
-
-
- static void spiSend(uint8_t b) {
- SPDR = b;
- while (!(SPSR & (1 << SPIF)));
- }
-
- static uint8_t spiRec(void) {
- spiSend(0XFF);
- return SPDR;
- }
-
- #endif
-
-
-
-
-
-
- uint8_t Sd2Card::cardCommand(uint8_t cmd, uint32_t arg)
- {
-
- waitNotBusy(300);
-
-
- spiSend(cmd | 0x40);
-
-
- for (int8_t s = 24; s >= 0; s -= 8) spiSend(arg >> s);
-
-
- uint8_t crc = 0XFF;
- if (cmd == CMD0) crc = 0X95;
- if (cmd == CMD8) crc = 0X87;
- spiSend(crc);
-
-
- for (uint8_t i = 0; ((status_ = spiRec()) & 0X80) && i != 0XFF; i++);
- return status_;
- }
-
- #ifdef SPI_HAS_TRANSACTION
- static uint8_t chip_select_asserted = 0;
- #endif
- void Sd2Card::chipSelectHigh(void) {
- digitalWrite(chipSelectPin_, HIGH);
- #ifdef SPI_HAS_TRANSACTION
- if (chip_select_asserted) {
- chip_select_asserted = 0;
- SPI.endTransaction();
- }
- #endif
- }
-
- void Sd2Card::chipSelectLow(void) {
- #ifdef SPI_HAS_TRANSACTION
- if (!chip_select_asserted) {
- chip_select_asserted = 1;
- SPI.beginTransaction(settings);
- }
- #endif
- digitalWrite(chipSelectPin_, LOW);
- }
-
-
- uint8_t Sd2Card::SD_init(uint8_t sckRateID, uint8_t chipSelectPin) {
- type_ = 0;
- chipSelectPin_ = chipSelectPin;
-
- unsigned int t0 = millis();
- uint32_t arg;
-
- digitalWrite(chipSelectPin_, HIGH);
- pinMode(chipSelectPin_, OUTPUT);
- digitalWrite(chipSelectPin_, HIGH);
-
- #if defined(USE_TEENSY3_SPI)
- spiBegin();
- spiInit(6);
- #elif defined(USE_TEENSY4_SPI)
- spiInit(6);
- pinMode(SS_PIN, OUTPUT);
- digitalWrite(SS_PIN, HIGH);
- #else
-
- pinMode(SPI_MISO_PIN, INPUT);
- pinMode(SPI_MOSI_PIN, OUTPUT);
- pinMode(SPI_SCK_PIN, OUTPUT);
-
- pinMode(SS_PIN, OUTPUT);
- digitalWrite(SS_PIN, HIGH);
-
- SPCR = (1 << SPE) | (1 << MSTR) | (1 << SPR1) | (1 << SPR0);
-
- SPSR &= ~(1 << SPI2X);
- #ifdef SPI_HAS_TRANSACTION
- settings = SPISettings(250000, MSBFIRST, SPI_MODE0);
- #endif
- #endif
-
- #ifdef SPI_HAS_TRANSACTION
- SPI.beginTransaction(settings);
- #endif
- for (uint8_t i = 0; i < 10; i++) spiSend(0XFF);
- #ifdef SPI_HAS_TRANSACTION
- SPI.endTransaction();
- #endif
- chipSelectLow();
-
- while ((status_ = cardCommand(CMD0, 0)) != R1_IDLE_STATE) {
- unsigned int d = millis() - t0;
- if (d > SD_INIT_TIMEOUT) {
- goto fail;
- }
- }
-
- if ((cardCommand(CMD8, 0x1AA) & R1_ILLEGAL_COMMAND)) {
- type_ = SD_CARD_TYPE_SD1;
- } else {
-
- for (uint8_t i = 0; i < 4; i++) status_ = spiRec();
- if (status_ != 0XAA) {
- goto fail;
- }
- type_ = SD_CARD_TYPE_SD2;
- }
-
- arg = (type_ == SD_CARD_TYPE_SD2) ? 0X40000000 : 0;
- while ((status_ = cardAcmd(ACMD41, arg)) != R1_READY_STATE) {
-
- unsigned int d = millis() - t0;
- if (d > SD_INIT_TIMEOUT) {
- goto fail;
- }
- }
-
- if (type_ == SD_CARD_TYPE_SD2) {
- if (cardCommand(CMD58, 0)) {
- goto fail;
- }
- if ((spiRec() & 0XC0) == 0XC0) type_ = SD_CARD_TYPE_SDHC;
-
- for (uint8_t i = 0; i < 3; i++) spiRec();
- }
- chipSelectHigh();
- return setSckRate(sckRateID);
-
- fail:
- chipSelectHigh();
- return false;
- }
-
-
- uint8_t Sd2Card::SD_readBlock(uint32_t block, uint8_t* dst)
- {
-
- if (type_ != SD_CARD_TYPE_SDHC) block <<= 9;
- chipSelectLow();
- if (cardCommand(CMD17, block)) {
- goto fail;
- }
- if (!waitStartBlock()) {
- goto fail;
- }
- #if defined(USE_TEENSY3_SPI) | defined(USE_TEENSY4_SPI)
- spiRec(dst, 512);
- spiRecIgnore(2);
- #else
-
- SPDR = 0XFF;
-
- for (uint16_t i = 0; i < 511; i++) {
- while (!(SPSR & (1 << SPIF)));
- dst[i] = SPDR;
- SPDR = 0XFF;
- }
-
- while (!(SPSR & (1 << SPIF)));
- dst[511] = SPDR;
-
- spiRec();
- spiRec();
- #endif
- chipSelectHigh();
- return true;
-
- fail:
- chipSelectHigh();
- return false;
- }
-
-
- uint8_t Sd2Card::setSckRate(uint8_t sckRateID) {
- #if defined(USE_TEENSY3_SPI) || defined(USE_TEENSY4_SPI)
- spiInit(sckRateID);
- return true;
- #else
- if (sckRateID > 6) sckRateID = 6;
-
- if ((sckRateID & 1) || sckRateID == 6) {
- SPSR &= ~(1 << SPI2X);
- } else {
- SPSR |= (1 << SPI2X);
- }
- SPCR &= ~((1 <<SPR1) | (1 << SPR0));
- SPCR |= (sckRateID & 4 ? (1 << SPR1) : 0)
- | (sckRateID & 2 ? (1 << SPR0) : 0);
- #ifdef SPI_HAS_TRANSACTION
- switch (sckRateID) {
- case 0: settings = SPISettings(8000000, MSBFIRST, SPI_MODE0); break;
- case 1: settings = SPISettings(4000000, MSBFIRST, SPI_MODE0); break;
- case 2: settings = SPISettings(2000000, MSBFIRST, SPI_MODE0); break;
- case 3: settings = SPISettings(1000000, MSBFIRST, SPI_MODE0); break;
- case 4: settings = SPISettings(500000, MSBFIRST, SPI_MODE0); break;
- case 5: settings = SPISettings(250000, MSBFIRST, SPI_MODE0); break;
- default: settings = SPISettings(125000, MSBFIRST, SPI_MODE0);
- }
- #endif
- return true;
- #endif
- }
-
-
- uint8_t Sd2Card::waitNotBusy(unsigned int timeoutMillis) {
- unsigned int t0 = millis();
- unsigned int d;
- do {
- if (spiRec() == 0XFF) return true;
- d = millis() - t0;
- }
- while (d < timeoutMillis);
- return false;
- }
-
-
- uint8_t Sd2Card::waitStartBlock(void) {
- unsigned int t0 = millis();
- while ((status_ = spiRec()) == 0XFF) {
- unsigned int d = millis() - t0;
- if (d > SD_READ_TIMEOUT) {
- return false;
- }
- }
- if (status_ != DATA_START_BLOCK) {
- return false;
- }
- return true;
- }
-
-
- uint8_t Sd2Card::SD_writeBlock(uint32_t blockNumber, const uint8_t* src) {
- #if SD_PROTECT_BLOCK_ZERO
-
- if (blockNumber == 0) {
- goto fail;
- }
- #endif
-
-
- if (type_ != SD_CARD_TYPE_SDHC) blockNumber <<= 9;
- chipSelectLow();
- if (cardCommand(CMD24, blockNumber)) {
- goto fail;
- }
- if (!writeData(DATA_START_BLOCK, src)) goto fail;
-
-
- if (!waitNotBusy(SD_WRITE_TIMEOUT)) {
- goto fail;
- }
-
- if (cardCommand(CMD13, 0) || spiRec()) {
- goto fail;
- }
- chipSelectHigh();
- return true;
-
- fail:
- chipSelectHigh();
- return false;
- }
-
-
- uint8_t Sd2Card::writeData(uint8_t token, const uint8_t* src) {
- #if defined(OPTIMIZE_HARDWARE_SPI) && !defined(USE_TEENSY4_SPI)
-
-
- SPDR = token;
-
-
- for (uint16_t i = 0; i < 512; i += 2) {
- while (!(SPSR & (1 << SPIF)));
- SPDR = src[i];
- while (!(SPSR & (1 << SPIF)));
- SPDR = src[i+1];
- }
-
-
- while (!(SPSR & (1 << SPIF)));
-
- #else
- spiSend(token);
- for (uint16_t i = 0; i < 512; i++) {
- spiSend(src[i]);
- }
- #endif
- spiSend(0xff);
- spiSend(0xff);
-
- status_ = spiRec();
- if ((status_ & DATA_RES_MASK) != DATA_RES_ACCEPTED) {
- return false;
- }
- return true;
- }
|