Skip to content

Instantly share code, notes, and snippets.

@dwblair
Created January 11, 2016 15:44
Show Gist options
  • Save dwblair/eb7246e24ac0e9addbe4 to your computer and use it in GitHub Desktop.
Save dwblair/eb7246e24ac0e9addbe4 to your computer and use it in GitHub Desktop.
#include <JeeLib.h>
#include <Wire.h>
#include <SPI.h>
#include <RTClib.h>
#include <RTC_DS3231.h>
#include<stdlib.h>
#include <SD.h>
//sleeping stuff
ISR(WDT_vect) { Sleepy::watchdogEvent(); }
//RTC stuff
RTC_DS3231 RTC;
//led
#define led 9
//battery stuff
#define batteryAnalogMeasurePin A3
#define batteryReadCircuitSwitch 4
// Thermistor definitions ------------------
// which analog pin to connect
#define THERMISTORPIN A0
// resistance at 25 degrees C
#define THERMISTORNOMINAL 10000
// temp. for nominal resistance (almost always 25 C)
#define TEMPERATURENOMINAL 25
// how many samples to take and average, more takes longer
// but is more 'smooth'
#define NUMSAMPLES 5
// The beta coefficient of the thermistor (usually 3000-4000)
#define BCOEFFICIENT 3950
// the value of the 'other' resistor
#define SERIESRESISTOR 9550
// debugging -- only do Serial output if debuggin
#define debug 1 // 0: don't print anything out; 1: print out debugging statements
// how long to sleep between measurements
#define sleepSeconds 3
// Thermistor variables ---------------
int samples[NUMSAMPLES];
// conductivity variables -----------------------------
int interruptPin = 1; //corresponds to D3, the interrupt pin for the conductivity probe
long pulseCount = 0; //a pulse counter variable
unsigned long pulseTime,lastTime, duration, totalDuration;
int samplingPeriod=3; // the number of seconds to measure 555 oscillations
int sensorBoard = 8; // the pin that powers the 555 subcircuit
// SD card details -----------------------
const int chipSelect = 7;
int SDpower = 6;
void setup() {
if (debug) {
Serial.begin(115200);
Serial.println("Debugging on.");
}
pinMode(SDpower,OUTPUT);
digitalWrite(SDpower,LOW);
if (!SD.begin(chipSelect)) {
if (debug) Serial.println("Card failed, or not present");
// don't do anything more:
for (int j=0;j<20;j++) {
digitalWrite(led,HIGH);
delay(10);
digitalWrite(led,LOW);
}
}
else if(debug) { Serial.println("SD Card present.");}
// begin I2C protocol (necessary for RTC, and any other I2C on board
Wire.begin();
// RTC -------------------------
initialize_RTC(); // NOTE: need to initialize I2C first -- but also for any other I2C library
// Note -- will say "RTC is NOT running" when debugging --
// set mode for battery circuit control pin, and turn the circuit off
pinMode(batteryReadCircuitSwitch,OUTPUT);
pinMode(sensorBoard,OUTPUT);
digitalWrite(batteryReadCircuitSwitch, HIGH);
pinMode(led, OUTPUT);
pinMode(sensorBoard,OUTPUT); //turns on the 555 timer and thermistor subcircuit
digitalWrite(sensorBoard,LOW); //turns on the 555 timer and thermistor subcircuit
}
void loop () {
uint8_t i;
//measure the battery --------------------
digitalWrite(batteryReadCircuitSwitch, LOW); //turn on battery measurement circuit
int batteryLevel = analogRead(batteryAnalogMeasurePin);
digitalWrite(batteryReadCircuitSwitch, HIGH); //turn off the battery measurement circuit
//measure temp -------------------------
//take temp samples
// take N samples in a row, with a slight delay
for (i=0; i< NUMSAMPLES; i++) {
samples[i] = analogRead(THERMISTORPIN);
//Serial.println(analogRead(THERMISTORPIN));
delay(10);
}
//measure conductivity ---------------
pulseCount=0; //reset the pulse counter
totalDuration=0; //reset the totalDuration of all pulses measured
attachInterrupt(interruptPin,onPulse,RISING); //attach an interrupt counter to interrupt pin 1 (digital pin #3) -- the only other possible pin on the 328p is interrupt pin #0 (digital pin #2)
pulseTime=micros(); // start the stopwatch
delay(samplingPeriod*1000); //give ourselves samplingPeriod seconds to make this measurement, during which the "onPulse" function will count up all the pulses, and sum the total time they took as 'totalDuration'
detachInterrupt(interruptPin); //we've finished sampling, so detach the interrupt function -- don't count any more pulses
//analyze temp -----------------------------
float average;
average = 0;
for (i=0; i< NUMSAMPLES; i++) {
average += samples[i];
}
average /= NUMSAMPLES;
// convert the value to resistance
average = 1023 / average - 1;
average = SERIESRESISTOR / average;
//Serial.print("Thermistor resistance ");
//Serial.println(average);
float steinhart;
steinhart = average / THERMISTORNOMINAL; // (R/Ro)
steinhart = log(steinhart); // ln(R/Ro)
steinhart /= BCOEFFICIENT; // 1/B * ln(R/Ro)
steinhart += 1.0 / (TEMPERATURENOMINAL + 273.15); // + (1/To)
steinhart = 1.0 / steinhart; // Invert
steinhart -= 273.15;
// analyze conductivity ---------------------------------
float freqHertz;
if (pulseCount>0) { //use this logic in case something went wrong
double durationS=(totalDuration/double(pulseCount))/1000000.; //the total duration, in seconds, per pulse (note that totalDuration was in microseconds)
freqHertz=1./durationS;
}
else {
freqHertz=0.;
}
// Get onboard temp from the RTC -------------
float rtcTemp = RTC.getTempAsFloat();
//get the time from the RTC -----------------
DateTime now = RTC.now();
long unixNow = now.unixtime();
// Write to SD card -------------------------------------
// make a string for assembling the data to log:
String dataString = "";
// dataString += String(unixNow);
dataString += now.unixtime();
dataString += ",";
dataString += now.year();
dataString += "-";
dataString += padInt(now.month(), 2);
dataString += "-";
dataString += padInt(now.day(), 2);
dataString += " ";
dataString += padInt(now.hour(), 2);
dataString += ":";
dataString += padInt(now.minute(), 2);
dataString += ":";
dataString += padInt(now.second(), 2);
dataString += ",";
char buffer[10];
dataString += dtostrf(rtcTemp, 5, 2, buffer);
dataString += ",";
dataString += dtostrf(steinhart,5,2,buffer);
dataString += ",";
dataString += dtostrf(freqHertz,6,2,buffer);
dataString += ",";
dataString += String(batteryLevel);
// open the file. note that only one file can be open at a time,
// so you have to close this one before opening another.
File dataFile = SD.open("datalog.txt", FILE_WRITE);
// if the file is available, write to it:
if (dataFile) {
dataFile.println(dataString);
dataFile.close();
// print to the serial port too:
if(debug) Serial.println(dataString);
digitalWrite(led, HIGH);
delay(20);
digitalWrite(led, LOW);
}
// if the file isn't open, pop up an error:
else {
//Serial.println("error opening datalog.txt");
}
// Finished with everything -- go to sleep
go_to_sleep_seconds(sleepSeconds); //
}
// Useful Functions -------------- ------------
void go_to_sleep_seconds(int seconds) {
int LOG_INTERVAL_BASE = 1000; // 1 sec
for (int k=0;k<seconds;k++) {
Sleepy::loseSomeTime(LOG_INTERVAL_BASE); //-- will interfere with serial, so don't use when debugging
}
}
// RTC functions
int initialize_RTC() {
RTC.begin();
// check on the RTC
if (! RTC.isrunning()) {
if (debug) Serial.println("RTC is NOT running!");
// following line sets the RTC to the date & time this sketch was compiled
RTC.adjust(DateTime(__DATE__, __TIME__));
}
DateTime now = RTC.now();
DateTime compiled = DateTime(__DATE__, __TIME__);
if (now.unixtime() < compiled.unixtime()) {
if(debug) Serial.println("RTC is older than compile time! Updating");
RTC.adjust(DateTime(__DATE__, __TIME__));
}
}
String padInt(int x, int pad) {
String strInt = String(x);
String str = "";
if (strInt.length() >= pad) {
return strInt;
}
for (int i=0; i < (pad-strInt.length()); i++) {
str += "0";
}
str += strInt;
return str;
}
String int2string(int x) {
// formats an integer as a string assuming x is in 1/100ths
String str = String(x);
int strLen = str.length();
if (strLen <= 2) {
str = "0." + str;
} else if (strLen <= 3) {
str = str.substring(0, 1) + "." + str.substring(1);
} else if (strLen <= 4) {
str = str.substring(0, 2) + "." + str.substring(2);
} else {
str = "-9999";
}
return str;
}
// Conductivity Measurement pulse count --------------
void onPulse()
{
pulseCount++;
//Serial.print("pulsecount=");
//Serial.println(pulseCount);
lastTime = pulseTime;
pulseTime = micros();
duration=pulseTime-lastTime;
totalDuration+=duration;
//Serial.println(totalDuration);
}
@dwblair
Copy link
Author

dwblair commented Jan 11, 2016

Requirements

Procedure:

  1. Place a CR1220 coin cell battery into the holder, with the "+" side facing "up"
  2. Place a microSD card into the microSD card holder
  3. Connect a Riffle-CT-0.1 Sensor Board, with a thermistor in the "THRM" screw terminal block
  4. Upload the above code using "Arduino UNO" as the board profile

Expected Output

The code is set up to measure temperature from the onboard RTC, temperature via the thermistor, the current battery level, and write these values to the SD card.

The format is:

UnixTime, HumanReadableTime, RTC_temp, thermistor_temp, conductivity_frequency, battery_level

Below is example output using the code "as-is", powered via USB, without plugging in a sensor board or a lithium battery. The thermistor temperature readings, conductivity frequencies, and battery level readouts are spurious:

Debugging on.
SD Card present.
1452508911,2016-01-11 10:41:51,25.25,20.09, 0.00,649
1452508917,2016-01-11 10:41:57,25.25,14.77, 0.00,645
1452508924,2016-01-11 10:42:04,25.25,22.28, 0.00,601
1452508930,2016-01-11 10:42:10,25.25,18.50, 0.00,665

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment