/* * Simple data logger. */ #include #include // SD chip select pin. Be sure to disable any other SPI devices such as Enet. const uint8_t chipSelect = SS; // Interval between data records in milliseconds. // The interval must be greater than the maximum SD write latency plus the // time to acquire and write data to the SD to avoid overrun errors. // Run the bench example to check the quality of your SD card. const uint32_t SAMPLE_INTERVAL_MS = 200; // Log file base name. Must be six characters or less. #define FILE_BASE_NAME "Data" //------------------------------------------------------------------------------ // File system object. SdFat sd; // Log file. SdFile file; // Time in micros for next data record. uint32_t logTime; //============================================================================== // User functions. Edit writeHeader() and logData() for your requirements. const uint8_t ANALOG_COUNT = 4; //------------------------------------------------------------------------------ // Write data header. void writeHeader() { file.print(F("micros")); for (uint8_t i = 0; i < ANALOG_COUNT; i++) { file.print(F(",adc")); file.print(i, DEC); } file.println(); } //------------------------------------------------------------------------------ // Log a data record. void logData() { uint16_t data[ANALOG_COUNT]; // Read all channels to avoid SD write latency between readings. for (uint8_t i = 0; i < ANALOG_COUNT; i++) { data[i] = analogRead(i); } // Write data to file. Start with log time in micros. file.print(logTime); // Write ADC data to CSV record. for (uint8_t i = 0; i < ANALOG_COUNT; i++) { file.write(','); file.print(data[i]); } file.println(); } //============================================================================== // Error messages stored in flash. #define error(msg) sd.errorHalt(F(msg)) //------------------------------------------------------------------------------ void setup() { const uint8_t BASE_NAME_SIZE = sizeof(FILE_BASE_NAME) - 1; char fileName[13] = FILE_BASE_NAME "00.csv"; Serial.begin(9600); while (!Serial) {} // wait for Leonardo delay(1000); Serial.println(F("Type any character to start")); while (!Serial.available()) {} // Initialize the SD card at SPI_HALF_SPEED to avoid bus errors with // breadboards. use SPI_FULL_SPEED for better performance. if (!sd.begin(chipSelect, SPI_HALF_SPEED)) { sd.initErrorHalt(); } // Find an unused file name. if (BASE_NAME_SIZE > 6) { error("FILE_BASE_NAME too long"); } while (sd.exists(fileName)) { if (fileName[BASE_NAME_SIZE + 1] != '9') { fileName[BASE_NAME_SIZE + 1]++; } else if (fileName[BASE_NAME_SIZE] != '9') { fileName[BASE_NAME_SIZE + 1] = '0'; fileName[BASE_NAME_SIZE]++; } else { error("Can't create file name"); } } if (!file.open(fileName, O_CREAT | O_WRITE | O_EXCL)) { error("file.open"); } do { delay(10); } while (Serial.read() >= 0); Serial.print(F("Logging to: ")); Serial.println(fileName); Serial.println(F("Type any character to stop")); // Write data header. writeHeader(); // Start on a multiple of the sample interval. logTime = micros()/(1000UL*SAMPLE_INTERVAL_MS) + 1; logTime *= 1000UL*SAMPLE_INTERVAL_MS; } //------------------------------------------------------------------------------ void loop() { // Time for next record. logTime += 1000UL*SAMPLE_INTERVAL_MS; // Wait for log time. int32_t diff; do { diff = micros() - logTime; } while (diff < 0); // Check for data rate too high. if (diff > 10) { error("Missed data record"); } logData(); // Force data to SD and update the directory entry to avoid data loss. if (!file.sync() || file.getWriteError()) { error("write error"); } if (Serial.available()) { // Close file and stop. file.close(); Serial.println(F("Done")); while(1) {} } }