| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595 | 
							- // Example to demonstrate write latency for preallocated exFAT files.
 
- // I suggest you write a PC program to convert very large bin files.
 
- //
 
- // The maximum data rate will depend on the quality of your SD,
 
- // the size of the FIFO, and using dedicated SPI.
 
- #include "SdFat.h"
 
- #include "FreeStack.h"
 
- #include "ExFatLogger.h"
 
- //------------------------------------------------------------------------------
 
- // This example was designed for exFAT but will support FAT16/FAT32.
 
- // Note: Uno will not support SD_FAT_TYPE = 3.
 
- // SD_FAT_TYPE = 0 for SdFat/File as defined in SdFatConfig.h,
 
- // 1 for FAT16/FAT32, 2 for exFAT, 3 for FAT16/FAT32 and exFAT.
 
- #define SD_FAT_TYPE 2
 
- //------------------------------------------------------------------------------
 
- // Interval between data records in microseconds.
 
- // Try 250 with Teensy 3.6, Due, or STM32.
 
- // Try 2000 with AVR boards.
 
- // Try 4000 with SAMD Zero boards.
 
- const uint32_t LOG_INTERVAL_USEC = 2000;
 
- // Set USE_RTC nonzero for file timestamps.
 
- // RAM use will be marginal on Uno with RTClib.
 
- // 0 - RTC not used
 
- // 1 - DS1307
 
- // 2 - DS3231
 
- // 3 - PCF8523
 
- #define USE_RTC 0
 
- #if USE_RTC
 
- #include "RTClib.h"
 
- #endif  // USE_RTC
 
- // LED to light if overruns occur.
 
- #define ERROR_LED_PIN -1
 
- /*
 
-   Change the value of SD_CS_PIN if you are using SPI and
 
-   your hardware does not use the default value, SS.
 
-   Common values are:
 
-   Arduino Ethernet shield: pin 4
 
-   Sparkfun SD shield: pin 8
 
-   Adafruit SD shields and modules: pin 10
 
- */
 
- // SDCARD_SS_PIN is defined for the built-in SD on some boards.
 
- #ifndef SDCARD_SS_PIN
 
- const uint8_t SD_CS_PIN = SS;
 
- #else  // SDCARD_SS_PIN
 
- // Assume built-in SD is used.
 
- const uint8_t SD_CS_PIN = SDCARD_SS_PIN;
 
- #endif  // SDCARD_SS_PIN
 
- // FIFO SIZE - 512 byte sectors.  Modify for your board.
 
- #ifdef __AVR_ATmega328P__
 
- // Use 512 bytes for 328 boards.
 
- #define FIFO_SIZE_SECTORS 1
 
- #elif defined(__AVR__)
 
- // Use 2 KiB for other AVR boards.
 
- #define FIFO_SIZE_SECTORS 4
 
- #else  // __AVR_ATmega328P__
 
- // Use 8 KiB for non-AVR boards.
 
- #define FIFO_SIZE_SECTORS 16
 
- #endif  // __AVR_ATmega328P__
 
- // Preallocate 1GiB file.
 
- const uint32_t PREALLOCATE_SIZE_MiB = 1024UL;
 
- // Try max SPI clock for an SD. Reduce SPI_CLOCK if errors occur.
 
- #define SPI_CLOCK SD_SCK_MHZ(50)
 
- // Try to select the best SD card configuration.
 
- #if HAS_SDIO_CLASS
 
- #define SD_CONFIG SdioConfig(FIFO_SDIO)
 
- #elif  ENABLE_DEDICATED_SPI
 
- #define SD_CONFIG SdSpiConfig(SD_CS_PIN, DEDICATED_SPI, SPI_CLOCK)
 
- #else  // HAS_SDIO_CLASS
 
- #define SD_CONFIG SdSpiConfig(SD_CS_PIN, SHARED_SPI, SPI_CLOCK)
 
- #endif  // HAS_SDIO_CLASS
 
- // Save SRAM if 328.
 
- #ifdef __AVR_ATmega328P__
 
- #include "MinimumSerial.h"
 
- MinimumSerial MinSerial;
 
- #define Serial MinSerial
 
- #endif  // __AVR_ATmega328P__
 
- //==============================================================================
 
- // Replace logRecord(), printRecord(), and ExFatLogger.h for your sensors.
 
- void logRecord(data_t* data, uint16_t overrun) {
 
-   if (overrun) {
 
-     // Add one since this record has no adc data. Could add overrun field.
 
-     overrun++;
 
-     data->adc[0] = 0X8000 | overrun;
 
-   } else {
 
-     for (size_t i = 0; i < ADC_COUNT; i++) {
 
-       data->adc[i] = analogRead(i);
 
-     }
 
-   }
 
- }
 
- //------------------------------------------------------------------------------
 
- void printRecord(Print* pr, data_t* data) {
 
-   static uint32_t nr = 0;
 
-   if (!data) {
 
-     pr->print(F("LOG_INTERVAL_USEC,"));
 
-     pr->println(LOG_INTERVAL_USEC);
 
-     pr->print(F("rec#"));
 
-     for (size_t i = 0; i < ADC_COUNT; i++) {
 
-       pr->print(F(",adc"));
 
-       pr->print(i);
 
-     }
 
-     pr->println();
 
-     nr = 0;
 
-     return;
 
-   }
 
-   if (data->adc[0] & 0X8000) {
 
-     uint16_t n = data->adc[0] & 0X7FFF;
 
-     nr += n;
 
-     pr->print(F("-1,"));
 
-     pr->print(n);
 
-     pr->println(F(",overuns"));
 
-   } else {
 
-     pr->print(nr++);
 
-     for (size_t i = 0; i < ADC_COUNT; i++) {
 
-       pr->write(',');
 
-       pr->print(data->adc[i]);
 
-     }
 
-     pr->println();
 
-   }
 
- }
 
- //==============================================================================
 
- const uint64_t PREALLOCATE_SIZE  =  (uint64_t)PREALLOCATE_SIZE_MiB << 20;
 
- // Max length of file name including zero byte.
 
- #define FILE_NAME_DIM 40
 
- // Max number of records to buffer while SD is busy.
 
- const size_t FIFO_DIM = 512*FIFO_SIZE_SECTORS/sizeof(data_t);
 
- #if SD_FAT_TYPE == 0
 
- typedef SdFat sd_t;
 
- typedef File file_t;
 
- #elif SD_FAT_TYPE == 1
 
- typedef SdFat32 sd_t;
 
- typedef File32 file_t;
 
- #elif SD_FAT_TYPE == 2
 
- typedef SdExFat sd_t;
 
- typedef ExFile file_t;
 
- #elif SD_FAT_TYPE == 3
 
- typedef SdFs sd_t;
 
- typedef FsFile file_t;
 
- #else  // SD_FAT_TYPE
 
- #error Invalid SD_FAT_TYPE
 
- #endif  // SD_FAT_TYPE
 
- sd_t sd;
 
- file_t binFile;
 
- file_t csvFile;
 
- // You may modify the filename.  Digits before the dot are file versions.
 
- char binName[] = "ExFatLogger00.bin";
 
- //------------------------------------------------------------------------------
 
- #if USE_RTC
 
- #if USE_RTC == 1
 
- RTC_DS1307 rtc;
 
- #elif USE_RTC == 2
 
- RTC_DS3231 rtc;
 
- #elif USE_RTC == 3
 
- RTC_PCF8523 rtc;
 
- #else  // USE_RTC == type
 
- #error USE_RTC type not implemented.
 
- #endif  // USE_RTC == type
 
- // Call back for file timestamps.  Only called for file create and sync().
 
- void dateTime(uint16_t* date, uint16_t* time, uint8_t* ms10) {
 
-   DateTime now = rtc.now();
 
-   // Return date using FS_DATE macro to format fields.
 
-   *date = FS_DATE(now.year(), now.month(), now.day());
 
-   // Return time using FS_TIME macro to format fields.
 
-   *time = FS_TIME(now.hour(), now.minute(), now.second());
 
-   // Return low time bits in units of 10 ms.
 
-   *ms10 = now.second() & 1 ? 100 : 0;
 
- }
 
- #endif  // USE_RTC
 
- //------------------------------------------------------------------------------
 
- #define error(s) sd.errorHalt(&Serial, F(s))
 
- #define dbgAssert(e) ((e) ? (void)0 : error("assert " #e))
 
- //-----------------------------------------------------------------------------
 
- // Convert binary file to csv file.
 
- void binaryToCsv() {
 
-   uint8_t lastPct = 0;
 
-   uint32_t t0 = millis();
 
-   data_t binData[FIFO_DIM];
 
-   if (!binFile.seekSet(512)) {
 
- 	  error("binFile.seek failed");
 
-   }
 
-   uint32_t tPct = millis();
 
-   printRecord(&csvFile, nullptr);
 
-   while (!Serial.available() && binFile.available()) {
 
-     int nb = binFile.read(binData, sizeof(binData));
 
-     if (nb <= 0 ) {
 
-       error("read binFile failed");
 
-     }
 
-     size_t nr = nb/sizeof(data_t);
 
-     for (size_t i = 0; i < nr; i++) {
 
-       printRecord(&csvFile, &binData[i]);
 
-     }
 
-     if ((millis() - tPct) > 1000) {
 
-       uint8_t pct = binFile.curPosition()/(binFile.fileSize()/100);
 
-       if (pct != lastPct) {
 
-         tPct = millis();
 
-         lastPct = pct;
 
-         Serial.print(pct, DEC);
 
-         Serial.println('%');
 
-         csvFile.sync();
 
-       }
 
-     }
 
-     if (Serial.available()) {
 
-       break;
 
-     }
 
-   }
 
-   csvFile.close();
 
-   Serial.print(F("Done: "));
 
-   Serial.print(0.001*(millis() - t0));
 
-   Serial.println(F(" Seconds"));
 
- }
 
- //------------------------------------------------------------------------------
 
- void clearSerialInput() {
 
-   uint32_t m = micros();
 
-   do {
 
-     if (Serial.read() >= 0) {
 
-       m = micros();
 
-     }
 
-   } while (micros() - m < 10000);
 
- }
 
- //-------------------------------------------------------------------------------
 
- void createBinFile() {
 
-   binFile.close();
 
-   while (sd.exists(binName)) {
 
-     char* p = strchr(binName, '.');
 
-     if (!p) {
 
-       error("no dot in filename");
 
-     }
 
-     while (true) {
 
-       p--;
 
-       if (p < binName || *p < '0' || *p > '9') {
 
-         error("Can't create file name");
 
-       }
 
-       if (p[0] != '9') {
 
-         p[0]++;
 
-         break;
 
-       }
 
-       p[0] = '0';
 
-     }
 
-   }
 
-   if (!binFile.open(binName, O_RDWR | O_CREAT)) {
 
-     error("open binName failed");
 
-   }
 
-   Serial.println(binName);
 
-   if (!binFile.preAllocate(PREALLOCATE_SIZE)) {
 
-     error("preAllocate failed");
 
-   }
 
-   Serial.print(F("preAllocated: "));
 
-   Serial.print(PREALLOCATE_SIZE_MiB);
 
-   Serial.println(F(" MiB"));
 
- }
 
- //-------------------------------------------------------------------------------
 
- bool createCsvFile() {
 
-   char csvName[FILE_NAME_DIM];
 
-   if (!binFile.isOpen()) {
 
-     Serial.println(F("No current binary file"));
 
-     return false;
 
-   }
 
-   // Create a new csvFile.
 
-   binFile.getName(csvName, sizeof(csvName));
 
-   char* dot = strchr(csvName, '.');
 
-   if (!dot) {
 
-     error("no dot in filename");
 
-   }
 
-   strcpy(dot + 1, "csv");
 
-   if (!csvFile.open(csvName, O_WRONLY | O_CREAT | O_TRUNC)) {
 
-     error("open csvFile failed");
 
-   }
 
-   clearSerialInput();
 
-   Serial.print(F("Writing: "));
 
-   Serial.print(csvName);
 
-   Serial.println(F(" - type any character to stop"));
 
-   return true;
 
- }
 
- //-------------------------------------------------------------------------------
 
- void logData() {
 
-   int32_t delta;  // Jitter in log time.
 
-   int32_t maxDelta = 0;
 
-   uint32_t maxLogMicros = 0;
 
-   uint32_t maxWriteMicros = 0;
 
-   size_t maxFifoUse = 0;
 
-   size_t fifoCount = 0;
 
-   size_t fifoHead = 0;
 
-   size_t fifoTail = 0;
 
-   uint16_t overrun = 0;
 
-   uint16_t maxOverrun = 0;
 
-   uint32_t totalOverrun = 0;
 
-   uint32_t fifoBuf[128*FIFO_SIZE_SECTORS];
 
-   data_t* fifoData = (data_t*)fifoBuf;
 
-   // Write dummy sector to start multi-block write.
 
-   dbgAssert(sizeof(fifoBuf) >= 512);
 
-   memset(fifoBuf, 0, sizeof(fifoBuf));
 
-   if (binFile.write(fifoBuf, 512) != 512) {
 
-     error("write first sector failed");
 
-   }
 
-   clearSerialInput();
 
-   Serial.println(F("Type any character to stop"));
 
-   // Wait until SD is not busy.
 
-   while (sd.card()->isBusy()) {}
 
-   // Start time for log file.
 
-   uint32_t m = millis();
 
-   // Time to log next record.
 
-   uint32_t logTime = micros();
 
-   while (true) {
 
-     // Time for next data record.
 
-     logTime += LOG_INTERVAL_USEC;
 
-     // Wait until time to log data.
 
-     delta = micros() - logTime;
 
-     if (delta > 0) {
 
-       Serial.print(F("delta: "));
 
-       Serial.println(delta);
 
-       error("Rate too fast");
 
-     }
 
-     while (delta < 0) {
 
-       delta = micros() - logTime;
 
-     }
 
-     if (fifoCount < FIFO_DIM) {
 
-       uint32_t m = micros();
 
-       logRecord(fifoData + fifoHead, overrun);
 
-       m = micros() - m;
 
-       if (m > maxLogMicros) {
 
-         maxLogMicros = m;
 
-       }
 
-       fifoHead = fifoHead < (FIFO_DIM - 1) ? fifoHead + 1 : 0;
 
-       fifoCount++;
 
-       if (overrun) {
 
-         if (overrun > maxOverrun) {
 
-           maxOverrun = overrun;
 
-         }
 
-         overrun = 0;
 
-       }
 
-     } else {
 
-       totalOverrun++;
 
-       overrun++;
 
-       if (overrun > 0XFFF) {
 
-         error("too many overruns");
 
-       }
 
-       if (ERROR_LED_PIN >= 0) {
 
-         digitalWrite(ERROR_LED_PIN, HIGH);
 
-       }
 
-     }
 
-     // Save max jitter.
 
-     if (delta > maxDelta) {
 
-       maxDelta = delta;
 
-     }
 
-     // Write data if SD is not busy.
 
-     if (!sd.card()->isBusy()) {
 
-       size_t nw = fifoHead > fifoTail ? fifoCount : FIFO_DIM - fifoTail;
 
-       // Limit write time by not writing more than 512 bytes.
 
-       const size_t MAX_WRITE = 512/sizeof(data_t);
 
-       if (nw > MAX_WRITE) nw = MAX_WRITE;
 
-       size_t nb = nw*sizeof(data_t);
 
-       uint32_t usec = micros();
 
-       if (nb != binFile.write(fifoData + fifoTail, nb)) {
 
-         error("write binFile failed");
 
-       }
 
-       usec = micros() - usec;
 
-       if (usec > maxWriteMicros) {
 
-         maxWriteMicros = usec;
 
-       }
 
-       fifoTail = (fifoTail + nw) < FIFO_DIM ? fifoTail + nw : 0;
 
-       if (fifoCount > maxFifoUse) {
 
-         maxFifoUse = fifoCount;
 
-       }
 
-       fifoCount -= nw;
 
-       if (Serial.available()) {
 
-         break;
 
-       }
 
-     }
 
-   }
 
-   Serial.print(F("\nLog time: "));
 
-   Serial.print(0.001*(millis() - m));
 
-   Serial.println(F(" Seconds"));
 
-   binFile.truncate();
 
-   binFile.sync();
 
-   Serial.print(("File size: "));
 
-   // Warning cast used for print since fileSize is uint64_t.
 
-   Serial.print((uint32_t)binFile.fileSize());
 
-   Serial.println(F(" bytes"));
 
-   Serial.print(F("totalOverrun: "));
 
-   Serial.println(totalOverrun);
 
-   Serial.print(F("FIFO_DIM: "));
 
-   Serial.println(FIFO_DIM);
 
-   Serial.print(F("maxFifoUse: "));
 
-   Serial.println(maxFifoUse);
 
-   Serial.print(F("maxLogMicros: "));
 
-   Serial.println(maxLogMicros);
 
-   Serial.print(F("maxWriteMicros: "));
 
-   Serial.println(maxWriteMicros);
 
-   Serial.print(F("Log interval: "));
 
-   Serial.print(LOG_INTERVAL_USEC);
 
-   Serial.print(F(" micros\nmaxDelta: "));
 
-   Serial.print(maxDelta);
 
-   Serial.println(F(" micros"));
 
- }
 
- //------------------------------------------------------------------------------
 
- void openBinFile() {
 
-   char name[FILE_NAME_DIM];
 
-   clearSerialInput();
 
-   Serial.println(F("Enter file name"));
 
-   if (!serialReadLine(name, sizeof(name))) {
 
-     return;
 
-   }
 
-   if (!sd.exists(name)) {
 
-     Serial.println(name);
 
-     Serial.println(F("File does not exist"));
 
-     return;
 
-   }
 
-   binFile.close();
 
-   if (!binFile.open(name, O_RDONLY)) {
 
-     Serial.println(name);
 
-     Serial.println(F("open failed"));
 
-     return;
 
-   }
 
-   Serial.println(F("File opened"));
 
- }
 
- //-----------------------------------------------------------------------------
 
- void printData() {
 
-   if (!binFile.isOpen()) {
 
-     Serial.println(F("No current binary file"));
 
-     return;
 
-   }
 
-   // Skip first dummy sector.
 
-   if (!binFile.seekSet(512)) {
 
-     error("seek failed");
 
-   }
 
-   clearSerialInput();
 
-   Serial.println(F("type any character to stop\n"));
 
-   delay(1000);
 
-   printRecord(&Serial, nullptr);
 
-   while (binFile.available() && !Serial.available()) {
 
-     data_t record;
 
-     if (binFile.read(&record, sizeof(data_t)) != sizeof(data_t)) {
 
-       error("read binFile failed");
 
-     }
 
-     printRecord(&Serial, &record);
 
-   }
 
- }
 
- //------------------------------------------------------------------------------
 
- void printUnusedStack() {
 
- #if HAS_UNUSED_STACK
 
-   Serial.print(F("\nUnused stack: "));
 
-   Serial.println(UnusedStack());
 
- #endif  // HAS_UNUSED_STACK
 
- }
 
- //------------------------------------------------------------------------------
 
- bool serialReadLine(char* str, size_t size) {
 
-   size_t n = 0;
 
-   while(!Serial.available()) {
 
-     yield();
 
-   }
 
-   while (true) {
 
-     int c = Serial.read();
 
-     if (c < ' ') break;
 
-     str[n++] = c;
 
-     if (n >= size) {
 
-       Serial.println(F("input too long"));
 
-       return false;
 
-     }
 
-     uint32_t m = millis();
 
-     while (!Serial.available() && (millis() - m) < 100){}
 
-     if (!Serial.available()) break;
 
-   }
 
-   str[n] = 0;
 
-   return true;
 
- }
 
- //------------------------------------------------------------------------------
 
- void testSensor() {
 
-   const uint32_t interval = 200000;
 
-   int32_t diff;
 
-   data_t data;
 
-   clearSerialInput();
 
-   Serial.println(F("\nTesting - type any character to stop\n"));
 
-   delay(1000);
 
-   printRecord(&Serial, nullptr);
 
-   uint32_t m = micros();
 
-   while (!Serial.available()) {
 
-     m += interval;
 
-     do {
 
-       diff = m - micros();
 
-     } while (diff > 0);
 
-     logRecord(&data, 0);
 
-     printRecord(&Serial, &data);
 
-   }
 
- }
 
- //------------------------------------------------------------------------------
 
- void setup() {
 
-   if (ERROR_LED_PIN >= 0) {
 
-     pinMode(ERROR_LED_PIN, OUTPUT);
 
-     digitalWrite(ERROR_LED_PIN, HIGH);
 
-   }
 
-   Serial.begin(9600);
 
-   // Wait for USB Serial
 
-   while (!Serial) {
 
-     yield();
 
-   }
 
-   delay(1000);
 
-   Serial.println(F("Type any character to begin"));
 
-   while (!Serial.available()) {
 
-     yield();
 
-   }
 
-   FillStack();
 
- #if !ENABLE_DEDICATED_SPI
 
-   Serial.println(F(
 
-     "\nFor best performance edit SdFatConfig.h\n"
 
-     "and set ENABLE_DEDICATED_SPI nonzero"));
 
- #endif  // !ENABLE_DEDICATED_SPI
 
-   Serial.print(FIFO_DIM);
 
-   Serial.println(F(" FIFO entries will be used."));
 
-   // Initialize SD.
 
-   if (!sd.begin(SD_CONFIG)) {
 
-     sd.initErrorHalt(&Serial);
 
-   }
 
- #if USE_RTC
 
-   if (!rtc.begin()) {
 
-     error("rtc.begin failed");
 
-   }
 
-   if (!rtc.isrunning()) {
 
-     // Set RTC to sketch compile date & time.
 
-     // rtc.adjust(DateTime(F(__DATE__), F(__TIME__)));
 
-     error("RTC is NOT running!");
 
-   }
 
-   // Set callback
 
-   FsDateTime::setCallback(dateTime);
 
- #endif  // USE_RTC
 
- }
 
- //------------------------------------------------------------------------------
 
- void loop() {
 
-   printUnusedStack();
 
-   // Read any Serial data.
 
-   clearSerialInput();
 
-   if (ERROR_LED_PIN >= 0) {
 
-     digitalWrite(ERROR_LED_PIN, LOW);
 
-   }
 
-   Serial.println();
 
-   Serial.println(F("type: "));
 
-   Serial.println(F("b - open existing bin file"));
 
-   Serial.println(F("c - convert file to csv"));
 
-   Serial.println(F("l - list files"));
 
-   Serial.println(F("p - print data to Serial"));
 
-   Serial.println(F("r - record data"));
 
-   Serial.println(F("t - test without logging"));
 
-   while(!Serial.available()) {
 
-     yield();
 
-   }
 
-   char c = tolower(Serial.read());
 
-   Serial.println();
 
-   if (c == 'b') {
 
-     openBinFile();
 
-   } else if (c == 'c') {
 
-     if (createCsvFile()) {
 
-       binaryToCsv();
 
-     }
 
-   } else if (c == 'l') {
 
-     Serial.println(F("ls:"));
 
-     sd.ls(&Serial, LS_DATE | LS_SIZE);
 
-   } else if (c == 'p') {
 
-     printData();
 
-   } else if (c == 'r') {
 
-     createBinFile();
 
-     logData();
 
-   } else if (c == 't') {
 
-     testSensor();
 
-   } else {
 
-     Serial.println(F("Invalid entry"));
 
-   }
 
- }
 
 
  |