ArduinoSleep
alternative à
ArduinoproMiniPowerDown
cen'est pas dit mais se rapporte sans dourte à ArduinoProMini (pas Usb, TTL avec convertisseur Usb externe pour la communication)
le souci parait résolu, lire: ArduinoSleepNoReboot , pas d'explication,
pour l'nstant pas trouvé le moyen d'utiliser deux fois de suite un appel à la librairie sleep
donc bypass: reboot
la consommation en veille est de moins de 3 mA.au pif 1000 heures ou plus d'un mois avec un
PowerBank bas de gamme.
(gaffe: vrai si on utilise un cordon
UsbPowerOnly)
// mettre un fil enre pin 9 et pin RST
/************************************************************************************
*
* Name : Sleep_n0m1.h
* Author : Noah Shibley / NoMi Design
* Date : July 10th 2011
* Version : 0.1
* Notes : Some of this code comes from "Cloudy" on the arduino forum
* http://www.arduino.cc/cgi-bin/yabb2/YaBB.pl?num=1292898715
*
* Sleep_n0m1 is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Sleep_n0m1 is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Sleep_n0m1. If not, see <http://www.gnu.org/licenses/>.
*
***********************************************************************************/
#include "Sleep_n0m1.h"
Sleep* Sleep::pSleep = 0;
Sleep::Sleep()
{
pSleep = this; //the ptr points to this object
timeSleep = 0; // total time due to sleep
calibv = 1.0; // ratio of real clock with WDT clock
byte isrcalled = 0; // WDT vector flag
sleepCycleCount = 0;
sleepCycleInterval = 100;
}
/********************************************************************
*
* setSleepMode
*
********************************************************************/
void Sleep::setSleepMode(int mode)
{
sleepMode_ = mode;
}
/********************************************************************
*
* calibrateTime
*
********************************************************************/
void Sleep::calibrateTime(unsigned long sleepTime, boolean &abortCycle) {
// timer0 continues to run in idle sleep mode
// * SLEEP_MODE_IDLE -the least power savings
// * SLEEP_MODE_ADC
// * SLEEP_MODE_PWR_SAVE
// * SLEEP_MODE_STANDBY
// * SLEEP_MODE_PWR_DOWN -the most power savings
// set_sleep_mode(SLEEP_MODE_IDLE);
set_sleep_mode(SLEEP_MODE_STANDBY);
long tt1=millis();
sleepWDT(sleepTime,abortCycle);
long tt2=millis();
calibv = (float) sleepTime/(tt2-tt1);
//Serial.println(calibv);
}
/********************************************************************
*
* WDTMillis
*
********************************************************************/
unsigned long Sleep::WDTMillis() {
return millis()+timeSleep;
}
/********************************************************************
*
* sleepNow
*
********************************************************************/
void Sleep::sleepInterrupt(int interrupt,int mode) {
if(mode == FALLING || mode == LOW)
{
int pin = interrupt + 2; //will fail on the mega
pinMode (pin, INPUT);
digitalWrite (pin, HIGH);
}
set_sleep_mode(sleepMode_);
sleep_enable();
attachInterrupt(interrupt,sleepHandler,mode);
sei(); //make sure interrupts are on!
sleep_mode();
//----------------------------- ZZZZZZ sleeping here----------------------
sleep_disable(); //disable sleep, awake now
detachInterrupt(interrupt);
}
/********************************************************************
*
* sleepDelay
*
********************************************************************/
void Sleep::sleepDelay(unsigned long sleepTime){
boolean abortCycle = false;
sleepDelay(sleepTime,abortCycle);
}
/********************************************************************
*
* sleepDelay
*
********************************************************************/
void Sleep::sleepDelay(unsigned long sleepTime, boolean &abortCycle) {
ADCSRA &= ~(1<<ADEN); // adc off
// PRR = 0xEF; // modules off
++sleepCycleCount;
sleepCycleCount = sleepCycleCount % sleepCycleInterval; //recalibrate every interval cycles
if(sleepCycleCount == 1)
{
calibrateTime(sleepTime,abortCycle);
}
else
{
set_sleep_mode(sleepMode_);
int trem = sleepWDT(sleepTime*calibv,abortCycle);
timeSleep += (sleepTime-trem);
}
// PRR = 0x00; //modules on
ADCSRA |= (1<<ADEN); // adc on
}
/********************************************************************
*
* sleepWDT
*
********************************************************************/
int Sleep::sleepWDT(unsigned long remainTime, boolean &abortCycle) {
#if defined(WDP3)
byte WDTps = 9; // WDT Prescaler value, 9 = 8192ms
#else
byte WDTps = 7; // WDT Prescaler value, 7 = 2048ms
#endif
isrcalled = 0;
sleep_enable();
while(remainTime > 0) {
//work out next prescale unit to use
while ((0x10<<WDTps) > remainTime && WDTps > 0) {
WDTps--;
}
// send prescaler mask to WDT_On
WDT_On((WDTps & 0x08 ? (1<<WDP3) : 0x00) | (WDTps & 0x07));
isrcalled=0;
while (isrcalled==0 && abortCycle == false) {
#if defined(__AVR_ATmega328P__)
// turn bod off
MCUCR |= (1<<BODS) | (1<<BODSE);
MCUCR &= ~(1<<BODSE); // must be done right before sleep
#endif
sleep_cpu(); // sleep here
}
// calculate remaining time
remainTime -= (0x10<<WDTps);
if ((long) remainTime < 0 ) {remainTime = 0;} //check for unsigned underflow, by converting to signed
}
sleep_disable();
return remainTime;
}
/********************************************************************
*
* WDT_On
*
********************************************************************/
void Sleep::WDT_On(byte psMask)
{
// prepare timed sequence first
byte ps = (psMask | (1<<WDIE)) & ~(1<<WDE);
cli();
wdt_reset();
/* Clear WDRF in MCUSR */
MCUSR &= ~(1<<WDRF);
// start timed sequence
WDTCSR |= (1<<WDCE) | (1<<WDE);
// set new watchdog timeout value
WDTCSR = ps;
sei();
}
/********************************************************************
*
* WDT_Off
*
********************************************************************/
void Sleep::WDT_Off() {
cli();
wdt_reset();
/* Clear WDRF in MCUSR */
MCUSR &= ~(1<<WDRF);
/* Write logical one to WDCE and WDE */
/* Keep old prescaler setting to prevent unintentional time-out */
WDTCSR |= (1<<WDCE) | (1<<WDE);
/* Turn off WDT */
WDTCSR = 0x00;
sei();
}
/********************************************************************
*
* sleepHandler ISR
*
********************************************************************/
void sleepHandler(void)
{
}
/********************************************************************
*
* WDT ISR
*
********************************************************************/
ISR(WDT_vect) {
Sleep::pSleep->WDT_Off();
Sleep::pSleep->isrcalled=1;
}
#include <Sleep_n0m1.h>
Sleep sleep;
unsigned long sleepTime; //how long you want the arduino to sleep
// Pin 13 has an LED connected on most Arduino boards.
// give it a name:
int led = 13;
int resetPin=9;
void setup()
{
digitalWrite(resetPin,HIGH);
delay(200);
pinMode(resetPin,OUTPUT);
Serial.begin(115200);
sleepTime = 50000; //set sleep time in ms, max sleep time is 49.7 days
}
void loop()
{
delay(100); ////delays are just for serial print, without serial they can be removed
Serial.println("execute your code here version 3 pwr down");
Serial.println("5 flash at 1 Hz");
// 10 blink
for (int ipof=1;ipof<5;ipof++) {
digitalWrite(led, HIGH); // turn the LED on (HIGH is the voltage level)
Serial.println("On");
delay(1000); // wait for a second
Serial.println("Off");
digitalWrite(led, LOW); // turn the LED off by making the voltage LOW
delay(1000); // wait for a second
}
Serial.print("sleeping for ");
Serial.println(sleepTime);
delay(100); //delay to allow serial to fully print before sleep
sleep.pwrDownMode(); //set sleep mode
sleep.sleepDelay(sleepTime); //sleep for: sleepTime
delay(100); ////delays are just for serial print, without serial they can be removed
Serial.println("Awaked...");
delay(1000);
// reboot !
digitalWrite(resetPin,LOW);
}