@mrme @Verkehrsrot OK so I did that for ESP deepSleep and using console to power or not the OLED screen. But now I’m running into a deeper problem
Each time I sleep and wake up, it seems the program restarts from scratch (as counter displayed on screen is not incremented, and ABP packet counter is not incremented neither). Here is the sketch I use:
/* **************************************************************
* Arduino sketch
* Author: Martijn Quaedvlieg / Jan de Laet (january 2017)
* Generated with Generate script by Jan de Laet
* Modified 12-1-2018 by Pierre Gorissen for use with
* U8X8 OLED + SX1276
* Modified 01-2-2018 by Nicolas Derive to use measurements from
* BME280 sensor and display them on the screen
* *************************************************************/
#include <SPI.h>
#define BUILTIN_LED 25
// define the activation method ABP or OTAA
#define ACT_METHOD_ABP
// show debug statements; comment next line to disable debug statements
#define DEBUG
#define CFG_eu868
/* **************************************************************
* keys for device (removed for forum posting)
* *************************************************************/
static const uint8_t PROGMEM NWKSKEY[16] = { };
static const uint8_t PROGMEM APPSKEY[16] = { };
static const uint32_t DEVADDR = ;
/* **************************************************************
* user settings
* *************************************************************/
// Schedule TX every this many seconds (might become longer due to duty
// cycle limitations).
// const unsigned TX_INTERVAL = 240; // unused
unsigned int countero = 0;
unsigned int serial_act = 0;
unsigned long starttime;
unsigned long cycle_length = TX_INTERVAL * 1000UL; // cycle in secs, currently unused;
// Uses LMIC libary by Thomas Telkamp and Matthijs Kooijman (https://github.com/matthijskooijman/arduino-lmic)
// Pin mappings based upon PCB Doug Larue
#include <lmic.h>
#include <hal/hal.h>
// Declare the job control structures
static osjob_t sendjob;
// These callbacks are only used in over-the-air activation, so they are
// left empty when ABP (we cannot leave them out completely unless
// DISABLE_JOIN is set in config.h, otherwise the linker will complain).
#ifdef ACT_METHOD_ABP
void os_getArtEui (u1_t* buf) { }
void os_getDevEui (u1_t* buf) { }
void os_getDevKey (u1_t* buf) { }
#else
void os_getArtEui (u1_t* buf) { memcpy_P(buf, APPEUI, 8);}
void os_getDevEui (u1_t* buf) { memcpy_P(buf, DEVEUI, 8);}
void os_getDevKey (u1_t* buf) { memcpy_P(buf, APPKEY, 16);}
#endif
/* **************************************************************
* Pin mapping
* *************************************************************/
const lmic_pinmap lmic_pins = {
.nss = 18,
.rxtx = LMIC_UNUSED_PIN,
.rst = 14,
.dio = {26, 33, 32},
};
/* **************************************************************
* OLED setup
* *************************************************************/
#include "SSD1306.h"
#include "images.h"
// the OLED used
SSD1306 display(0x3c, 21, 22);
void logo(){
display.clear();
display.drawXbm(0,5,logo_width,logo_height,logo_bits);
display.display();
}
/* **************************************************************
* Sensor setup
* *************************************************************/
#include <Wire.h>
//#include "BlueDot_BME280.h"
#include <Seeed_BME280.h>
//BlueDot_BME280 bme280 = BlueDot_BME280();
BME280 bme;
// sensor variables
float tempC = 0.0;
float pressure = 0.0;
float humidity = 0.0;
float altitudeMeter = 0.0;
// meteofrance.fr : "En moyenne, la pression atmosphérique diminue de 1 hPa tous les 8 mètres"
#define ALT 44
#define COR (ALT/8.0)
unsigned int counter = 0;
// data to send
static uint8_t dataTX[8];
/* **************************************************************
* setup
* *************************************************************/
void setup() {
//Set baud rate
Serial.begin(115200);
// Wait (max 10 seconds) for the Serial Monitor
while ((!Serial) && (millis() < 10000)){ }
init_node();
init_sensor();
pinMode(16,OUTPUT);
pinMode(2,OUTPUT);
if (Serial) {
serial_act = 1;
digitalWrite(16, LOW); // set GPIO16 low to reset OLED
delay(50);
digitalWrite(16, HIGH); // while OLED is running, must set GPIO16 in high
display.init();
display.flipScreenVertically();
display.setFont(ArialMT_Plain_10);
logo();
delay(1500);
}
else {
digitalWrite(16, LOW);
}
}
starttime = millis();
}
/* **************************************************************
* loop
* *************************************************************/
void loop() {
do_sense();
// check if need to send
build_data();
do_send();
// delay(240000);
// if ((millis() - starttime) > cycle_length) { build_data(); do_send(); starttime = millis(); } // don't know how to rewrite this with ESP deepsleep
Serial.println("Bring ESP to sleep mode 5minutes");
/* 5e6 is 5x10^6 microseconds */
ESP.deepSleep(240e6);
}
/* **************************************************************
* sensor code, typical would be init_sensor(), do_sense(), build_data()
* *************************************************************/
/* **************************************************************
* init the sensor
* *************************************************************/
void init_sensor() {
while (!bme.init()){
Serial.print("BME280 error!");
delay(2000);
}
}
/* **************************************************************
* do the reading
* *************************************************************/
void do_sense() {
// Read temperature as Celsius (the default)
tempC = bme.getTemperature();
// Read humidity
humidity = bme.getHumidity();
// Read the pressure in hPa
pressure = bme.getPressure();
// Read the altitude in Meters
altitudeMeter = bme.calcAltitude(pressure);
pressure = pressure/100;
// Check if any reads failed and exit early (to try again).
if (isnan(pressure) || isnan(tempC) || isnan(humidity) || isnan(altitudeMeter)) {
#ifdef DEBUG
Serial.println("Failed to read from BME280 sensor!");
#endif
return;
}
if (serial_act == 1) {
display.clear();
display.setTextAlignment(TEXT_ALIGN_LEFT);
display.setFont(ArialMT_Plain_10);
//u8x8.setCursor(0, 3);
//u8x8.printf("Temp: %.1fC", tempC);
String tempS = "Température : " + String(tempC) + "°C";
display.drawString(0, 1, tempS);
String humidityS = "Humidité : " + String(humidity) + "%";
display.drawString(0, 12, humidityS);
//u8x8.setCursor(0, 5);
// u8x8.printf("Press: %.1fhPa", pressure);
String presS = "Pression : " + String(pressure) + "hPa";
display.drawString(0, 23, presS);
String alt = "Altitude : " + String(altitudeMeter) + "m";
display.drawString(0, 34, alt);
String counters = "Paquets envoyés : " + String(countero);
display.drawString(0, 45, counters);
display.display();
}
#ifdef DEBUG
Serial.print(F(" temp Celcius:"));
Serial.print(tempC);
Serial.print(F(" humidity:"));
Serial.print(humidity);
Serial.print(F(" pressure:"));
Serial.print(pressure);
Serial.print(F(" altitude (meters):"));
Serial.print(altitudeMeter);
Serial.println(F(""));
#endif
delay(2000);
}
/* **************************************************************
* build data to transmit in dataTX
*
* Suggested payload function for this data
*
* function Decoder(bytes, port) {
* var temp = parseInt(bytes[0] + (bytes[1] << 8 ) - 500) / 10 ;
* var humidity = parseInt(bytes[2] + (bytes[3] << 8 ) ;
* var pressure = parseInt(bytes[4] + (bytes[5] << 8 )) / 10;
* // don't know why, but in my case the altitude was about 90 meter to low, so added 90 meters
* var altitude = parseInt(bytes[6] + (bytes[7] << 8 ) - 100) / 10;
* return { temp: temp,
* humidity: humidity,
* pressure: pressure,
* altitude: altitude
* };
* }
* *************************************************************/
void build_data() {
int dtempC = (tempC + 50) * 10;
int dpressure = pressure * 10;
int daltitude = (altitudeMeter + 100) * 10;
dataTX[0] = dtempC;
dataTX[1] = dtempC >> 8;
dataTX[2] = int(humidity);
dataTX[3] = int(humidity) >> 8;
dataTX[4] = dpressure;
dataTX[5] = dpressure >> 8;
dataTX[6] = daltitude;
dataTX[7] = daltitude >> 8;
}
/* **************************************************************
* radio code, typical would be init_node(), do_send(), etc
* *************************************************************/
/* **************************************************************
* init the Node
* *************************************************************/
void init_node() {
#ifdef VCC_ENABLE
// For Pinoccio Scout boards
pinMode(VCC_ENABLE, OUTPUT);
digitalWrite(VCC_ENABLE, HIGH);
delay(1000);
#endif
// LMIC init
os_init();
// Reset the MAC state. Session and pending data transfers will be discarded.
LMIC_reset();
#ifdef ACT_METHOD_ABP
// Set static session parameters. Instead of dynamically establishing a session
// by joining the network, precomputed session parameters are be provided.
#ifdef PROGMEM
// On AVR, these values are stored in flash and only copied to RAM
// once. Copy them to a temporary buffer here, LMIC_setSession will
// copy them into a buffer of its own again.
uint8_t appskey[sizeof(APPSKEY)];
uint8_t nwkskey[sizeof(NWKSKEY)];
memcpy_P(appskey, APPSKEY, sizeof(APPSKEY));
memcpy_P(nwkskey, NWKSKEY, sizeof(NWKSKEY));
LMIC_setSession (0x1, DEVADDR, nwkskey, appskey);
#else
// If not running an AVR with PROGMEM, just use the arrays directly
LMIC_setSession (0x1, DEVADDR, NWKSKEY, APPSKEY);
#endif
#if defined(CFG_eu868)
// Set up the channels used by the Things Network, which corresponds
// to the defaults of most gateways. Without this, only three base
// channels from the LoRaWAN specification are used, which certainly
// works, so it is good for debugging, but can overload those
// frequencies, so be sure to configure the full frequency range of
// your network here (unless your network autoconfigures them).
// Setting up channels should happen after LMIC_setSession, as that
// configures the minimal channel set.
// NA-US channels 0-71 are configured automatically
LMIC_setupChannel(0, 868100000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(1, 868300000, DR_RANGE_MAP(DR_SF12, DR_SF7B), BAND_CENTI); // g-band
LMIC_setupChannel(2, 868500000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(3, 867100000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(4, 867300000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(5, 867500000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(6, 867700000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(7, 867900000, DR_RANGE_MAP(DR_SF12, DR_SF7), BAND_CENTI); // g-band
LMIC_setupChannel(8, 868800000, DR_RANGE_MAP(DR_FSK, DR_FSK), BAND_MILLI); // g2-band
// TTN defines an additional channel at 869.525Mhz using SF9 for class B
// devices' ping slots. LMIC does not have an easy way to define set this
// frequency and support for class B is spotty and untested, so this
// frequency is not configured here.
#elif defined(CFG_us915)
// NA-US channels 0-71 are configured automatically
// but only one group of 8 should (a subband) should be active
// TTN recommends the second sub band, 1 in a zero based count.
// https://github.com/TheThingsNetwork/gateway-conf/blob/master/US-global_conf.json
LMIC_selectSubBand(1);
#endif
#if defined(DEBUG)
LMIC_disableChannel(1);
LMIC_disableChannel(2);
LMIC_disableChannel(3);
LMIC_disableChannel(4);
LMIC_disableChannel(5);
LMIC_disableChannel(6);
LMIC_disableChannel(7);
LMIC_disableChannel(8);
#endif
// Enable data rate adaptation
// LMIC_setAdrMode(1);
// Disable link check validation
LMIC_setLinkCheckMode(0);
// TTN uses SF9 for its RX2 window.
LMIC.dn2Dr = DR_SF9;
// Set data rate and transmit power (note: txpow seems to be ignored by the library)
LMIC_setDrTxpow(DR_SF7,14);
#endif
#ifdef ACT_METHOD_OTAA
// got this fix from forum: https://www.thethingsnetwork.org/forum/t/over-the-air-activation-otaa-with-lmic/1921/36
LMIC_setClockError(MAX_CLOCK_ERROR * 1 / 100);
#endif
}
/* **************************************************************
* send the message
* *************************************************************/
void do_send() {
Serial.print(millis());
Serial.print(F(" Sending.. "));
send_message(&sendjob);
// wait for send to complete
Serial.print(millis());
Serial.print(F(" Waiting.. "));
while ( (LMIC.opmode & OP_JOINING) or (LMIC.opmode & OP_TXRXPEND) ) { os_runloop_once(); }
Serial.print(millis());
Serial.println(F(" TX_COMPLETE"));
countero++;
}
/* *****************************************************************************
* send_message
* ****************************************************************************/
void send_message(osjob_t* j) {
// Check if there is not a current TX/RX job running
if (LMIC.opmode & OP_TXRXPEND) {
Serial.println(F("OP_TXRXPEND, not sending"));
} else {
// Prepare upstream data transmission at the next possible time.
LMIC_setTxData2(1, dataTX, sizeof(dataTX), 0);
Serial.println(F("Packet queued"));
}
}
/*******************************************************************************/
void onEvent (ev_t ev) {
switch (ev) {
case EV_SCAN_TIMEOUT:
Serial.println(F("EV_SCAN_TIMEOUT"));
break;
case EV_BEACON_FOUND:
Serial.println(F("EV_BEACON_FOUND"));
break;
case EV_BEACON_MISSED:
Serial.println(F("EV_BEACON_MISSED"));
break;
case EV_BEACON_TRACKED:
Serial.println(F("EV_BEACON_TRACKED"));
break;
case EV_JOINING:
Serial.println(F("EV_JOINING"));
break;
case EV_JOINED:
Serial.println(F("EV_JOINED"));
// Disable link check validation (automatically enabled
// during join, but not supported by TTN at this time).
LMIC_setLinkCheckMode(0);
break;
case EV_RFU1:
Serial.println(F("EV_RFU1"));
break;
case EV_JOIN_FAILED:
Serial.println(F("EV_JOIN_FAILED"));
break;
case EV_REJOIN_FAILED:
Serial.println(F("EV_REJOIN_FAILED"));
break;
case EV_TXCOMPLETE:
Serial.println(F("EV_TXCOMPLETE (includes waiting for RX windows)"));
if (LMIC.dataLen) {
// data received in rx slot after tx
Serial.print(F("Data Received: "));
Serial.write(LMIC.frame+LMIC.dataBeg, LMIC.dataLen);
Serial.println();
}
// schedule next transmission
// os_setTimedCallback(&sendjob, os_getTime() + sec2osticks(TX_INTERVAL), send_message);
break;
case EV_LOST_TSYNC:
Serial.println(F("EV_LOST_TSYNC"));
break;
case EV_RESET:
Serial.println(F("EV_RESET"));
break;
case EV_RXCOMPLETE:
// data received in ping slot
Serial.println(F("EV_RXCOMPLETE"));
break;
case EV_LINK_DEAD:
Serial.println(F("EV_LINK_DEAD"));
break;
case EV_LINK_ALIVE:
Serial.println(F("EV_LINK_ALIVE"));
break;
default:
Serial.println(F("Unknown event"));
break;
}
}
Any idea how to fix this? Now that I’m using ABP, that’s a pain but not critical, but when I will switch to OTAA, it will be absolutely not doable to re-register each time…
Thanks for your help.