BeieliScale/Arduino/beescale_lora_27sep2018/beescale_lora_27sep2018.ino

675 lines
18 KiB
C++

#include <lmic.h>
#include <hal/hal.h>
#include <SPI.h>
#include "HX711.h"
#include "Adafruit_Si7021.h"
#include "Adafruit_FRAM_SPI.h"
// Defines
#define TIMEOUTMS 60000
#define SENDDIFFTHRESHOLD 999
#define DONEPIN A5
#define VBATPIN A7
#define MAX_VALUES_TO_SEND 5
#define MAX_SHORT 32767
#define JOIN_AFTER_N_STARTUPS 10
// Enumerations
enum which {
NWKSKEY,
APPSKEY,
DEVADDR,
APPEUI,
DEVEUI,
APPKEY
};
// Constants
const uint8_t FRAM_CS = 10;
#if defined(ARDUINO_SAMD_ZERO) && defined(SERIAL_PORT_USBVIRTUAL)
// Required for Serial on Zero based boards
#define Serial SERIAL_PORT_USBVIRTUAL
#endif
// Data Types
typedef struct {
// for ABP
u1_t nwkskey[16];
u1_t appskey[16];
u4_t devaddr;
// for OTAA
u1_t appeui[8]; // Swisscom: F0:3D:29:AC:71:00:00:01
u1_t deveui[8];
u1_t appkey[16];
// session information
u4_t framecounterup;
u4_t framecounterdown;
long cal_w1_0;
long cal_w2_0;
float cal_w1_factor;
float cal_w2_factor;
short weight[MAX_VALUES_TO_SEND];
short temperature[MAX_VALUES_TO_SEND];
byte my_position;
byte logging; // 0: no, otherwise: yes
byte startup_counter;
} FRAM_data;
typedef struct {
byte version; // Versionierung des Paketformats
short weight[MAX_VALUES_TO_SEND]; // Gewicht in 10-Gramm
short temperature[MAX_VALUES_TO_SEND]; // Temperatur in 1/10 Grad Celsius
byte vbat; // Batteriespannung (1 Einheit => 20 mV)
} LORA_data;
// Global Variables
bool lora_data_sent = false;
bool usb_cable_attached = false;
FRAM_data fram_data;
LORA_data lora_data;
Adafruit_FRAM_SPI fram = Adafruit_FRAM_SPI(FRAM_CS); // use hardware SPI
//static uint8_t mydata[] = "Hello, world!";
static osjob_t sendjob;
char mystring[100];
// Temp./Humidity Sensor
Adafruit_Si7021 temp_sensor;
// Scales
HX711 scale1;
HX711 scale2;
// These callbacks are only used in over-the-air activation, so they are
// left empty here (we cannot leave them out completely unless
// DISABLE_JOIN is set in config.h, otherwise the linker will complain).
void os_getArtEui (u1_t* buf) { memcpy_P(buf, fram_data.appeui, 8); }
void os_getDevEui (u1_t* buf) { memcpy_P(buf, fram_data.deveui, 8); }
void os_getDevKey (u1_t* buf) { memcpy_P(buf, fram_data.appkey, 16); }
// Pin mapping
const lmic_pinmap lmic_pins = {
.nss = 8,
.rxtx = LMIC_UNUSED_PIN,
.rst = 4,
.dio = {3, 6, LMIC_UNUSED_PIN},
};
void logit(char* logstring) {
if (fram_data.logging) {
Serial.print(millis());
Serial.print(": ");
Serial.println(logstring);
}
}
void print_byte_array(u1_t arr[], int n) {
int i;
for (i = 0; i < n; i++)
{
if (arr[i] < 16) Serial.write('0');
Serial.print(arr[i], HEX);
}
}
void ShowFRAMData() {
Serial.println('{');
Serial.print(" \"nwkskey\": \"");
print_byte_array(fram_data.nwkskey, 16);
Serial.println("\", ");
Serial.print(" \"appskey\": \"");
print_byte_array(fram_data.appskey, 16);
Serial.println("\", ");
Serial.print(" \"devaddr\": \"");
Serial.print(fram_data.devaddr, HEX);
Serial.println("\",");
Serial.print(" \"appeui\": \"");
print_byte_array(fram_data.appeui, 8);
Serial.println("\", ");
Serial.print(" \"deveui\": \"");
print_byte_array(fram_data.deveui, 8);
Serial.println("\", ");
Serial.print(" \"appkey\": \"");
print_byte_array(fram_data.appkey, 16);
Serial.println("\", ");
Serial.print(" \"framecounterup\": ");
Serial.print(fram_data.framecounterup);
Serial.println(",");
Serial.print(" \"framecounterdown\": ");
Serial.print(fram_data.framecounterdown);
Serial.println(",");
Serial.print(" \"cal_w1_0\": ");
Serial.print(fram_data.cal_w1_0);
Serial.println(",");
Serial.print(" \"cal_w2_0\": ");
Serial.print(fram_data.cal_w2_0);
Serial.println(",");
Serial.print(" \"cal_w1_factor\": ");
Serial.print(fram_data.cal_w1_factor);
Serial.println(",");
Serial.print(" \"cal_w2_factor\": ");
Serial.print(fram_data.cal_w2_factor);
Serial.println(",");
Serial.print(" \"weight\": [");
for (int i = 0; i < MAX_VALUES_TO_SEND; i++) {
Serial.print(fram_data.weight[i]);
if (i < (MAX_VALUES_TO_SEND - 1)) {
Serial.print(",");
}
}
Serial.println("],");
Serial.print(" \"temperature\": [");
for (int i = 0; i < MAX_VALUES_TO_SEND; i++) {
Serial.print(fram_data.temperature[i]);
if (i < (MAX_VALUES_TO_SEND - 1)) {
Serial.print(",");
}
}
Serial.println("],");
Serial.print(" \"my_position\": ");
Serial.print(fram_data.my_position);
Serial.println(",");
Serial.print(" \"startup_counter\": ");
Serial.println(fram_data.startup_counter);
Serial.println("}");
}
void onEvent (ev_t ev) {
logit("Received LoraWAN Event");
switch (ev) {
case EV_SCAN_TIMEOUT:
logit("EV_SCAN_TIMEOUT");
break;
case EV_BEACON_FOUND:
logit("EV_BEACON_FOUND");
break;
case EV_BEACON_MISSED:
logit("EV_BEACON_MISSED");
break;
case EV_BEACON_TRACKED:
logit("EV_BEACON_TRACKED");
break;
case EV_JOINING:
logit("EV_JOINING");
break;
case EV_JOINED:
logit("EV_JOINED");
memcpy(fram_data.nwkskey,LMIC.nwkKey,16);
memcpy(fram_data.appskey,LMIC.artKey,16);
fram_data.devaddr = LMIC.devaddr;
break;
case EV_RFU1:
logit("EV_RFU1");
break;
case EV_JOIN_FAILED:
logit("EV_JOIN_FAILED");
break;
case EV_REJOIN_FAILED:
logit("EV_REJOIN_FAILED");
break;
case EV_TXCOMPLETE:
logit("EV_TXCOMPLETE (includes waiting for RX windows)");
if (LMIC.txrxFlags & TXRX_ACK)
logit("Received ack");
lora_data_sent = true;
if (LMIC.dataLen) {
sprintf(mystring,"Received %d bytes of payload", LMIC.dataLen);
}
break;
case EV_LOST_TSYNC:
logit("EV_LOST_TSYNC");
break;
case EV_RESET:
logit("EV_RESET");
break;
case EV_RXCOMPLETE:
// data received in ping slot
logit("EV_RXCOMPLETE");
break;
case EV_LINK_DEAD:
logit("EV_LINK_DEAD");
break;
case EV_LINK_ALIVE:
logit("EV_LINK_ALIVE");
break;
default:
logit("Unknown event");
break;
}
}
void InitFeather() {
// DONEPIN must be low...
pinMode(DONEPIN, OUTPUT);
digitalWrite(DONEPIN, LOW);
}
bool isUSBCableAttached() {
// Reading battery; Value is in Millivolts
int vbat = GetBat();
// Wir nehmen an, dass das USB-Kabel gesteckt ist, wenn die Batteriespannung
// groesser als 4.4V ist (Schalter muss auf 0 sein)
usb_cable_attached = (vbat >= 4400);
// fuer Debugzwecke
pinMode(LED_BUILTIN, OUTPUT);
if (usb_cable_attached) {
digitalWrite(LED_BUILTIN, HIGH);
} else {
digitalWrite(LED_BUILTIN, LOW);
}
return usb_cable_attached;
}
void InitSerial() {
Serial.begin(115200);
Serial.setTimeout(6000);
//while (!Serial);
delay(1000);
logit("Serial Port initialized");
}
bool ShouldDataBeSent() {
// Data should be sent if difference is too big or if count is reached
bool res = false;
if (fram_data.my_position >= MAX_VALUES_TO_SEND) {
res = true;
}
if (fram_data.my_position > 1) {
if ((fram_data.weight[(fram_data.my_position - 1)] - fram_data.weight[(fram_data.my_position - 2)]) >= SENDDIFFTHRESHOLD) {
res = true;
}
}
return res;
}
void InitLORA() {
// LMIC init
os_init();
// Reset the MAC state. Session and pending data transfers will be discarded.
LMIC_reset();
// see https://startiot.telenor.com/learning/getting-started-with-adafruit-feather-m0-lora/
LMIC_setClockError(MAX_CLOCK_ERROR * 1 / 100);
// We normally use ABP, but make a join for the first startup and after JOIN_AFTER_N_STARTUPS startups
if (fram_data.startup_counter == 0) {
// do nothing
logit("we do nothing => OTAA");
}
else if (fram_data.startup_counter >= JOIN_AFTER_N_STARTUPS) {
fram_data.startup_counter = 0;
} else {
LMIC_setSession (0x1, fram_data.devaddr, fram_data.nwkskey, fram_data.appskey);
// We set the frame counters
LMIC.seqnoUp = fram_data.framecounterup;
LMIC.seqnoDn = fram_data.framecounterdown;
}
// Disable link check validation
//LMIC_setLinkCheckMode(0);
// TTN uses SF9 for its RX2 window.
//LMIC.dn2Dr = DR_SF9;
// Swisscom uses SF12 for its RX2 window.
//LMIC.dn2Dr = DR_SF12;
// Set data rate and transmit power for uplink (note: txpow seems to be ignored by the library)
//LMIC_setDrTxpow(DR_SF7, 14);
//LMIC_setDrTxpow(DR_SF12, 14);
}
void do_send(osjob_t* j) {
// Check if there is not a current TX/RX job running
if (LMIC.opmode & OP_TXRXPEND) {
logit("OP_TXRXPEND, not sending");
} else {
// Prepare upstream data transmission at the next possible time.
//LMIC_setTxData2(1, mydata, sizeof(mydata)-1, 0);
LMIC_setTxData2(1, (uint8_t*) &lora_data, sizeof(LORA_data), 0);
logit("Packet queued");
}
}
void InitAndReaFRAM() {
// we set #8 to High (CS of Lora Module), to be able to use FRAM
pinMode(8, INPUT_PULLUP);
digitalWrite(8, HIGH);
if (fram.begin()) {
logit("Found SPI FRAM");
} else {
logit("No SPI FRAM found ... check your connections\r\n");
}
// wir lesen die FRAM-Werte
logit("Reading FRAM...");
fram.read(0x0, (uint8_t*) &fram_data, sizeof(FRAM_data));
fram_data.logging = 1;
logit("FRAM was read...");
}
float GetWeight() {
long raw_weight1 = scale1.read_average(3);
long raw_weight2 = scale2.read_average(3);
return (((raw_weight1 - fram_data.cal_w1_0) / fram_data.cal_w1_factor) + ((raw_weight2 - fram_data.cal_w2_0) / fram_data.cal_w2_factor));
}
float GetTemp() {
return temp_sensor.readTemperature();
}
int GetBat() {
float vb = analogRead(VBATPIN);
vb *= 2; // we divided by 2, so multiply back
vb *= 3.3; // Multiply by 3.3V, our reference voltage
return vb;
}
void ReadSensors() {
temp_sensor = Adafruit_Si7021();
// we initialize the scales
logit("Initializing the scale");
scale1.begin(A3, A2);
scale2.begin(A1, A0);
// Now we initialize the Si7021
if (!temp_sensor.begin()) {
logit("Did not find Si7021 sensor!");
}
lora_data.vbat = (byte)(GetBat() / 20);
fram_data.weight[fram_data.my_position] = (int)GetWeight() / 10;
fram_data.temperature[fram_data.my_position] = (int)GetTemp() * 10;
fram_data.my_position++;
}
void SendLoraData() {
logit("Now sending packet...");
lora_data.version = 1;
for (int i = 0; i < MAX_VALUES_TO_SEND; i++) {
lora_data.weight[i] = fram_data.weight[i];
lora_data.temperature[i] = fram_data.temperature[i];
}
// Start job
do_send(&sendjob);
unsigned long starttime;
starttime = millis();
bool logged = false;
while (!lora_data_sent && ((millis() - starttime) < TIMEOUTMS)) {
os_runloop_once();
if ((millis() % 1000) == 0) {
if (!logged) {
logit("Loop until sent...");
logged = true;
}
} else {
logged = false;
}
}
if (lora_data_sent) {
logit("Lora Data was sent!");
} else {
logit("Timeout elapsed...");
}
// We clear the data...
for (int i = 0; i < MAX_VALUES_TO_SEND; i++) {
fram_data.weight[i] = MAX_SHORT;
fram_data.temperature[i] = MAX_SHORT;
}
fram_data.my_position = 0;
// We set the frame counters
fram_data.framecounterup = LMIC.seqnoUp;
fram_data.framecounterdown = LMIC.seqnoDn;
}
void SaveFRAM() {
logit("we save the fram_data");
fram.writeEnable(true);
fram.write(0x0, (uint8_t *) &fram_data, sizeof(FRAM_data) );
fram.writeEnable(false);
logit("fram_data saved...");
}
boolean isValidHexKey(String hk, int length) {
if (hk.length() != length) {
return false;
}
char mychar;
for (int i = 0; i < hk.length(); i++) {
mychar = hk.charAt(i);
if (not(isHexadecimalDigit(mychar))) {
return false;
}
}
return true;
}
byte dehex(char c) { // Get nibble value 0...15 from character c
// Treat digit if c<'A', else letter
return c < 'A' ? c & 0xF : 9 + (c & 0xF);
// Above assumes that c is a 'hex digit' in 0...9, A or a ... F or f.
// It would make more sense to just use 16 consecutive characters,
// like eg 0123456789:;<=>? or @ABCDEFGHIJKLMNO so the above
// could just say `return c & 0xF;`
}
void WriteKey (String input, which w) {
const char *hin = input.c_str(); // Get character array
int clen = input.length() / 2;
// Next line invalid in C++, ok in C99. Probably need to
// instead declare a fixed-length array, cmd[MAXCMDLEN], etc
char cmd[clen + 1]; // Leave a byte for null terminator
for (int i = 0; i < 2 * clen; i += 2) {
cmd[i / 2] = dehex(hin[i]) << 4 | dehex(hin[i + 1]);
}
cmd[clen] = 0; // Null-byte terminator
if (w == APPSKEY) {
memcpy(&fram_data.appskey, &cmd, 16);
} else if (w == NWKSKEY) {
memcpy(&fram_data.nwkskey, &cmd, 16);
} else if (w == DEVADDR) {
logit("write device address");
fram_data.devaddr = strtol(hin, 0, 16);
} else if (w == APPEUI) {
memcpy(&fram_data.appeui, &cmd, 8);
} else if (w == DEVEUI) {
memcpy(&fram_data.deveui, &cmd, 8);
} else if (w == APPKEY) {
memcpy(&fram_data.appkey, &cmd, 16);
} else {
logit("Invalid which");
}
SaveFRAM();
}
void Setup() {
String s = "";
while (s != "exit") {
s = Serial.readStringUntil('\n');
s.trim();
if (s == "") {
//Serial.println("Leerzeile wird ignoriert...");
}
else if (s == "setup") {
Serial.println("{ \"msg\": \"Entering setup mode\" }");
}
else if (s.startsWith("setnwkskey")) {
String key;
key = s.substring(11);
if (isValidHexKey(key, 32)) {
WriteKey(key, NWKSKEY);
Serial.println("{ \"msg\": \"setnwkskey was successful\" }");
}
else {
Serial.println("{ \"msg\": \"Ist kein gueltiger Hex Key mit 32 Zeichen\" }");
}
}
else if (s.startsWith("setappskey")) {
String key;
key = s.substring(11);
if (isValidHexKey(key, 32)) {
WriteKey(key, APPSKEY);
Serial.println("{ \"msg\": \"setappskey was successful\" }");
}
else {
Serial.println("{ \"msg\": \"Ist kein gueltiger Hex Key mit 32 Zeichen\" }");
}
}
else if (s.startsWith("setdevaddr")) {
String key;
key = s.substring(11);
if (isValidHexKey(key, 8)) {
WriteKey(key, DEVADDR);
Serial.println("{ \"msg\": \"setdevaddr was successful\" }");
}
else {
Serial.println("{ \"msg\": \"Ist kein gueltiger Hex Key mit 32 Zeichen\" }");
}
}
else if (s.startsWith("setappeui")) {
String key;
key = s.substring(10);
if (isValidHexKey(key, 16)) {
WriteKey(key, APPEUI);
Serial.println("{ \"msg\": \"setappeui was successful\" }");
}
else {
Serial.println("{ \"msg\": \"Ist kein gueltiger Hex Key mit 16 Zeichen\" }");
}
}
else if (s.startsWith("setdeveui")) {
String key;
key = s.substring(10);
if (isValidHexKey(key, 16)) {
WriteKey(key, DEVEUI);
Serial.println("{ \"msg\": \"setdeveui was successful\" }");
}
else {
Serial.println("{ \"msg\": \"Ist kein gueltiger Hex Key mit 16 Zeichen\" }");
}
}
else if (s.startsWith("setappkey")) {
String key;
key = s.substring(10);
if (isValidHexKey(key, 32)) {
WriteKey(key, APPKEY);
Serial.println("{ \"msg\": \"setappkey was successful\" }");
}
else {
Serial.println("{ \"msg\": \"Ist kein gueltiger Hex Key mit 32 Zeichen\" }");
}
}
else if (s == "getvalues") {
Serial.println('{');
Serial.print(" \"weight\": ");
Serial.print(GetWeight());
Serial.println(", ");
Serial.print(" \"temperature\": ");
Serial.print(GetTemp());
Serial.println(", ");
Serial.print(" \"batt\": ");
Serial.println(GetBat());
Serial.println("}");
}
else if (s == "getrawvalues") {
long raw_weight1 = scale1.read_average(3);
long raw_weight2 = scale2.read_average(3);
Serial.println('{');
Serial.print(" \"w1_raw\": ");
Serial.print(raw_weight1);
Serial.println(", ");
Serial.print(" \"w2_raw\": ");
Serial.print(raw_weight2);
Serial.println("");
Serial.println("}");
}
else if (s == "getframdata") {
ShowFRAMData();
}
else if (s == "calibrate_zero_1") {
long raw_weight1 = scale1.read_average(3);
fram_data.cal_w1_0 = (int)raw_weight1;
Serial.println("{ \"msg\": \"calibrate_zero_1 was successful\" }");
}
else if (s == "calibrate_zero_2") {
long raw_weight2 = scale2.read_average(3);
fram_data.cal_w2_0 = (int)raw_weight2;
Serial.println("{ \"msg\": \"calibrate_zero_2 was successful\" }");
}
else if (s.startsWith("calibrate_1")) {
String w1_gramm;
w1_gramm = s.substring(12);
long raw_weight1 = scale1.read_average(3);
//fram_data.cal_w1_factor = (w1_gramm.toFloat() / (float)raw_weight1);
fram_data.cal_w1_factor = (((float)raw_weight1 - fram_data.cal_w1_0)/ w1_gramm.toFloat());
Serial.println("{ \"msg\": \"calibrate_1 was successful\" }");
}
else if (s.startsWith("calibrate_2")) {
String w2_gramm;
w2_gramm = s.substring(12);
long raw_weight2 = scale2.read_average(3);
fram_data.cal_w2_factor = (((float)raw_weight2 - fram_data.cal_w2_0)/ w2_gramm.toFloat());
Serial.println("{ \"msg\": \"calibrate_2 was successful\" }");
}
else if (s == "exit") {
// we exit the loop
}
else {
Serial.println("{ \"msg\": \"You sent me an unknown command (valid commands: exit,calibrate_zero_1,calibrate_zero_2,calibrate_1,calibrate_2,getvalues,getrawvalues,setnwkskey,setappskey,setdevaddr,setappeui,setdeveui,setappkey,getframdata)\" }");
}
}
Serial.println("We exited the setup loop...");
}
void TurnOff() {
logit("We turn off...");
logit("Jetzt senden wir das DONE Signal...");
delay(500);
while (1) {
digitalWrite(DONEPIN, HIGH);
delay(5);
digitalWrite(DONEPIN, LOW);
delay(5);
}
}
void setup() {
fram_data.logging = 1;
//fram_data.startup_counter = 0;
InitFeather();
if (isUSBCableAttached()) {
InitSerial();
}
InitAndReaFRAM();
//fram_data.startup_counter = 0;
ReadSensors();
if (isUSBCableAttached() && Serial.available()) {
Setup();
}
if (ShouldDataBeSent()) {
InitLORA();
SendLoraData();
}
fram_data.startup_counter++;
SaveFRAM();
TurnOff();
}
void loop() {
}