Well it's been just over a year since this blog started and what better way to celebrate than revising an old project? I'd had a look at Frankenstein's thermometer before, using a neat trick with a mono jack plug and a stereo socket to switch the power. However I wasn't really pleased with this arrangement, because it separated the thermistor from the main box when it wasn't in use.
The idea was to upgrade the firmware to put the ATtiny into deep sleep after a short time. The complete code is now:
The main additions are:
The idea was to upgrade the firmware to put the ATtiny into deep sleep after a short time. The complete code is now:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
* Set IDE to "ATtiny84@1MHz, BOD disabled" | |
* fuses: H:FF, E:DF, L:62 | |
* (Can program with Arduino as ISP using one of these: | |
* http://programmablehardware.blogspot.ie/2011/05/in-system-programming.html ) | |
*/ | |
#include <math.h> | |
#include <Interrupted.h> | |
// thermistor parameters | |
const double r0 = 10000; | |
const double beta = 3977.0; | |
// bias resistor | |
const double rb = 9990; | |
const double zeroC = 273.15; | |
const double tzero = 25 + zeroC; | |
#define IDLE_MS 10000L | |
#define DIGIT_MS 500L | |
#define DEBOUNCE_MS 500L | |
#define THERMISTOR A7 | |
#define BUTTON 9 | |
#define DIVIDER_GND 8 | |
#define A_PIN 6 | |
#define TIMER 0 | |
#define IDLE 1 | |
Analog thermistor(THERMISTOR); | |
Port pins; | |
Pin button(BUTTON, pins); | |
Timer timer(TIMER, DIGIT_MS); | |
Delay idle(IDLE, IDLE_MS); | |
Devices devices; | |
const uint8_t A = _BV(0); | |
const uint8_t B = _BV(1); | |
const uint8_t C = _BV(2); | |
const uint8_t D = _BV(3); | |
const uint8_t E = _BV(4); | |
const uint8_t F = _BV(5); | |
const uint8_t G = _BV(6); | |
static const uint8_t digits[] = { | |
A | B | C | D | E | F, | |
B | C, | |
A | B | G | E | D, | |
A | B | G | C | D, | |
F | G | B | C, | |
A | F | G | C | D, | |
A | F | E | G | C | D, | |
A | B | C, | |
A | B | C | D | E | F | G, | |
A | B | C | F | G, | |
}; | |
static const uint8_t minus = G; | |
#define CENTIGRADE 0 | |
#define FAHRENHEIT 1 | |
#define ABSOLUTE 2 | |
static const uint8_t units[] = { | |
A | F | E | D, | |
A | F | E | G, | |
A | B | C | E | F | G, | |
}; | |
void display(uint8_t bits) | |
{ | |
uint8_t b = 1; | |
for (int i = 0; i < 7; i++, b <<= 1) | |
digitalWrite(A_PIN - i, bits & b); | |
} | |
double read_temp() | |
{ | |
double rt = rb * (1023.0 / thermistor.read() - 1); | |
double rtk = 1.0 / tzero + log(rt / r0) / beta; | |
return 1.0 / rtk; | |
} | |
uint8_t *reformat(int u, double t, uint8_t *buf) | |
{ | |
uint8_t *p = buf; | |
*p++ = units[u]; | |
if (u != ABSOLUTE) { | |
t -= zeroC; | |
if (u == FAHRENHEIT) | |
t = t * 9.0 / 5.0 + 32; | |
} | |
if (t < 0) { | |
t = -t; | |
*p++ = minus; | |
} | |
int temp = (int)(t + 0.5); | |
bool first = true; | |
for (int pow = 100; pow > 0; pow /= 10) { | |
int dig = temp / pow; | |
if (!first || dig != 0 || pow == 1) { | |
temp -= dig * pow; | |
*p++ = digits[dig]; | |
first = false; | |
} | |
} | |
*p++ = 0; | |
display(buf[0]); | |
return buf+1; | |
} | |
void setup(void) | |
{ | |
devices.add(button); | |
devices.add(thermistor); | |
devices.add(timer); | |
devices.add(idle); | |
devices.begin(); | |
for (int i = 0; i < 7; i++) | |
pinMode(A_PIN - i, OUTPUT); | |
pinMode(DIVIDER_GND, OUTPUT); | |
digitalWrite(DIVIDER_GND, LOW); | |
thermistor.enable(); | |
idle.enable(); | |
} | |
void loop(void) | |
{ | |
static int u = CENTIGRADE; | |
static bool sleeping = false; | |
static uint8_t buf[8], *p = buf; | |
static double curr_temp = 0.0; | |
static uint32_t last; | |
uint32_t now = millis(); | |
switch (devices.select()) { | |
case BUTTON: | |
if (sleeping) { | |
sleeping = false; | |
last = now; | |
pinMode(DIVIDER_GND, OUTPUT); | |
digitalWrite(DIVIDER_GND, LOW); | |
thermistor.wake(); | |
thermistor.enable(); | |
idle.enable(); | |
} else if (now - last > DEBOUNCE_MS) { | |
last = now; | |
if (++u > ABSOLUTE) | |
u = CENTIGRADE; | |
p = reformat(u, curr_temp, buf); | |
} | |
break; | |
case TIMER: | |
if (*p) { | |
display(*p++); | |
timer.enable(); | |
} else | |
thermistor.enable(); | |
break; | |
case THERMISTOR: | |
curr_temp = read_temp(); | |
p = reformat(u, curr_temp, buf); | |
timer.enable(); | |
break; | |
case IDLE: | |
idle.disable(); | |
timer.disable(); | |
thermistor.disable(); | |
thermistor.sleep(); | |
pinMode(DIVIDER_GND, INPUT); | |
display(0); | |
sleeping = true; | |
break; | |
} | |
} |
The main additions are:
- the sleep() routine which causes the ATtiny to enter a power-down mode,
- the PCINT1_vect which wakes it up again when the push-button is pressed.
The sleep routine switches off all of the LEDs and the ADC before putting the ATtiny to sleep. When it wakes up again, the ADC must be re-enabled.
With the new software installed, the device consumes about 11mA when on but 0.16mA when off. With a nominal capacity of 1000mAh, this would drain the batteries in about 6000 hours (or 250 days). This was a bit surprising, as was the discovery that this current was still drawn when the AVR was removed altogether. This indicated that the source of the drain was the voltage divider comprising the thermistor and a 10k resistor. (See the circuit diagram in the original posting.) At room temperature, the current drawn would be 3v/20k or 0.15mA.
Rather than complicate the circuit with a MOSFET switch, we decided simply to increase the value of the bias resistor from 10k to 47k. This reduced the current by a factor of 4 increasing the battery life to over 2 years. (The calculation of the Thermistor equation has been cleaned up a bit too: the original made the assumption that the 10k resistor would cancel the thermistor's resistance at 25C.)
No comments:
Post a Comment