Thank you. If I recall correctly, some of the numbers posted in Post#14 came from this program. I know that turning off the Serial.print statements will make things run faster.
Ultimately, I want something like:
0 msec corresponding output from sensor
1 msec corresponding output from sensor
2 msec corresponding output from sensor
:
1000 msec corresponding output from sensor
:
10000 msec corresponding output from sensor
I read that the Adalogger with RTC cannot produce such high time resolution. It missed some of the changes in sensory inputs.
// Add sd card
#include <Wire.h>
#include "RTClib.h"
#include <SPI.h>
#include "FS.h" // File system library
#include <SD.h> // SD card library
#define SD_CS 10 // For SD card
#define RECORD_DURATION 10000 // Save data for 10 seconds.
#define SD_MOSI 35
#define SD_MISO 37
#define SD_SCLK 36
#define SD_CS 10
File myFile;
// Keep track of number of seconds to run before stopping the program
int count_seconds = 0;
RTC_PCF8523 rtc;
const int SQW_PIN = 9; // Interrupt pin
volatile unsigned long timestamp;
volatile unsigned long current_time_in_micros;
volatile unsigned long elapsed;
volatile bool new_second = false;
// Interrupt Service Routine
void IRAM_ATTR onSecond() {
timestamp = micros();
new_second = true;
}
void setup() {
Serial.begin(115200);
while (!Serial)
{
// Wait for serial port to connect. Needed for native USB port only
;
}
Serial.println("Setup start");
// ------------------ SD CARD ----------------------
SPI.begin(SD_SCLK, SD_MISO, SD_MOSI, SD_CS);
if (!SD.begin(SD_CS))
{
Serial.println("SD Card MOUNT FAIL");
}
else
{
Serial.println("SD Card MOUNT SUCCESS");
myFile = SD.open("/data.txt", FILE_WRITE);
}
if (!rtc.begin()) {
Serial.println("Couldn't find RTC");
while (1);
}
// Configure SQW pin for 1KHz
//https://adafruit.github.io/RTClib/html/class_r_t_c___p_c_f8523.html#afac1866e69541cfb93251dee683f0e63
rtc.writeSqwPinMode(PCF8523_SquareWave1kHz);
//rtc.writeSqwPinMode(PCF8523_SquareWave4kHz);
//rtc.writeSqwPinMode(PCF8523_SquareWave1HZ);
//rtc.writeSqwPinMode(PCF8523_SquareWave32kHz);
//rtc.writeSqwPinMode(PCF8523_SquareWave8kHz);
pinMode(SQW_PIN, INPUT_PULLUP);
attachInterrupt(digitalPinToInterrupt(SQW_PIN), onSecond, FALLING);
if (! rtc.initialized() || rtc.lostPower()) {
Serial.println("RTC is NOT initialized, let's set the time!");
// When time needs to be set on a new device, or after a power loss, the
// following line sets the RTC to the date & time this sketch was compiled
rtc.adjust(DateTime(F(__DATE__), F(__TIME__)));
// This line sets the RTC with an explicit date & time, for example to set
// January 21, 2014 at 3am you would call:
// rtc.adjust(DateTime(2014, 1, 21, 3, 0, 0));
//
// Note: allow 2 seconds after inserting battery or applying external power
// without battery before calling adjust(). This gives the PCF8523's
// crystal oscillator time to stabilize. If you call adjust() very quickly
// after the RTC is powered, lostPower() may still return true.
}
// When time needs to be re-set on a previously configured device, the
// following line sets the RTC to the date & time this sketch was compiled
// rtc.adjust(DateTime(F(__DATE__), F(__TIME__)));
// This line sets the RTC with an explicit date & time, for example to set
// January 21, 2014 at 3am you would call:
// rtc.adjust(DateTime(2014, 1, 21, 3, 0, 0));
// When the RTC was stopped and stays connected to the battery, it has
// to be restarted by clearing the STOP bit. Let's do this to ensure
// the RTC is running.
rtc.start();
// The PCF8523 can be calibrated for:
// - Aging adjustment
// - Temperature compensation
// - Accuracy tuning
// The offset mode to use, once every two hours or once every minute.
// The offset Offset value from -64 to +63. See the Application Note for calculation of offset values.
// https://www.nxp.com/docs/en/application-note/AN11247.pdf
// The deviation in parts per million can be calculated over a period of observation. Both the drift (which can be negative)
// and the observation period must be in seconds. For accuracy the variation should be observed over about 1 week.
// Note: any previous calibration should cancelled prior to any new observation period.
// Example - RTC gaining 43 seconds in 1 week
float drift = 43; // seconds plus or minus over oservation period - set to 0 to cancel previous calibration.
float period_sec = (7 * 86400); // total obsevation period in seconds (86400 = seconds in 1 day: 7 days = (7 * 86400) seconds )
float deviation_ppm = (drift / period_sec * 1000000); // deviation in parts per million (μs)
float drift_unit = 4.34; // use with offset mode PCF8523_TwoHours
// float drift_unit = 4.069; //For corrections every min the drift_unit is 4.069 ppm (use with offset mode PCF8523_OneMinute)
int offset = round(deviation_ppm / drift_unit);
// rtc.calibrate(PCF8523_TwoHours, offset); // Un-comment to perform calibration once drift (seconds) and observation period (seconds) are correct
// rtc.calibrate(PCF8523_TwoHours, 0); // Un-comment to cancel previous calibration
Serial.print("Offset is "); Serial.println(offset); // Print to control offset
}
void loop() {
if (new_second) {
DateTime now = rtc.now();
new_second = false; // Reset flag
current_time_in_micros = micros();
elapsed = (current_time_in_micros-timestamp)/1000;
//elapsed = (current_time_in_micros-timestamp);
// Print timestamp with approximate milliseconds
Serial.print(now.year(), DEC);
Serial.print('/');
Serial.print(now.month(), DEC);
Serial.print('/');
Serial.print(now.day(), DEC);
Serial.print(" ");
Serial.print(now.hour(), DEC);
Serial.print(':');
Serial.print(now.minute(), DEC);
Serial.print(':');
Serial.print(now.second(), DEC);
Serial.print("micros(): ");
Serial.print(current_time_in_micros);
Serial.print(" - ");
Serial.print("timestamp: ");
Serial.println(timestamp);
Serial.print(" = elapsed: ");
// Calculate milliseconds elapsed since last 1KHz edge
Serial.print(elapsed);
Serial.println(" ms");
// Write data to file
myFile.print(now.hour(), DEC);
myFile.print(':');
myFile.print(now.minute(), DEC);
myFile.print(':');
myFile.print(now.second(), DEC);
myFile.print(':');
myFile.print(current_time_in_micros);
myFile.print(':');
myFile.println(timestamp);
myFile.print(':');
myFile.println(elapsed);
myFile.flush();
if (count_seconds > RECORD_DURATION)
{
myFile.close();
}
else
{
;
//count_seconds++;
}
count_seconds++;
}
//delay(10);
}