/* * This sketch attempts to initialize an SD card and analyze its structure. */ #include #include /* * SD chip select pin. Common values are: * * Arduino Ethernet shield, pin 4. * SparkFun SD shield, pin 8. * Adafruit SD shields and modules, pin 10. * Default SD chip select is the SPI SS pin. */ const uint8_t SD_CHIP_SELECT = SS; /* * Set DISABLE_CHIP_SELECT to disable a second SPI device. * For example, with the Ethernet shield, set DISABLE_CHIP_SELECT * to 10 to disable the Ethernet controller. */ const int8_t DISABLE_CHIP_SELECT = -1; SdFat sd; // serial output steam ArduinoOutStream cout(Serial); // global for card size uint32_t cardSize; // global for card erase size uint32_t eraseSize; //------------------------------------------------------------------------------ // store error strings in flash #define sdErrorMsg(msg) sdErrorMsg_P(PSTR(msg)); void sdErrorMsg_P(const char* str) { cout << pgm(str) << endl; if (sd.card()->errorCode()) { cout << pstr("SD errorCode: "); cout << hex << int(sd.card()->errorCode()) << endl; cout << pstr("SD errorData: "); cout << int(sd.card()->errorData()) << dec << endl; } } //------------------------------------------------------------------------------ uint8_t cidDmp() { cid_t cid; if (!sd.card()->readCID(&cid)) { sdErrorMsg("readCID failed"); return false; } cout << pstr("\nManufacturer ID: "); cout << hex << int(cid.mid) << dec << endl; cout << pstr("OEM ID: ") << cid.oid[0] << cid.oid[1] << endl; cout << pstr("Product: "); for (uint8_t i = 0; i < 5; i++) { cout << cid.pnm[i]; } cout << pstr("\nVersion: "); cout << int(cid.prv_n) << '.' << int(cid.prv_m) << endl; cout << pstr("Serial number: ") << hex << cid.psn << dec << endl; cout << pstr("Manufacturing date: "); cout << int(cid.mdt_month) << '/'; cout << (2000 + cid.mdt_year_low + 10 * cid.mdt_year_high) << endl; cout << endl; return true; } //------------------------------------------------------------------------------ uint8_t csdDmp() { csd_t csd; uint8_t eraseSingleBlock; if (!sd.card()->readCSD(&csd)) { sdErrorMsg("readCSD failed"); return false; } if (csd.v1.csd_ver == 0) { eraseSingleBlock = csd.v1.erase_blk_en; eraseSize = (csd.v1.sector_size_high << 1) | csd.v1.sector_size_low; } else if (csd.v2.csd_ver == 1) { eraseSingleBlock = csd.v2.erase_blk_en; eraseSize = (csd.v2.sector_size_high << 1) | csd.v2.sector_size_low; } else { cout << pstr("csd version error\n"); return false; } eraseSize++; cout << pstr("cardSize: ") << 0.000512*cardSize; cout << pstr(" MB (MB = 1,000,000 bytes)\n"); cout << pstr("flashEraseSize: ") << int(eraseSize) << pstr(" blocks\n"); cout << pstr("eraseSingleBlock: "); if (eraseSingleBlock) { cout << pstr("true\n"); } else { cout << pstr("false\n"); } return true; } //------------------------------------------------------------------------------ // print partition table uint8_t partDmp() { cache_t *p = sd.vol()->cacheClear(); if (!p) { sdErrorMsg("cacheClear failed"); return false; } if (!sd.card()->readBlock(0, p->data)) { sdErrorMsg("read MBR failed"); return false; } for (uint8_t ip = 1; ip < 5; ip++) { part_t *pt = &p->mbr.part[ip - 1]; if ((pt->boot & 0X7F) != 0 || pt->firstSector > cardSize) { cout << pstr("\nNo MBR. Assuming Super Floppy format.\n"); return true; } } cout << pstr("\nSD Partition Table\n"); cout << pstr("part,boot,type,start,length\n"); for (uint8_t ip = 1; ip < 5; ip++) { part_t *pt = &p->mbr.part[ip - 1]; cout << int(ip) << ',' << hex << int(pt->boot) << ',' << int(pt->type); cout << dec << ',' << pt->firstSector <<',' << pt->totalSectors << endl; } return true; } //------------------------------------------------------------------------------ void volDmp() { cout << pstr("\nVolume is FAT") << int(sd.vol()->fatType()) << endl; cout << pstr("blocksPerCluster: ") << int(sd.vol()->blocksPerCluster()) << endl; cout << pstr("clusterCount: ") << sd.vol()->clusterCount() << endl; cout << pstr("freeClusters: "); uint32_t volFree = sd.vol()->freeClusterCount(); cout << volFree << endl; float fs = 0.000512*volFree*sd.vol()->blocksPerCluster(); cout << pstr("freeSpace: ") << fs << pstr(" MB (MB = 1,000,000 bytes)\n"); cout << pstr("fatStartBlock: ") << sd.vol()->fatStartBlock() << endl; cout << pstr("fatCount: ") << int(sd.vol()->fatCount()) << endl; cout << pstr("blocksPerFat: ") << sd.vol()->blocksPerFat() << endl; cout << pstr("rootDirStart: ") << sd.vol()->rootDirStart() << endl; cout << pstr("dataStartBlock: ") << sd.vol()->dataStartBlock() << endl; if (sd.vol()->dataStartBlock() % eraseSize) { cout << pstr("Data area is not aligned on flash erase boundaries!\n"); cout << pstr("Download and use formatter from www.sdsd.card()->org/consumer!\n"); } } //------------------------------------------------------------------------------ void setup() { Serial.begin(9600); while(!Serial) {} // wait for Leonardo // use uppercase in hex and use 0X base prefix cout << uppercase << showbase << endl; // pstr stores strings in flash to save RAM cout << pstr("SdFat version: ") << SD_FAT_VERSION << endl; if (DISABLE_CHIP_SELECT < 0) { cout << pstr( "\nAssuming the SD is the only SPI device.\n" "Edit DISABLE_CHIP_SELECT to disable another device.\n"); } else { cout << pstr("\nDisabling SPI device on pin "); cout << int(DISABLE_CHIP_SELECT) << endl; pinMode(DISABLE_CHIP_SELECT, OUTPUT); digitalWrite(DISABLE_CHIP_SELECT, HIGH); } cout << pstr("\nAssuming the SD chip select pin is: ") <= 0) {} // pstr stores strings in flash to save RAM cout << pstr("\ntype any character to start\n"); while (Serial.read() <= 0) {} delay(400); // catch Due reset problem uint32_t t = millis(); // initialize the SD card at SPI_HALF_SPEED to avoid bus errors with // breadboards. use SPI_FULL_SPEED for better performance. if (!sd.cardBegin(SD_CHIP_SELECT, SPI_HALF_SPEED)) { sdErrorMsg("\ncardBegin failed"); return; } t = millis() - t; cardSize = sd.card()->cardSize(); if (cardSize == 0) { sdErrorMsg("cardSize failed"); return; } cout << pstr("\ninit time: ") << t << " ms" << endl; cout << pstr("\nCard type: "); switch (sd.card()->type()) { case SD_CARD_TYPE_SD1: cout << pstr("SD1\n"); break; case SD_CARD_TYPE_SD2: cout << pstr("SD2\n"); break; case SD_CARD_TYPE_SDHC: if (cardSize < 70000000) { cout << pstr("SDHC\n"); } else { cout << pstr("SDXC\n"); } break; default: cout << pstr("Unknown\n"); } if (!cidDmp()) return; if (!csdDmp()) return; uint32_t ocr; if (!sd.card()->readOCR(&ocr)) { sdErrorMsg("\nreadOCR failed"); return; } cout << pstr("OCR: ") << hex << ocr << dec << endl; if (!partDmp()) return; if (!sd.fsBegin()) { sdErrorMsg("\nFile System initialization failed.\n"); return; } volDmp(); }