Jade Perreault
Created May 17, 2019 © CC BY-NC-SA

Elderly Environmental monitoring and automation solution

Bringing together environmental sensors and control modules to enable to non mobile to control and monitor their environment effectively.

AdvancedFull instructions providedOver 2 days100
Elderly Environmental monitoring and automation solution

Things used in this project

Story

Read more

Custom parts and enclosures

BOM for Sensor Shield

Click on the Links and Buy

BOM For Switch Board

Schematics

Gerber FIle For Sensor Board W/Relay

Mesh Switch Gerber

Code

Working Code GPS /Audio/Smoke TEMP,Humidity Data

C/C++
// This #include statement was automatically added by the Particle IDE.
#include <Particle-GPS.h>
#include "MAX30105.h"
#include <Wire.h>
#include <SPI.h>
#include <Adafruit_Sensor.h>
#include <Adafruit_BME280.h>
#include "MAX30105.h"
#define BME_SCK D4
#define BME_MISO D3
#define BME_MOSI D2
#define BME_CS D5
#include <Wire.h>
MAX30105 particleSensor;
#define debug Serial //Uncomment this line if you're using an Uno or ESP
#define SEALEVELPRESSURE_HPA (1013.25)
Adafruit_BME280 bme; // I2C

//========================
Gps _gps = Gps(&Serial1);
Timer _timer = Timer(1, onSerialData);
//=======================
unsigned long delayTime;
//========================
int mic = A4;
const int sampleTime = 50; 
int micOut;
int preValue = 0;
//=======================



void setup() {
   Serial.begin(9600);
   Wire.begin();
   while (!Serial)
       ; // time to get serial running
//   Serial.println(F("BME280 test"));
  unsigned status;

   status = bme.begin();
   if (!status)
   {
 
   }
   Serial.println("-- Default Test --");
   delayTime = 1000;
   Serial.println();
   //============================
   debug.begin(9600);
   debug.println("MAX30105 Basic Readings Example");
 // Initialize sensor
 if (particleSensor.begin() == false) {
   debug.println("MAX30105 was not found. Please check wiring/power. ");
   while (1);
 }
 particleSensor.setup(); //Configure sensor. Use 6.4mA for LED drive
 //=================================
  _gps.begin(9600);
  _gps.sendCommand(PMTK_SET_NMEA_OUTPUT_ALLDATA);
  _timer.start();
 //===============================
}




void onSerialData()
{
  _gps.onSerialData();
}


void loop() {
    
  char message[128];
  char message2[128];
   printValues();
   delay(delayTime);
   sprintf(message,"Temp  %.2f, Pressure %.2f, Alt %.2f, Humidity %.2f",(bme.readTemperature()),(bme.readPressure() / 100.0F),(bme.readAltitude(SEALEVELPRESSURE_HPA)),(bme.readHumidity())) ;
//   sprintf(message2,"Temp  %.2f, Pressure %.2f, Alt %.2f, Humidity %.2f",(bme.readTemperature()),(bme.readPressure() / 100.0F),(bme.readAltitude(SEALEVELPRESSURE_HPA)),(bme.readHumidity())) ;
   Particle.publish("Sensor_Array1",message,PRIVATE);
 //==============================
 int micOutput = findPTPAmp();
   VUMeter(micOutput);  
//==============================
 debug.print(" R[");
 debug.print(particleSensor.getRed());
 debug.print("] IR[");
 debug.print(particleSensor.getIR());
 debug.print("] G[");
 debug.print(particleSensor.getGreen());
 debug.print("]");
 debug.println();

sprintf(message2,"RED %.2d, IR %.2d, GREEN %.2d",particleSensor.getRed(),particleSensor.getIR(),particleSensor.getGreen()) ;
Particle.publish("Sensor_Array2",message2,PRIVATE);
 //=================================
  // ***
  // *** Get the Antenna Status ($PGTOP).
  // ***
  Pgtop pgtop = Pgtop(_gps);
  if (pgtop.parse())
  {
    Serial.println("1) Antenna Status ($PGTOP)");
    Serial.println("======================================================");
    Serial.print("Command ID: "); Serial.println(pgtop.commandId);
    Serial.print("Antenna Status: "); Serial.println(pgtop.reference);
    Serial.println("");
  }

  // ***
  // *** Get the Global Positioning System Fixed Data ($GPGGA).
  // ***
  Gga gga = Gga(_gps);
  if (gga.parse())
  {
    Serial.println("2) Global Positioning System Fixed Data ($GPGGA)");
    Serial.println("======================================================");
    Serial.print("UTC Time: "); Serial.println(gga.utcTime);
    Serial.print("Latitude: "); Serial.println(gga.latitude);
    Serial.print("North/SouthIndicator: "); Serial.println(gga.northSouthIndicator);
    Serial.print("Longitude: "); Serial.println(gga.longitude);
    Serial.print("East/WestIndicator: "); Serial.println(gga.eastWestIndicator);
    Serial.print("Position Fix Indicator: "); Serial.println(gga.positionFixIndicator);
    Serial.print("Satellites Used: "); Serial.println(gga.satellitesUsed);
    Serial.print("Horizontal Dilution of Precision: "); Serial.println(gga.hdop);
    Serial.print("Altitude: "); Serial.print(gga.altitude); Serial.print(" "); Serial.println(gga.altitudeUnit);
    Serial.print("Geoidal Separation: "); Serial.print(gga.geoidalSeparation); Serial.print(" "); Serial.println(gga.geoidalSeparationUnit);
    Serial.print("Age of Diff. Corr.: "); Serial.println(gga.ageOfDiffCorr);
    Serial.println("");
  }

  // ***
  // *** Get the Recommended Minimum Navigation Information ($GPRMC).
  // ***
  Rmc rmc = Rmc(_gps);
  if (rmc.parse())
  {
    Serial.println("3) Recommended Minimum Navigation Information ($GPRMC)");
    Serial.println("======================================================");
    Serial.print("UTC Time: "); Serial.println(rmc.utcTime);
    Serial.print("Latitude: "); Serial.println(rmc.latitude);
    Serial.print("North/SouthIndicator: "); Serial.println(rmc.northSouthIndicator);
    Serial.print("Longitude: "); Serial.println(rmc.longitude);
    Serial.print("East/WestIndicator: "); Serial.println(rmc.eastWestIndicator);
    Serial.print("Speed Over Ground: "); Serial.println(rmc.speedOverGround);
    Serial.print("Course Over Ground: "); Serial.println(rmc.courseOverGround);
    Serial.print("Date: "); Serial.println(rmc.date);
    Serial.print("Magnetic Variation: "); Serial.print(rmc.magneticVariation); Serial.print(" "); Serial.println(rmc.magneticVariationDirection);
    Serial.print("Mode: "); Serial.println(rmc.mode);
    Serial.println("");
  }

  delay(1000);
  
char message4[128];  
sprintf(message4,"Latitude %d,Longitude %d, Speed %d",(rmc.latitude),(rmc.longitude),(rmc.speedOverGround)) ;
Particle.publish("Sensor_Array4",message4,PRIVATE);
delay(100); 
 
 //=============================
}

int findPTPAmp(){

   unsigned long startTime= millis();  // Start of sample window
   unsigned int PTPAmp = 0; 
   unsigned int maxAmp = 0;
   unsigned int minAmp = 4095;
   while(millis() - startTime < sampleTime) {
      micOut = analogRead(mic);
      if( micOut < 4095) //prevent erroneous readings
      {
        if (micOut > maxAmp)
        {
          maxAmp = micOut; //save only the max reading
        }
        else if (micOut < minAmp)
        {
          minAmp = micOut; //save only the min reading
        }
      }
   }

  PTPAmp = maxAmp - minAmp; // (max amp) - (min amp) = peak-to-peak amplitude
  double micOut_Volts = (PTPAmp * 3.3) / 4095; // Convert ADC into voltage
  return PTPAmp;   
}

int VUMeter(int micAmp){
  int fill = map(micAmp, 0, 4095, 0, 255); 
  while(fill != preValue)
  {
    Serial.print("Audio__");
    Serial.println(fill);
    preValue = fill;
  }
char message3[128];
sprintf(message3,"Audio %d",(fill)) ;
Particle.publish("Sensor_Array3",message3,PRIVATE);  
}
void printValues()
{
   Serial.print("Temperature = ");
   Serial.print(bme.readTemperature());
   Serial.println(" *C");
   Serial.print("Pressure = ");
   Serial.print(bme.readPressure() / 100.0F);
   Serial.println(" hPa");
   Serial.print("Approx. Altitude = ");
   Serial.print(bme.readAltitude(SEALEVELPRESSURE_HPA));
   Serial.println(" m");
   Serial.print("Humidity = ");
   Serial.print(bme.readHumidity());
   Serial.println(" %");
   Serial.println();
}

MAX30105.h

C/C++
/*************************************************** 
This is a library written for the Maxim MAX30105 Optical Smoke Detector
It should also work with the MAX30102. However, the MAX30102 does not have a Green LED.
These sensors use I2C to communicate, as well as a single (optional)
interrupt line that is not currently supported in this driver.
Written by Peter Jansen and Nathan Seidle (SparkFun)
BSD license, all text above must be included in any redistribution.
*****************************************************/
#pragma once
#if (ARDUINO >= 100)
#include "Arduino.h"
#else
#include "WProgram.h"
#endif
#include <Wire.h>
#define MAX30105_ADDRESS          0x57 //7-bit I2C Address
//Note that MAX30102 has the same I2C address and Part ID
#define I2C_SPEED_STANDARD        100000
#define I2C_SPEED_FAST            400000
class MAX30105 {
public: 
 MAX30105(void);
 boolean begin(TwoWire &wirePort = Wire, uint32_t i2cSpeed = I2C_SPEED_STANDARD, uint8_t i2caddr = MAX30105_ADDRESS);
 uint32_t getRed(void); //Returns immediate red value
 uint32_t getIR(void); //Returns immediate IR value
 uint32_t getGreen(void); //Returns immediate green value
 bool safeCheck(uint8_t maxTimeToCheck); //Given a max amount of time, check for new data
 // Configuration
 void softReset();
 void shutDown(); 
 void wakeUp(); 
 void setLEDMode(uint8_t mode);
 void setADCRange(uint8_t adcRange);
 void setSampleRate(uint8_t sampleRate);
 void setPulseWidth(uint8_t pulseWidth);
 void setPulseAmplitudeRed(uint8_t value);
 void setPulseAmplitudeIR(uint8_t value);
 void setPulseAmplitudeGreen(uint8_t value);
 void setPulseAmplitudeProximity(uint8_t value);
 void setProximityThreshold(uint8_t threshMSB);
 //Multi-led configuration mode (page 22)
 void enableSlot(uint8_t slotNumber, uint8_t device); //Given slot number, assign a device to slot
 void disableSlots(void);
 // Data Collection
 //Interrupts (page 13, 14)
 uint8_t getINT1(void); //Returns the main interrupt group
 uint8_t getINT2(void); //Returns the temp ready interrupt
 void enableAFULL(void); //Enable/disable individual interrupts
 void disableAFULL(void);
 void enableDATARDY(void);
 void disableDATARDY(void);
 void enableALCOVF(void);
 void disableALCOVF(void);
 void enablePROXINT(void);
 void disablePROXINT(void);
 void enableDIETEMPRDY(void);
 void disableDIETEMPRDY(void);
 //FIFO Configuration (page 18)
 void setFIFOAverage(uint8_t samples);
 void enableFIFORollover();
 void disableFIFORollover();
 void setFIFOAlmostFull(uint8_t samples);
 //FIFO Reading
 uint16_t check(void); //Checks for new data and fills FIFO
 uint8_t available(void); //Tells caller how many new samples are available (head - tail)
 void nextSample(void); //Advances the tail of the sense array
 uint32_t getFIFORed(void); //Returns the FIFO sample pointed to by tail
 uint32_t getFIFOIR(void); //Returns the FIFO sample pointed to by tail
 uint32_t getFIFOGreen(void); //Returns the FIFO sample pointed to by tail
 uint8_t getWritePointer(void);
 uint8_t getReadPointer(void);
 void clearFIFO(void); //Sets the read/write pointers to zero
 //Proximity Mode Interrupt Threshold
 void setPROXINTTHRESH(uint8_t val);
 // Die Temperature
 float readTemperature();
 float readTemperatureF();
 // Detecting ID/Revision
 uint8_t getRevisionID();
 uint8_t readPartID();  
 // Setup the IC with user selectable settings
 void setup(byte powerLevel = 0x1F, byte sampleAverage = 4, byte ledMode = 3, int sampleRate = 400, int pulseWidth = 411, int adcRange = 4096);
 // Low-level I2C communication
 uint8_t readRegister8(uint8_t address, uint8_t reg);
 void writeRegister8(uint8_t address, uint8_t reg, uint8_t value);
private:
 TwoWire *_i2cPort; //The generic connection to user's chosen I2C hardware
 uint8_t _i2caddr;
 //activeLEDs is the number of channels turned on, and can be 1 to 3. 2 is common for Red+IR.
 byte activeLEDs; //Gets set during setup. Allows check() to calculate how many bytes to read from FIFO
 uint8_t revisionID; 
 void readRevisionID();
 void bitMask(uint8_t reg, uint8_t mask, uint8_t thing);
};

MAX30105.cpp

C/C++
/***************************************************
 This is a library written for the Maxim MAX30105 Optical Smoke Detector
 It should also work with the MAX30102. However, the MAX30102 does not have a Green LED.
 These sensors use I2C to communicate, as well as a single (optional)
 interrupt line that is not currently supported in this driver.
 Written by Peter Jansen and Nathan Seidle (SparkFun)
 BSD license, all text above must be included in any redistribution.
*****************************************************/
#include "MAX30105.h"
#define BUFFER_LENGTH 32
// Status Registers
static const uint8_t MAX30105_INTSTAT1 =		0x00;
static const uint8_t MAX30105_INTSTAT2 =		0x01;
static const uint8_t MAX30105_INTENABLE1 =		0x02;
static const uint8_t MAX30105_INTENABLE2 =		0x03;
// FIFO Registers
static const uint8_t MAX30105_FIFOWRITEPTR = 	0x04;
static const uint8_t MAX30105_FIFOOVERFLOW = 	0x05;
static const uint8_t MAX30105_FIFOREADPTR = 	0x06;
static const uint8_t MAX30105_FIFODATA =		0x07;
// Configuration Registers
static const uint8_t MAX30105_FIFOCONFIG = 		0x08;
static const uint8_t MAX30105_MODECONFIG = 		0x09;
static const uint8_t MAX30105_PARTICLECONFIG = 	0x0A;    // Note, sometimes listed as "SPO2" config in datasheet (pg. 11)
static const uint8_t MAX30105_LED1_PULSEAMP = 	0x0C;
static const uint8_t MAX30105_LED2_PULSEAMP = 	0x0D;
static const uint8_t MAX30105_LED3_PULSEAMP = 	0x0E;
static const uint8_t MAX30105_LED_PROX_AMP = 	0x10;
static const uint8_t MAX30105_MULTILEDCONFIG1 = 0x11;
static const uint8_t MAX30105_MULTILEDCONFIG2 = 0x12;
// Die Temperature Registers
static const uint8_t MAX30105_DIETEMPINT = 		0x1F;
static const uint8_t MAX30105_DIETEMPFRAC = 	0x20;
static const uint8_t MAX30105_DIETEMPCONFIG = 	0x21;
// Proximity Function Registers
static const uint8_t MAX30105_PROXINTTHRESH = 	0x30;
// Part ID Registers
static const uint8_t MAX30105_REVISIONID = 		0xFE;
static const uint8_t MAX30105_PARTID = 			0xFF;    // Should always be 0x15. Identical to MAX30102.
// MAX30105 Commands
// Interrupt configuration (pg 13, 14)
static const uint8_t MAX30105_INT_A_FULL_MASK =		(byte)~0b10000000;
static const uint8_t MAX30105_INT_A_FULL_ENABLE = 	0x80;
static const uint8_t MAX30105_INT_A_FULL_DISABLE = 	0x00;
static const uint8_t MAX30105_INT_DATA_RDY_MASK = (byte)~0b01000000;
static const uint8_t MAX30105_INT_DATA_RDY_ENABLE =	0x40;
static const uint8_t MAX30105_INT_DATA_RDY_DISABLE = 0x00;
static const uint8_t MAX30105_INT_ALC_OVF_MASK = (byte)~0b00100000;
static const uint8_t MAX30105_INT_ALC_OVF_ENABLE = 	0x20;
static const uint8_t MAX30105_INT_ALC_OVF_DISABLE = 0x00;
static const uint8_t MAX30105_INT_PROX_INT_MASK = (byte)~0b00010000;
static const uint8_t MAX30105_INT_PROX_INT_ENABLE = 0x10;
static const uint8_t MAX30105_INT_PROX_INT_DISABLE = 0x00;
static const uint8_t MAX30105_INT_DIE_TEMP_RDY_MASK = (byte)~0b00000010;
static const uint8_t MAX30105_INT_DIE_TEMP_RDY_ENABLE = 0x02;
static const uint8_t MAX30105_INT_DIE_TEMP_RDY_DISABLE = 0x00;
static const uint8_t MAX30105_SAMPLEAVG_MASK =	(byte)~0b11100000;
static const uint8_t MAX30105_SAMPLEAVG_1 = 	0x00;
static const uint8_t MAX30105_SAMPLEAVG_2 = 	0x20;
static const uint8_t MAX30105_SAMPLEAVG_4 = 	0x40;
static const uint8_t MAX30105_SAMPLEAVG_8 = 	0x60;
static const uint8_t MAX30105_SAMPLEAVG_16 = 	0x80;
static const uint8_t MAX30105_SAMPLEAVG_32 = 	0xA0;
static const uint8_t MAX30105_ROLLOVER_MASK = 	0xEF;
static const uint8_t MAX30105_ROLLOVER_ENABLE = 0x10;
static const uint8_t MAX30105_ROLLOVER_DISABLE = 0x00;
static const uint8_t MAX30105_A_FULL_MASK = 	0xF0;
// Mode configuration commands (page 19)
static const uint8_t MAX30105_SHUTDOWN_MASK = 	0x7F;
static const uint8_t MAX30105_SHUTDOWN = 		0x80;
static const uint8_t MAX30105_WAKEUP = 			0x00;
static const uint8_t MAX30105_RESET_MASK = 		0xBF;
static const uint8_t MAX30105_RESET = 			0x40;
static const uint8_t MAX30105_MODE_MASK = 		0xF8;
static const uint8_t MAX30105_MODE_REDONLY = 	0x02;
static const uint8_t MAX30105_MODE_REDIRONLY = 	0x03;
static const uint8_t MAX30105_MODE_MULTILED = 	0x07;
// Particle sensing configuration commands (pgs 19-20)
static const uint8_t MAX30105_ADCRANGE_MASK = 	0x9F;
static const uint8_t MAX30105_ADCRANGE_2048 = 	0x00;
static const uint8_t MAX30105_ADCRANGE_4096 = 	0x20;
static const uint8_t MAX30105_ADCRANGE_8192 = 	0x40;
static const uint8_t MAX30105_ADCRANGE_16384 = 	0x60;
static const uint8_t MAX30105_SAMPLERATE_MASK = 0xE3;
static const uint8_t MAX30105_SAMPLERATE_50 = 	0x00;
static const uint8_t MAX30105_SAMPLERATE_100 = 	0x04;
static const uint8_t MAX30105_SAMPLERATE_200 = 	0x08;
static const uint8_t MAX30105_SAMPLERATE_400 = 	0x0C;
static const uint8_t MAX30105_SAMPLERATE_800 = 	0x10;
static const uint8_t MAX30105_SAMPLERATE_1000 = 0x14;
static const uint8_t MAX30105_SAMPLERATE_1600 = 0x18;
static const uint8_t MAX30105_SAMPLERATE_3200 = 0x1C;
static const uint8_t MAX30105_PULSEWIDTH_MASK = 0xFC;
static const uint8_t MAX30105_PULSEWIDTH_69 = 	0x00;
static const uint8_t MAX30105_PULSEWIDTH_118 = 	0x01;
static const uint8_t MAX30105_PULSEWIDTH_215 = 	0x02;
static const uint8_t MAX30105_PULSEWIDTH_411 = 	0x03;
//Multi-LED Mode configuration (pg 22)
static const uint8_t MAX30105_SLOT1_MASK = 		0xF8;
static const uint8_t MAX30105_SLOT2_MASK = 		0x8F;
static const uint8_t MAX30105_SLOT3_MASK = 		0xF8;
static const uint8_t MAX30105_SLOT4_MASK = 		0x8F;
static const uint8_t SLOT_NONE = 				0x00;
static const uint8_t SLOT_RED_LED = 			0x01;
static const uint8_t SLOT_IR_LED = 				0x02;
static const uint8_t SLOT_GREEN_LED = 			0x03;
static const uint8_t SLOT_NONE_PILOT = 			0x04;
static const uint8_t SLOT_RED_PILOT =			0x05;
static const uint8_t SLOT_IR_PILOT = 			0x06;
static const uint8_t SLOT_GREEN_PILOT = 		0x07;
static const uint8_t MAX_30105_EXPECTEDPARTID = 0x15;
//The MAX30105 stores up to 32 samples on the IC
//This is additional local storage to the microcontroller
const int STORAGE_SIZE = 4; //Each long is 4 bytes so limit this to fit on your micro
struct Record
{
 uint32_t red[STORAGE_SIZE];
 uint32_t IR[STORAGE_SIZE];
 uint32_t green[STORAGE_SIZE];
 byte head;
 byte tail;
} sense; //This is our circular buffer of readings from the sensor
MAX30105::MAX30105() {
 // Constructor
}
boolean MAX30105::begin(TwoWire &wirePort, uint32_t i2cSpeed, uint8_t i2caddr) {
 _i2cPort = &wirePort; //Grab which port the user wants us to use
 _i2cPort->begin();
 _i2cPort->setClock(i2cSpeed);
 _i2caddr = i2caddr;
 // Step 1: Initial Communciation and Verification
 // Check that a MAX30105 is connected
 if (!readPartID() == MAX_30105_EXPECTEDPARTID) {
   // Error -- Part ID read from MAX30105 does not match expected part ID.
   // This may mean there is a physical connectivity problem (broken wire, unpowered, etc).
   return false;
 }
 // Populate revision ID
 readRevisionID();
 return true;
}
//
// Configuration
//
//Begin Interrupt configuration
uint8_t MAX30105::getINT1(void) {
 return (readRegister8(_i2caddr, MAX30105_INTSTAT1));
}
uint8_t MAX30105::getINT2(void) {
 return (readRegister8(_i2caddr, MAX30105_INTSTAT2));
}
void MAX30105::enableAFULL(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_A_FULL_MASK, MAX30105_INT_A_FULL_ENABLE);
}
void MAX30105::disableAFULL(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_A_FULL_MASK, MAX30105_INT_A_FULL_DISABLE);
}
void MAX30105::enableDATARDY(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_DATA_RDY_MASK, MAX30105_INT_DATA_RDY_ENABLE);
}
void MAX30105::disableDATARDY(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_DATA_RDY_MASK, MAX30105_INT_DATA_RDY_DISABLE);
}
void MAX30105::enableALCOVF(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_ALC_OVF_MASK, MAX30105_INT_ALC_OVF_ENABLE);
}
void MAX30105::disableALCOVF(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_ALC_OVF_MASK, MAX30105_INT_ALC_OVF_DISABLE);
}
void MAX30105::enablePROXINT(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_PROX_INT_MASK, MAX30105_INT_PROX_INT_ENABLE);
}
void MAX30105::disablePROXINT(void) {
 bitMask(MAX30105_INTENABLE1, MAX30105_INT_PROX_INT_MASK, MAX30105_INT_PROX_INT_DISABLE);
}
void MAX30105::enableDIETEMPRDY(void) {
 bitMask(MAX30105_INTENABLE2, MAX30105_INT_DIE_TEMP_RDY_MASK, MAX30105_INT_DIE_TEMP_RDY_ENABLE);
}
void MAX30105::disableDIETEMPRDY(void) {
 bitMask(MAX30105_INTENABLE2, MAX30105_INT_DIE_TEMP_RDY_MASK, MAX30105_INT_DIE_TEMP_RDY_DISABLE);
}
//End Interrupt configuration
void MAX30105::softReset(void) {
 bitMask(MAX30105_MODECONFIG, MAX30105_RESET_MASK, MAX30105_RESET);
 // Poll for bit to clear, reset is then complete
 // Timeout after 100ms
 unsigned long startTime = millis();
 while (millis() - startTime < 100)
 {
   uint8_t response = readRegister8(_i2caddr, MAX30105_MODECONFIG);
   if ((response & MAX30105_RESET) == 0) break; //We're done!
   delay(1); //Let's not over burden the I2C bus
 }
}
void MAX30105::shutDown(void) {
 // Put IC into low power mode (datasheet pg. 19)
 // During shutdown the IC will continue to respond to I2C commands but will
 // not update with or take new readings (such as temperature)
 bitMask(MAX30105_MODECONFIG, MAX30105_SHUTDOWN_MASK, MAX30105_SHUTDOWN);
}
void MAX30105::wakeUp(void) {
 // Pull IC out of low power mode (datasheet pg. 19)
 bitMask(MAX30105_MODECONFIG, MAX30105_SHUTDOWN_MASK, MAX30105_WAKEUP);
}
void MAX30105::setLEDMode(uint8_t mode) {
 // Set which LEDs are used for sampling -- Red only, RED+IR only, or custom.
 // See datasheet, page 19
 bitMask(MAX30105_MODECONFIG, MAX30105_MODE_MASK, mode);
}
void MAX30105::setADCRange(uint8_t adcRange) {
 // adcRange: one of MAX30105_ADCRANGE_2048, _4096, _8192, _16384
 bitMask(MAX30105_PARTICLECONFIG, MAX30105_ADCRANGE_MASK, adcRange);
}
void MAX30105::setSampleRate(uint8_t sampleRate) {
 // sampleRate: one of MAX30105_SAMPLERATE_50, _100, _200, _400, _800, _1000, _1600, _3200
 bitMask(MAX30105_PARTICLECONFIG, MAX30105_SAMPLERATE_MASK, sampleRate);
}
void MAX30105::setPulseWidth(uint8_t pulseWidth) {
 // pulseWidth: one of MAX30105_PULSEWIDTH_69, _188, _215, _411
 bitMask(MAX30105_PARTICLECONFIG, MAX30105_PULSEWIDTH_MASK, pulseWidth);
}
// NOTE: Amplitude values: 0x00 = 0mA, 0x7F = 25.4mA, 0xFF = 50mA (typical)
// See datasheet, page 21
void MAX30105::setPulseAmplitudeRed(uint8_t amplitude) {
 writeRegister8(_i2caddr, MAX30105_LED1_PULSEAMP, amplitude);
}
void MAX30105::setPulseAmplitudeIR(uint8_t amplitude) {
 writeRegister8(_i2caddr, MAX30105_LED2_PULSEAMP, amplitude);
}
void MAX30105::setPulseAmplitudeGreen(uint8_t amplitude) {
 writeRegister8(_i2caddr, MAX30105_LED3_PULSEAMP, amplitude);
}
void MAX30105::setPulseAmplitudeProximity(uint8_t amplitude) {
 writeRegister8(_i2caddr, MAX30105_LED_PROX_AMP, amplitude);
}
void MAX30105::setProximityThreshold(uint8_t threshMSB) {
 // Set the IR ADC count that will trigger the beginning of particle-sensing mode.
 // The threshMSB signifies only the 8 most significant-bits of the ADC count.
 // See datasheet, page 24.
 writeRegister8(_i2caddr, MAX30105_PROXINTTHRESH, threshMSB);
}
//Given a slot number assign a thing to it
//Devices are SLOT_RED_LED or SLOT_RED_PILOT (proximity)
//Assigning a SLOT_RED_LED will pulse LED
//Assigning a SLOT_RED_PILOT will ??
void MAX30105::enableSlot(uint8_t slotNumber, uint8_t device) {
 uint8_t originalContents;
 switch (slotNumber) {
   case (1):
     bitMask(MAX30105_MULTILEDCONFIG1, MAX30105_SLOT1_MASK, device);
     break;
   case (2):
     bitMask(MAX30105_MULTILEDCONFIG1, MAX30105_SLOT2_MASK, device << 4);
     break;
   case (3):
     bitMask(MAX30105_MULTILEDCONFIG2, MAX30105_SLOT3_MASK, device);
     break;
   case (4):
     bitMask(MAX30105_MULTILEDCONFIG2, MAX30105_SLOT4_MASK, device << 4);
     break;
   default:
     //Shouldn't be here!
     break;
 }
}
//Clears all slot assignments
void MAX30105::disableSlots(void) {
 writeRegister8(_i2caddr, MAX30105_MULTILEDCONFIG1, 0);
 writeRegister8(_i2caddr, MAX30105_MULTILEDCONFIG2, 0);
}
//
// FIFO Configuration
//
//Set sample average (Table 3, Page 18)
void MAX30105::setFIFOAverage(uint8_t numberOfSamples) {
 bitMask(MAX30105_FIFOCONFIG, MAX30105_SAMPLEAVG_MASK, numberOfSamples);
}
//Resets all points to start in a known state
//Page 15 recommends clearing FIFO before beginning a read
void MAX30105::clearFIFO(void) {
 writeRegister8(_i2caddr, MAX30105_FIFOWRITEPTR, 0);
 writeRegister8(_i2caddr, MAX30105_FIFOOVERFLOW, 0);
 writeRegister8(_i2caddr, MAX30105_FIFOREADPTR, 0);
}
//Enable roll over if FIFO over flows
void MAX30105::enableFIFORollover(void) {
 bitMask(MAX30105_FIFOCONFIG, MAX30105_ROLLOVER_MASK, MAX30105_ROLLOVER_ENABLE);
}
//Disable roll over if FIFO over flows
void MAX30105::disableFIFORollover(void) {
 bitMask(MAX30105_FIFOCONFIG, MAX30105_ROLLOVER_MASK, MAX30105_ROLLOVER_DISABLE);
}
//Set number of samples to trigger the almost full interrupt (Page 18)
//Power on default is 32 samples
//Note it is reverse: 0x00 is 32 samples, 0x0F is 17 samples
void MAX30105::setFIFOAlmostFull(uint8_t numberOfSamples) {
 bitMask(MAX30105_FIFOCONFIG, MAX30105_A_FULL_MASK, numberOfSamples);
}
//Read the FIFO Write Pointer
uint8_t MAX30105::getWritePointer(void) {
 return (readRegister8(_i2caddr, MAX30105_FIFOWRITEPTR));
}
//Read the FIFO Read Pointer
uint8_t MAX30105::getReadPointer(void) {
 return (readRegister8(_i2caddr, MAX30105_FIFOREADPTR));
}
// Die Temperature
// Returns temp in C
float MAX30105::readTemperature() {
 // Step 1: Config die temperature register to take 1 temperature sample
 writeRegister8(_i2caddr, MAX30105_DIETEMPCONFIG, 0x01);
 // Poll for bit to clear, reading is then complete
 // Timeout after 100ms
 unsigned long startTime = millis();
 while (millis() - startTime < 100)
 {
   uint8_t response = readRegister8(_i2caddr, MAX30105_DIETEMPCONFIG);
   if ((response & 0x01) == 0) break; //We're done!
   delay(1); //Let's not over burden the I2C bus
 }
 //TODO How do we want to fail? With what type of error?
 //? if(millis() - startTime >= 100) return(-999.0);
 // Step 2: Read die temperature register (integer)
 int8_t tempInt = readRegister8(_i2caddr, MAX30105_DIETEMPINT);
 uint8_t tempFrac = readRegister8(_i2caddr, MAX30105_DIETEMPFRAC);
 // Step 3: Calculate temperature (datasheet pg. 23)
 return (float)tempInt + ((float)tempFrac * 0.0625);
}
// Returns die temp in F
float MAX30105::readTemperatureF() {
 float temp = readTemperature();
 if (temp != -999.0) temp = temp * 1.8 + 32.0;
 return (temp);
}
// Set the PROX_INT_THRESHold
void MAX30105::setPROXINTTHRESH(uint8_t val) {
 writeRegister8(_i2caddr, MAX30105_PROXINTTHRESH, val);
}
//
// Device ID and Revision
//
uint8_t MAX30105::readPartID() {
 return readRegister8(_i2caddr, MAX30105_PARTID);
}
void MAX30105::readRevisionID() {
 revisionID = readRegister8(_i2caddr, MAX30105_REVISIONID);
}
uint8_t MAX30105::getRevisionID() {
 return revisionID;
}
//Setup the sensor
//The MAX30105 has many settings. By default we select:
// Sample Average = 4
// Mode = MultiLED
// ADC Range = 16384 (62.5pA per LSB)
// Sample rate = 50
//Use the default setup if you are just getting started with the MAX30105 sensor
void MAX30105::setup(byte powerLevel, byte sampleAverage, byte ledMode, int sampleRate, int pulseWidth, int adcRange) {
 softReset(); //Reset all configuration, threshold, and data registers to POR values
 //FIFO Configuration
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 //The chip will average multiple samples of same type together if you wish
 if (sampleAverage == 1) setFIFOAverage(MAX30105_SAMPLEAVG_1); //No averaging per FIFO record
 else if (sampleAverage == 2) setFIFOAverage(MAX30105_SAMPLEAVG_2);
 else if (sampleAverage == 4) setFIFOAverage(MAX30105_SAMPLEAVG_4);
 else if (sampleAverage == 8) setFIFOAverage(MAX30105_SAMPLEAVG_8);
 else if (sampleAverage == 16) setFIFOAverage(MAX30105_SAMPLEAVG_16);
 else if (sampleAverage == 32) setFIFOAverage(MAX30105_SAMPLEAVG_32);
 else setFIFOAverage(MAX30105_SAMPLEAVG_4);
 //setFIFOAlmostFull(2); //Set to 30 samples to trigger an 'Almost Full' interrupt
 enableFIFORollover(); //Allow FIFO to wrap/roll over
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 //Mode Configuration
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 if (ledMode == 3) setLEDMode(MAX30105_MODE_MULTILED); //Watch all three LED channels
 else if (ledMode == 2) setLEDMode(MAX30105_MODE_REDIRONLY); //Red and IR
 else setLEDMode(MAX30105_MODE_REDONLY); //Red only
 activeLEDs = ledMode; //Used to control how many bytes to read from FIFO buffer
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 //Particle Sensing Configuration
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 if(adcRange < 4096) setADCRange(MAX30105_ADCRANGE_2048); //7.81pA per LSB
 else if(adcRange < 8192) setADCRange(MAX30105_ADCRANGE_4096); //15.63pA per LSB
 else if(adcRange < 16384) setADCRange(MAX30105_ADCRANGE_8192); //31.25pA per LSB
 else if(adcRange == 16384) setADCRange(MAX30105_ADCRANGE_16384); //62.5pA per LSB
 else setADCRange(MAX30105_ADCRANGE_2048);
 if (sampleRate < 100) setSampleRate(MAX30105_SAMPLERATE_50); //Take 50 samples per second
 else if (sampleRate < 200) setSampleRate(MAX30105_SAMPLERATE_100);
 else if (sampleRate < 400) setSampleRate(MAX30105_SAMPLERATE_200);
 else if (sampleRate < 800) setSampleRate(MAX30105_SAMPLERATE_400);
 else if (sampleRate < 1000) setSampleRate(MAX30105_SAMPLERATE_800);
 else if (sampleRate < 1600) setSampleRate(MAX30105_SAMPLERATE_1000);
 else if (sampleRate < 3200) setSampleRate(MAX30105_SAMPLERATE_1600);
 else if (sampleRate == 3200) setSampleRate(MAX30105_SAMPLERATE_3200);
 else setSampleRate(MAX30105_SAMPLERATE_50);
 //The longer the pulse width the longer range of detection you'll have
 //At 69us and 0.4mA it's about 2 inches
 //At 411us and 0.4mA it's about 6 inches
 if (pulseWidth < 118) setPulseWidth(MAX30105_PULSEWIDTH_69); //Page 26, Gets us 15 bit resolution
 else if (pulseWidth < 215) setPulseWidth(MAX30105_PULSEWIDTH_118); //16 bit resolution
 else if (pulseWidth < 411) setPulseWidth(MAX30105_PULSEWIDTH_215); //17 bit resolution
 else if (pulseWidth == 411) setPulseWidth(MAX30105_PULSEWIDTH_411); //18 bit resolution
 else setPulseWidth(MAX30105_PULSEWIDTH_69);
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 //LED Pulse Amplitude Configuration
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 //Default is 0x1F which gets us 6.4mA
 //powerLevel = 0x02, 0.4mA - Presence detection of ~4 inch
 //powerLevel = 0x1F, 6.4mA - Presence detection of ~8 inch
 //powerLevel = 0x7F, 25.4mA - Presence detection of ~8 inch
 //powerLevel = 0xFF, 50.0mA - Presence detection of ~12 inch
 setPulseAmplitudeRed(powerLevel);
 setPulseAmplitudeIR(powerLevel);
 setPulseAmplitudeGreen(powerLevel);
 setPulseAmplitudeProximity(powerLevel);
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 //Multi-LED Mode Configuration, Enable the reading of the three LEDs
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 enableSlot(1, SLOT_RED_LED);
 if (ledMode > 1) enableSlot(2, SLOT_IR_LED);
 if (ledMode > 2) enableSlot(3, SLOT_GREEN_LED);
 //enableSlot(1, SLOT_RED_PILOT);
 //enableSlot(2, SLOT_IR_PILOT);
 //enableSlot(3, SLOT_GREEN_PILOT);
 //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
 clearFIFO(); //Reset the FIFO before we begin checking the sensor
}
//
// Data Collection
//
//Tell caller how many samples are available
uint8_t MAX30105::available(void)
{
 uint8_t numberOfSamples = sense.head - sense.tail;
 if (numberOfSamples < 0) numberOfSamples += STORAGE_SIZE;
 return (numberOfSamples);
}
//Report the most recent red value
uint32_t MAX30105::getRed(void)
{
 //Check the sensor for new data for 250ms
 if(safeCheck(250))
   return (sense.red[sense.head]);
 else
   return(0); //Sensor failed to find new data
}
//Report the most recent IR value
uint32_t MAX30105::getIR(void)
{
 //Check the sensor for new data for 250ms
 if(safeCheck(250))
   return (sense.IR[sense.head]);
 else
   return(0); //Sensor failed to find new data
}
//Report the most recent Green value
uint32_t MAX30105::getGreen(void)
{
 //Check the sensor for new data for 250ms
 if(safeCheck(250))
   return (sense.green[sense.head]);
 else
   return(0); //Sensor failed to find new data
}
//Report the next Red value in the FIFO
uint32_t MAX30105::getFIFORed(void)
{
 return (sense.red[sense.tail]);
}
//Report the next IR value in the FIFO
uint32_t MAX30105::getFIFOIR(void)
{
 return (sense.IR[sense.tail]);
}
//Report the next Green value in the FIFO
uint32_t MAX30105::getFIFOGreen(void)
{
 return (sense.green[sense.tail]);
}
//Advance the tail
void MAX30105::nextSample(void)
{
 if(available()) //Only advance the tail if new data is available
 {
   sense.tail++;
   sense.tail %= STORAGE_SIZE; //Wrap condition
 }
}
//Polls the sensor for new data
//Call regularly
//If new data is available, it updates the head and tail in the main struct
//Returns number of new samples obtained
uint16_t MAX30105::check(void)
{
 //Read register FIDO_DATA in (3-byte * number of active LED) chunks
 //Until FIFO_RD_PTR = FIFO_WR_PTR
 byte readPointer = getReadPointer();
 byte writePointer = getWritePointer();
 int numberOfSamples = 0;
 //Do we have new data?
 if (readPointer != writePointer)
 {
   //Calculate the number of readings we need to get from sensor
   numberOfSamples = writePointer - readPointer;
   if (numberOfSamples < 0) numberOfSamples += 32; //Wrap condition
   //We now have the number of readings, now calc bytes to read
   //For this example we are just doing Red and IR (3 bytes each)
   int bytesLeftToRead = numberOfSamples * activeLEDs * 3;
   //Get ready to read a burst of data from the FIFO register
   _i2cPort->beginTransmission(MAX30105_ADDRESS);
   _i2cPort->write(MAX30105_FIFODATA);
   _i2cPort->endTransmission();
   //Wire.requestFrom() is limited to BUFFER_LENGTH which is 32 on the Uno
   //We may need to read as many as 288 bytes so we read in blocks no larger than 32
   //BUFFER_LENGTH should work with other platforms with larger requestFrom buffers
   while (bytesLeftToRead > 0)
   {
     int toGet = bytesLeftToRead;
     if (toGet > BUFFER_LENGTH)
     {
       //If toGet is 32 this is bad because we read 6 bytes (Red+IR * 3 = 6) at a time
       //32 % 6 = 2 left over. We don't want to request 32 bytes, we want to request 30.
       //32 % 9 (Red+IR+GREEN) = 5 left over. We want to request 27.
       toGet = BUFFER_LENGTH - (BUFFER_LENGTH % (activeLEDs * 3)); //Trim toGet to be a multiple of the samples we need to read
     }
     bytesLeftToRead -= toGet;
     //Request toGet number of bytes from sensor
     _i2cPort->requestFrom(MAX30105_ADDRESS, toGet);
     while (toGet > 0)
     {
       sense.head++; //Advance the head of the storage struct
       sense.head %= STORAGE_SIZE; //Wrap condition
       byte temp[sizeof(uint32_t)]; //Array of 4 bytes that we will convert into long
       uint32_t tempLong;
       //Burst read three bytes - RED
       temp[3] = 0;
       temp[2] = _i2cPort->read();
       temp[1] = _i2cPort->read();
       temp[0] = _i2cPort->read();
       //Convert array to long
       memcpy(&tempLong, temp, sizeof(tempLong));
		tempLong &= 0x3FFFF; //Zero out all but 18 bits
       sense.red[sense.head] = tempLong; //Store this reading into the sense array
       if (activeLEDs > 1)
       {
         //Burst read three more bytes - IR
         temp[3] = 0;
         temp[2] = _i2cPort->read();
         temp[1] = _i2cPort->read();
         temp[0] = _i2cPort->read();
         //Convert array to long
         memcpy(&tempLong, temp, sizeof(tempLong));
		  tempLong &= 0x3FFFF; //Zero out all but 18 bits
		  sense.IR[sense.head] = tempLong;
       }
       if (activeLEDs > 2)
       {
         //Burst read three more bytes - Green
         temp[3] = 0;
         temp[2] = _i2cPort->read();
         temp[1] = _i2cPort->read();
         temp[0] = _i2cPort->read();
         //Convert array to long
         memcpy(&tempLong, temp, sizeof(tempLong));
		  tempLong &= 0x3FFFF; //Zero out all but 18 bits
         sense.green[sense.head] = tempLong;
       }
       toGet -= activeLEDs * 3;
     }
   } //End while (bytesLeftToRead > 0)
 } //End readPtr != writePtr
 return (numberOfSamples); //Let the world know how much new data we found
}
//Check for new data but give up after a certain amount of time
//Returns true if new data was found
//Returns false if new data was not found
bool MAX30105::safeCheck(uint8_t maxTimeToCheck)
{
 uint32_t markTime = millis();
 while(1)
 {
	if(millis() - markTime > maxTimeToCheck) return(false);
	if(check() == true) //We found new data!
	  return(true);
	delay(1);
 }
}
//Given a register, read it, mask it, and then set the thing
void MAX30105::bitMask(uint8_t reg, uint8_t mask, uint8_t thing)
{
 // Grab current register context
 uint8_t originalContents = readRegister8(_i2caddr, reg);
 // Zero-out the portions of the register we're interested in
 originalContents = originalContents & mask;
 // Change contents
 writeRegister8(_i2caddr, reg, originalContents | thing);
}
//
// Low-level I2C Communication
//
uint8_t MAX30105::readRegister8(uint8_t address, uint8_t reg) {
 _i2cPort->beginTransmission(address);
 _i2cPort->write(reg);
 _i2cPort->endTransmission(false);
 _i2cPort->requestFrom(address, 1);   // Request 1 byte
 int tries = 0;
 while (!_i2cPort->available())
 {
   delay(1);
   if (tries++ > 200) break;
 }
 if (tries == 200) return (0); //Fail
 return (_i2cPort->read());
}
void MAX30105::writeRegister8(uint8_t address, uint8_t reg, uint8_t value) {
 _i2cPort->beginTransmission(address);
 _i2cPort->write(reg);
 _i2cPort->write(value);
 _i2cPort->endTransmission();

Credits

Jade Perreault

Jade Perreault

32 projects • 59 followers
27 Years as a Journeyman Electrician , 4 Years as a PCB design engineer for Penteon Corporation Out of New York ..

Comments