Move libraries their own directory
This commit is contained in:
263
arduino/libraries/Chronodot/Chronodot.cpp
Normal file
263
arduino/libraries/Chronodot/Chronodot.cpp
Normal file
@ -0,0 +1,263 @@
|
||||
// Code by JeeLabs http://news.jeelabs.org/code/
|
||||
// Released to the public domain! Enjoy!
|
||||
//
|
||||
// Modified for Chronodot / DS3132 with
|
||||
// temperature data by Stephanie Maks
|
||||
// http://planetstephanie.net/
|
||||
|
||||
|
||||
#include <Wire.h>
|
||||
#include <avr/pgmspace.h>
|
||||
#include "Chronodot.h"
|
||||
#if ARDUINO >= 100
|
||||
#include "Arduino.h"
|
||||
#else
|
||||
#include "WProgram.h"
|
||||
#endif
|
||||
|
||||
#define CHRONODOT_ADDRESS 0x68
|
||||
#define SECONDS_PER_DAY 86400L
|
||||
|
||||
#define SECONDS_FROM_1970_TO_2000 946684800
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// utility code, some of this could be exposed in the DateTime API if needed
|
||||
|
||||
static uint8_t daysInMonth [] PROGMEM = { 31,28,31,30,31,30,31,31,30,31,30,31 };
|
||||
|
||||
// number of days since 2000/01/01, valid for 2001..2099
|
||||
static uint16_t date2days(uint16_t y, uint8_t m, uint8_t d) {
|
||||
if (y >= 2000)
|
||||
y -= 2000;
|
||||
uint16_t days = d;
|
||||
for (uint8_t i = 1; i < m; ++i)
|
||||
days += pgm_read_byte(daysInMonth + i - 1);
|
||||
if (m > 2 && y % 4 == 0)
|
||||
++days;
|
||||
return days + 365 * y + (y + 3) / 4 - 1;
|
||||
}
|
||||
|
||||
static long time2long(uint16_t days, uint8_t h, uint8_t m, uint8_t s) {
|
||||
return ((days * 24L + h) * 60 + m) * 60 + s;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// DateTime implementation - ignores time zones and DST changes
|
||||
// NOTE: also ignores leap seconds, see http://en.wikipedia.org/wiki/Leap_second
|
||||
|
||||
DateTime::DateTime (uint32_t t) {
|
||||
t -= SECONDS_FROM_1970_TO_2000; // bring to 2000 timestamp from 1970
|
||||
|
||||
ss = t % 60;
|
||||
t /= 60;
|
||||
mm = t % 60;
|
||||
t /= 60;
|
||||
hh = t % 24;
|
||||
uint16_t days = t / 24;
|
||||
uint8_t leap;
|
||||
for (yOff = 0; ; ++yOff) {
|
||||
leap = yOff % 4 == 0;
|
||||
if (days < 365 + leap)
|
||||
break;
|
||||
days -= 365 + leap;
|
||||
}
|
||||
for (m = 1; ; ++m) {
|
||||
uint8_t daysPerMonth = pgm_read_byte(daysInMonth + m - 1);
|
||||
if (leap && m == 2)
|
||||
++daysPerMonth;
|
||||
if (days < daysPerMonth)
|
||||
break;
|
||||
days -= daysPerMonth;
|
||||
}
|
||||
d = days + 1;
|
||||
}
|
||||
|
||||
DateTime::DateTime (uint16_t year, uint8_t month, uint8_t day, uint8_t hour, uint8_t min, uint8_t sec, int tempF, float tempC) {
|
||||
if (year >= 2000)
|
||||
year -= 2000;
|
||||
yOff = year;
|
||||
m = month;
|
||||
d = day;
|
||||
hh = hour;
|
||||
mm = min;
|
||||
ss = sec;
|
||||
ttf = tempF;
|
||||
ttc = tempC;
|
||||
}
|
||||
|
||||
static uint8_t conv2d(const char* p) {
|
||||
uint8_t v = 0;
|
||||
if ('0' <= *p && *p <= '9')
|
||||
v = *p - '0';
|
||||
return 10 * v + *++p - '0';
|
||||
}
|
||||
|
||||
// A convenient constructor for using "the compiler's time":
|
||||
// DateTime now (__DATE__, __TIME__);
|
||||
// NOTE: using PSTR would further reduce the RAM footprint
|
||||
DateTime::DateTime (const char* date, const char* time) {
|
||||
// sample input: date = "Dec 26 2009", time = "12:34:56"
|
||||
yOff = conv2d(date + 9);
|
||||
// Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec
|
||||
switch (date[0]) {
|
||||
case 'J': m = date[1] == 'a' ? 1 : m = date[2] == 'n' ? 6 : 7; break;
|
||||
case 'F': m = 2; break;
|
||||
case 'A': m = date[2] == 'r' ? 4 : 8; break;
|
||||
case 'M': m = date[2] == 'r' ? 3 : 5; break;
|
||||
case 'S': m = 9; break;
|
||||
case 'O': m = 10; break;
|
||||
case 'N': m = 11; break;
|
||||
case 'D': m = 12; break;
|
||||
}
|
||||
d = conv2d(date + 4);
|
||||
hh = conv2d(time);
|
||||
mm = conv2d(time + 3);
|
||||
ss = conv2d(time + 6);
|
||||
}
|
||||
|
||||
uint8_t DateTime::dayOfWeek() const {
|
||||
uint16_t day = date2days(yOff, m, d);
|
||||
return (day + 6) % 7; // Jan 1, 2000 is a Saturday, i.e. returns 6
|
||||
}
|
||||
|
||||
long DateTime::secondstime(void) const {
|
||||
long t;
|
||||
uint16_t days = date2days(yOff, m, d);
|
||||
t = time2long(days, hh, mm, ss);
|
||||
return t;
|
||||
}
|
||||
|
||||
uint32_t DateTime::unixtime(void) const {
|
||||
uint32_t t;
|
||||
uint16_t days = date2days(yOff, m, d);
|
||||
t = time2long(days, hh, mm, ss);
|
||||
t += SECONDS_FROM_1970_TO_2000; // seconds from 1970 to 2000
|
||||
return t;
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// RTC_DS3231 implementation
|
||||
|
||||
static uint8_t bcd2bin (uint8_t val) { return val - 6 * (val >> 4); }
|
||||
static uint8_t bin2bcd (uint8_t val) { return val + 6 * (val / 10); }
|
||||
|
||||
uint8_t Chronodot::begin(void) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
uint8_t Chronodot::isrunning(void) {
|
||||
Wire.beginTransmission(CHRONODOT_ADDRESS);
|
||||
#if ARDUINO >= 100
|
||||
Wire.write((byte)0x0F);
|
||||
#else
|
||||
Wire.send(0x0F);
|
||||
#endif
|
||||
Wire.endTransmission();
|
||||
|
||||
Wire.requestFrom(CHRONODOT_ADDRESS, 1);
|
||||
#if ARDUINO >= 100
|
||||
uint8_t ss = Wire.read();
|
||||
#else
|
||||
uint8_t ss = Wire.receive();
|
||||
#endif
|
||||
return !(ss>>7);
|
||||
}
|
||||
|
||||
void Chronodot::adjust(const DateTime& dt) {
|
||||
// send new date & time to chronodot
|
||||
Wire.beginTransmission(CHRONODOT_ADDRESS);
|
||||
#if ARDUINO >= 100
|
||||
Wire.write((byte)0); // memory address
|
||||
Wire.write(bin2bcd(dt.second())); // byte 0
|
||||
Wire.write(bin2bcd(dt.minute())); // byte 1
|
||||
Wire.write(bin2bcd(dt.hour())); // byte 2
|
||||
Wire.write(bin2bcd(0)); // byte 3
|
||||
Wire.write(bin2bcd(dt.day())); // byte 4
|
||||
Wire.write(bin2bcd(dt.month())); // byte 5
|
||||
Wire.write(bin2bcd(dt.year() - 2000)); // byte 6
|
||||
#else
|
||||
Wire.send(0);
|
||||
Wire.send(bin2bcd(dt.second()));
|
||||
Wire.send(bin2bcd(dt.minute()));
|
||||
Wire.send(bin2bcd(dt.hour()));
|
||||
Wire.send(bin2bcd(0));
|
||||
Wire.send(bin2bcd(dt.day()));
|
||||
Wire.send(bin2bcd(dt.month()));
|
||||
Wire.send(bin2bcd(dt.year() - 2000));
|
||||
#endif
|
||||
Wire.endTransmission();
|
||||
// now get the control byte - we need to set bit 7 to zero
|
||||
Wire.beginTransmission(CHRONODOT_ADDRESS);
|
||||
#if ARDUINO >= 100
|
||||
Wire.write((byte)0x0F);
|
||||
#else
|
||||
Wire.send(0x0F);
|
||||
#endif
|
||||
Wire.endTransmission();
|
||||
Wire.requestFrom(CHRONODOT_ADDRESS, 1);
|
||||
#if ARDUINO >= 100
|
||||
uint8_t ss = Wire.read();
|
||||
#else
|
||||
uint8_t ss = Wire.receive();
|
||||
#endif
|
||||
ss &= ~(1 << 7); // clear OSF bit
|
||||
Wire.beginTransmission(CHRONODOT_ADDRESS);
|
||||
#if ARDUINO >= 100
|
||||
Wire.write((byte)0x0F);
|
||||
Wire.write((byte)ss);
|
||||
#else
|
||||
Wire.send(0x0F);
|
||||
Wire.send(ss);
|
||||
#endif
|
||||
Wire.endTransmission();
|
||||
}
|
||||
|
||||
DateTime Chronodot::now() {
|
||||
Wire.beginTransmission(CHRONODOT_ADDRESS);
|
||||
#if ARDUINO >= 100
|
||||
Wire.write((byte)0);
|
||||
#else
|
||||
Wire.send(0);
|
||||
#endif
|
||||
Wire.endTransmission();
|
||||
|
||||
Wire.requestFrom(CHRONODOT_ADDRESS, 19);
|
||||
byte blah[20];
|
||||
int i;
|
||||
for(i=0; i<20; i++) {
|
||||
#if ARDUINO >= 100
|
||||
blah[i] = Wire.read();
|
||||
#else
|
||||
blah[i] = Wire.receive();
|
||||
#endif
|
||||
}
|
||||
uint8_t ss = bcd2bin(blah[0] & 0x7F);
|
||||
uint8_t mm = bcd2bin(blah[1]);
|
||||
uint8_t hh = bcd2bin(blah[2]);
|
||||
uint8_t d = bcd2bin(blah[4]);
|
||||
uint8_t m = bcd2bin(blah[5]);
|
||||
uint16_t y = bcd2bin(blah[6]) + 2000;
|
||||
float ttc = (float)(int)blah[17];
|
||||
byte portion = blah[18];
|
||||
if(portion == 0b01000000) ttc += 0.25;
|
||||
if(portion == 0b10000000) ttc += 0.5;
|
||||
if(portion == 0b11000000) ttc += 0.75;
|
||||
float degF = (((ttc * 9.0) / 5.0) + 32.5);
|
||||
int ttf = (int)degF;
|
||||
return DateTime (y, m, d, hh, mm, ss, ttf, ttc);
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
// RTC_Millis implementation
|
||||
|
||||
long RTC_Millis::offset = 0;
|
||||
|
||||
void RTC_Millis::adjust(const DateTime& dt) {
|
||||
offset = dt.unixtime() - millis() / 1000;
|
||||
}
|
||||
|
||||
DateTime RTC_Millis::now() {
|
||||
return (uint32_t)(offset + millis() / 1000);
|
||||
}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
60
arduino/libraries/Chronodot/Chronodot.h
Normal file
60
arduino/libraries/Chronodot/Chronodot.h
Normal file
@ -0,0 +1,60 @@
|
||||
// Code by JeeLabs http://news.jeelabs.org/code/
|
||||
// Released to the public domain! Enjoy!
|
||||
//
|
||||
// Modified for Chronodot / DS3132 with
|
||||
// temperature data by Stephanie Maks
|
||||
// http://planetstephanie.net/
|
||||
|
||||
#ifndef CHRONODOT_H
|
||||
#define CHRONODOT_H
|
||||
|
||||
// Simple general-purpose date/time class (no TZ / DST / leap second handling!)
|
||||
class DateTime {
|
||||
public:
|
||||
DateTime (uint32_t t =0);
|
||||
DateTime (uint16_t year, uint8_t month, uint8_t day,
|
||||
uint8_t hour =0, uint8_t min =0, uint8_t sec =0, int tempF =0, float tempC = 0.0);
|
||||
DateTime (const char* date, const char* time);
|
||||
uint16_t year() const { return 2000 + yOff; }
|
||||
uint8_t month() const { return m; }
|
||||
uint8_t day() const { return d; }
|
||||
uint8_t hour() const { return hh; }
|
||||
uint8_t minute() const { return mm; }
|
||||
uint8_t second() const { return ss; }
|
||||
int tempF() const { return ttf; }
|
||||
float tempC() const { return ttc; }
|
||||
uint8_t dayOfWeek() const;
|
||||
|
||||
// 32-bit times as seconds since 1/1/2000
|
||||
long secondstime() const;
|
||||
// 32-bit times as seconds since 1/1/1970
|
||||
uint32_t unixtime(void) const;
|
||||
|
||||
protected:
|
||||
uint8_t yOff, m, d, hh, mm, ss;
|
||||
int ttf;
|
||||
float ttc;
|
||||
};
|
||||
|
||||
// Chronodot based on the DS3231 chip connected via I2C and the Wire library
|
||||
class Chronodot {
|
||||
public:
|
||||
static uint8_t begin(void);
|
||||
static void adjust(const DateTime& dt);
|
||||
uint8_t isrunning(void);
|
||||
static DateTime now();
|
||||
};
|
||||
|
||||
// RTC using the internal millis() clock, has to be initialized before use
|
||||
// NOTE: this clock won't be correct once the millis() timer rolls over (>49d?)
|
||||
class RTC_Millis {
|
||||
public:
|
||||
static void begin(const DateTime& dt) { adjust(dt); }
|
||||
static void adjust(const DateTime& dt);
|
||||
static DateTime now();
|
||||
|
||||
protected:
|
||||
static long offset;
|
||||
};
|
||||
|
||||
#endif CHRONODOT_H
|
Reference in New Issue
Block a user