Sketch with - Sensors: SDP3x, AHT45 (SHT40/DPS310 commented out) | 100mA Battery Charging | SOC | Charging Notification | BLE Connect Beeps
/*********************************************************************
Sensors in use: SDP3x | AHT45
Sensors Commented out but ready: SHT45 | DPS310
07 June added Charging notification (requires TotalVario Beta 9-2)
06 June added BLE connection beeps
04 June 2023 - used Bernd's sprintf sketch
22 April 2023 - Working with XIAO nRF (ie: not XIAO mBed device).
INFO:
MCU Hardware - Seeed XIAO nRF52840 (not "Sense" model)
Using Arduino IDE 1.8.19
Using Arduino 'Board' -> "Seeed nRF52 Boards" -> "Seeed XIAO nRF52840"
*********************************************************************/
// ----------LIBRARIES--------------
#include <bluefruit.h> // for BLE - Adafruit BLE Library
#include <Adafruit_LittleFS.h>
#include <InternalFileSystem.h>
#include <Wire.h> // for I2C comms
#include <avr/dtostrf.h> // for dtostrf
#include <sdpsensor.h> // Sensirion SDP3x library
#include <AHTxx.h> // Asair AHTx library
//#include <SensirionI2CSht4x.h> // Sensirion SHT4x library
//#include <Adafruit_DPS310.h> // Adafruit DPS310 library
// ---------CONSTANTS---------------
// Function Timing
const int IntervalSDP = 99; // number of mS before IAS function repeats and also sends all BLE data
//const int IntervalDPS = 113; // number of mS before DPS function repeats and collects Atmospheric Pressure
const int IntervalSHT = 4800; // number of mS before Temperature Humidity function repeats
const int IntervalBTV = 9963; // number of mS before Battery Voltage function repeats
const int IntervalLED = 3313; // number of mS before the Battery Charging function repeats (turns on Red LED)
const int LightLED = 500; // millisecs that the (red) LED, indicating charging, is on
// -----------CLASSES---------------
// BLE Services
BLEDfu bledfu; // OTA DFU service
BLEDis bledis; // device information
BLEUart bleuart; // uart over ble
// Dual Pressure Sensor
SDP3XSensor sdp;
// Temperature & Humidity Sensor
//SensirionI2CSht4x sht4x;
AHTxx aht20(AHTXX_ADDRESS_X38, AHT2x_SENSOR); //sensor address, sensor type
// Absolute Pressure Sensor
//Adafruit_DPS310 dps;
//------------VARIABLES-------------
// Startup Melody Notes & Variables
#define NOTE_C6 1047
#define NOTE_E6 1319
#define NOTE_G6 1568
#define REST 0
int tempo = 90;
int buzzer = 1;
int melody[] = {
NOTE_C6,
16,
NOTE_G6,
16,
NOTE_E6,
16,
NOTE_C6,
32,
NOTE_G6,
-16,
NOTE_E6,
8,
};
int notes = sizeof(melody) / sizeof(melody[0]) / 2;
int wholenote = (60000 * 4) / tempo;
int divider = 0, noteDuration = 0;
// Function Timing
unsigned long currentMillis = 0; // Stores the value of millis() in each iteration of loop()
unsigned long previousSDPMillis = 0;
long previousSHTMillis = 0;
long previousDPSMillis = 0;
long previousBTVMillis = 0;
long previousLEDMillis = 0;
bool flashLEDstate = 0; // Used to record whether the LED is on or off
// Sensirion Differential Pressure Variable
float sdpPres; // Variable for the differential pressure
// Infinion Atmospheric Pressure Variable
float absPres = 966.0; // Variable for the Atmospheric absolute pressure
// Ambient Environment Variables
float ambTemp = 0;
float ambHumi = 0;
float ahtValue; // To store T/Rh result temporarily
// Battery Variables
float batVolt = 4.00; // Battery Volts
float batPrct = 50.0; // Battery SOC %
bool battChrg = 0; // Battery Charging
//===================SETUP========================
void setup() {
//Serial.begin(115200); // For debugging
// Initialise I2C
Wire.begin();
// Setup Xiao Battery Monitoring
pinMode(VBAT_ENABLE, OUTPUT); // Set the battery Voltage divider ground leg as an Output
pinMode(23, INPUT); // Set monitoring leg as an Input
// Setup Xiao Battery Charging Rate
pinMode(22, OUTPUT); // Set the charger mA selection pin as an Output
digitalWrite(22, LOW); // Charge Current = 100mA (22 High = 50mA)
// Setup Buzzer Output
pinMode(D1, OUTPUT); // D1 output to Buzzer
// Initialise Sensirion SDP
int ret = sdp.init();
if (ret == 0) {
} else {
while (true) {
delay(500);
}
}
// Initialise Asair AHT2x Temp|Humidity Sensor
while (aht20.begin() != true) {
delay(200);
}
// // Initialise Sensirion SHT
// sht4x.begin(Wire);
// // Initialise Infinion DPS sensor
// if (!dps.begin_I2C()) { // Can pass in I2C address here
// while (1) yield();
// }
// dps.configurePressure(DPS310_16HZ, DPS310_32SAMPLES); // See DPS310 datasheet for combinations
// dps.configureTemperature(DPS310_2HZ, DPS310_2SAMPLES);
// Startup Melody
{
for (int thisNote = 0; thisNote < notes * 2; thisNote = thisNote + 2) {
divider = melody[thisNote + 1];
if (divider > 0) {
// regular note, just proceed
noteDuration = (wholenote) / divider;
} else if (divider < 0) {
// dotted notes are represented with negative durations!!
noteDuration = (wholenote) / abs(divider);
noteDuration *= 1.5; // increases the duration in half for dotted notes
}
tone(buzzer, melody[thisNote], noteDuration * 0.9);
delay(noteDuration);
noTone(buzzer);
}
delay(200);
}
// Battery SoC Beeps
batVolt = analogRead(PIN_VBAT) * (0.003515625F) * (1510.0 / 510.0);
int n;
{
if (batVolt > 3.9) {
// fully charged ... four beeps
for (n = 1; n <= 4; n++) {
tone(buzzer, 880, 200);
delay(400);
}
} else if (batVolt > 3.8) {
// well charged ... three beeps
for (n = 1; n <= 3; n++) {
tone(buzzer, 880, 200);
delay(400);
}
} else if (batVolt > 3.7) {
// some charge ... two beeps
for (n = 1; n <= 2; n++) {
tone(buzzer, 880, 200);
delay(400);
}
} else {
// discharged ... one beep
tone(buzzer, 440, 300);
delay(400);
}
}
noTone(buzzer);
delay(100);
//BLE
#if CFG_DEBUG
// Blocking wait for connection when debug mode is enabled via IDE
while (!Serial) yield();
#endif
// Config the peripheral connection with maximum bandwidth
// more SRAM required by SoftDevice
// Note: All config***() function must be called before begin()
Bluefruit.configPrphBandwidth(BANDWIDTH_MAX);
Bluefruit.begin();
Bluefruit.setTxPower(-12); // Set max power. Accepted values are: -40, -30, -20, -16, -12, -8, -4, 0, 4
//Bluefruit.setName(getMcuUniqueID()); // useful testing with multiple central connections or (eg): Bluefruit.setName("Name");
Bluefruit.setName("ProbeS4");
Bluefruit.Periph.setConnectCallback(connect_callback);
Bluefruit.Periph.setDisconnectCallback(disconnect_callback);
// To be consistent OTA DFU should be added first if it exists
bledfu.begin();
// Configure and Start Device Information Service
bledis.setManufacturer("Seeed");
bledis.setModel("XIAOnRF");
bledis.begin();
// Configure and Start BLE Uart Service
bleuart.begin();
// Set up and start advertising
startAdv();
}
void startAdv(void) {
// Advertising packet
Bluefruit.Advertising.addFlags(BLE_GAP_ADV_FLAGS_LE_ONLY_GENERAL_DISC_MODE);
Bluefruit.Advertising.addTxPower();
// Include bleuart 128-bit uuid
Bluefruit.Advertising.addService(bleuart);
// Secondary Scan Response packet (optional)
// Since there is no room for 'Name' in Advertising packet
Bluefruit.ScanResponse.addName();
Bluefruit.Advertising.restartOnDisconnect(true);
Bluefruit.Advertising.setInterval(32, 244); // in unit of 0.625 ms
Bluefruit.Advertising.setFastTimeout(30); // number of seconds in fast mode
Bluefruit.Advertising.start(0); // 0 = Don't stop advertising after n seconds
}
//===================LOOP========================
void loop() {
currentMillis = millis(); // Capture the latest value of millis()
readSDP(); // This function reads SDP, sends to variable then calls buildMSG for BLEprinting
////readDPS(); // This function reads DPS function reads Atmospheric Pressure and sends to variable
readSHT(); // This function reads ambient Temperature/Humidity and sends to variable
readBTV(); // This function reads Battery Voltage and sends to variable
flashLED(); // This function flashes the LED if USB is plugged (Battery charging)
}
//================BLE FUNCTIONS=====================
// Callback invoked when central connects
void connect_callback(uint16_t conn_handle) {
// Get the reference to current connection
BLEConnection* connection = Bluefruit.Connection(conn_handle);
char central_name[32] = { 0 };
connection->getPeerName(central_name, sizeof(central_name));
// Serial.print("Connected to "); // For debugging
// Serial.println(central_name); // For debugging
tone(buzzer, 440, 400);
delay(400);
tone(buzzer, 880, 200);
}
void disconnect_callback(uint16_t conn_handle, uint8_t reason) {
(void)conn_handle;
(void)reason;
// Serial.print("Disconnected, reason = 0x"); // For debugging
// Serial.println(reason, HEX); // For debugging
tone(buzzer, 440, 600);
delay(400);
tone(buzzer, 330, 600);
}
//================ SENSOR FUNCTIONS =====================
// readSPD+++++++++++
void readSDP() // read Differential Pressure from SDP snsor, append to a NEMA message, then print / write to serial / BLE
{
if (currentMillis >= previousSDPMillis + IntervalSDP) // run 'readIAS' function only once time is up
{
previousSDPMillis = currentMillis; // save the time when change was made
int ret = sdp.readSample();
if (ret == 0) {
sdpPres = sdp.getDifferentialPressure();
//Serial.println(sdpPres); // Print Differential Pressure for debugging
}
//sdpPres = sdpPres - 0.02; // Offset if needed
//sdpPres = sdpPres * (966 / absPres); // Correction of presssure differential wrt altitude/QNH
if (sdpPres < 0) { // If there is a negative pressure set Differential pressure to zero
sdpPres = 0;
}
buildMSG(); // Call function to build the TotalVario sentence, then transmit to BLE
}
}
//// readDPS+++++++++++
//void readDPS() // read Atmospheric Pressure from DPS sensor
//{
// if (currentMillis >= previousDPSMillis + IntervalDPS) // run 'readDPS' function only once time is up
// {
// previousDPSMillis = currentMillis; // save the time when change was made
// sensors_event_t temp_event, pressure_event;
// while (!dps.temperatureAvailable() || !dps.pressureAvailable()) {
// return; // wait until there's something to read
// }
// dps.getEvents(&temp_event, &pressure_event);
// absPres = pressure_event.pressure;
// }
//}
// readSHT+++++++++++
void readSHT() // read Temperature & Humidity
{
if (currentMillis >= previousSHTMillis + IntervalSHT) // run 'readTH' function only once time is up
{
previousSHTMillis = currentMillis; // save the time when change was made
// sht4x.measureHighPrecision(ambTemp, ambHumi);
// Serial.println(ambTemp); // For debugging
// Serial.println(ambHumi); // For debugging
ahtValue = aht20.readTemperature(); //read 6-bytes via I2C, takes 80 milliseconds
ambTemp = ahtValue;
//Serial.println(ambTemp); // For debugging
ahtValue = aht20.readHumidity(); //read another 6-bytes via I2C, takes 80 milliseconds
ambHumi = ahtValue;
//Serial.println(ambHumi); // For debugging
}
}
// readBTV+++++++++++
void readBTV() // read battery Volts
{
if (currentMillis >= previousBTVMillis + IntervalBTV) // run 'readBV' function only once time is up
{
previousBTVMillis = currentMillis; // save the time when change was made
batVolt = batVolt = analogRead(PIN_VBAT) * (0.003515625F) * (1510.0 / 510.0);
batPrct = -4370 + (2105 * batVolt) - (247.5144 * sq(batVolt)); // approximates battery SOC from Volts. Uses quadratic to approx.
if (batPrct < 0) // If battery Voltage is less than 0% set batPrct to 0%
{
batPrct = 0;
}
if (batPrct > 100) // If battery Voltage is more than 100% set batPrct to 100%
{
batPrct = 100;
}
//Serial.println(batPrct); // For debugging
}
}
// flashLED+++++++++++
void flashLED() // flash the LED if battery is charging
{
// If battery is charging go to next step flash LED
if (flashLEDstate == 0) // if the LED is off, wait for the interval to expire before turning it on
{
if (currentMillis - previousLEDMillis >= IntervalLED) // time is up, so turn LED on
{
// Test to see if battery is charging
battChrg = !digitalRead(23);
//battChrg = digitalRead(23); // Inverse of battery charging - For debugging
if (battChrg > 0) {
digitalWrite(LED_RED, LOW); // LED on
flashLEDstate = 1; // Set the LED state to 1 or "on"
previousLEDMillis += IntervalLED; // save the time when change was made
}
}
} else // ie if LED is on. If on, we must wait for the duration to expire before turning it off
{
if (currentMillis - previousLEDMillis >= LightLED) // time is up, so turn LED off
{
digitalWrite(LED_RED, HIGH); // LED off
flashLEDstate = 0; // Set the LED state to 0 or "off"
previousLEDMillis += LightLED; // save the time when change was made
}
}
}
//================ MESSAGE BUILD FUNCTIONS =====================
// $PTVSOAR,<type>,<value>,<type>,<value>,...*<checksum>
// Note: Checksum is optional
// buildMSG+++++++++++
void buildMSG() // Function to build sentence
{
String msg = F("$PTVSOAR,MNA,FlyingSilicon,MMO,ProbeS4,MSN,2023Q2001"); // FIRST Half of $PTVSOAR Sentence
//printMSG(msg); // Call the printMSG function to transmit to BLE then come back here ...
// // SECOND Half of $PTVSOAR Sentence
//String msg = F("$PTV"); // Short PTV sentence structure (if using)
char buf[100]; // enough characters
sprintf(buf,
",PIT,%06.2f"
//",%08.3f"
",OAT,%+03.0f"
",OAH,%03.0f"
",PCT,%03.0f"
",CHG,%d",
sdpPres // Pitot Differential Pressure 999.99
//, absPres // Atmospheric Pressure 9999.999
,
ambTemp // Air Temperature -99 .. +99
,
ambHumi // Air Relative Humidity 000 - 100
,
batPrct // Battery State of Charge % 000 - 100
,
battChrg > 0 ? 1 : 0); // Battery Charging Flag
msg += buf;
msg.concat(" \n"); // End of sentence notifier for TV
printMSG(msg); // Call the printMSG function to transmit to BLE then come back here ...
}
// END OF BUILD MSG ---------------------------------------------------------
// printMSG+++++++++++
void printMSG(String msg) // Function to transmit to BLE
{
uint8_t buf[64]; // create a variable 'buf1' of size 64
msg.getBytes(buf, msg.length() + 1); // move msg1 characters into 'buf', for length of msg
bleuart.write(buf, msg.length() + 0); // write 'buf2' characters to BLE, for length of msg
msg = ""; // Clear the msg buffer for next read and print
}
// END OF BLE UART WRITE MSG FUNCTION ----------------------------------------