2022-11-16 16:03:33 +01:00

261 lines
5.1 KiB
C++

//
// FILE: Max44007.cpp
// AUTHOR: Rob Tillaart
// VERSION: 0.1.2
// PURPOSE: library for Max44007 lux sensor Arduino
// URL: https://github.com/RobTillaart/Max44007
//
// HISTORY: see changelog.md
#include "Max44007.h"
// MAX44007 KEY VALUES
#define MAX44007_MIN_LUX (0.025)
#define MAX44007_MAX_LUX (104448.0)
#if defined(ESP8266) || defined(ESP32)
Max44007::Max44007(const uint8_t address, const uint8_t dataPin, const uint8_t clockPin)
{
_address = address;
_data = 0;
_error = MAX44007_OK;
_wire = &Wire;
if ((dataPin < 255) && (clockPin < 255))
{
_wire->begin(dataPin, clockPin);
} else {
_wire->begin();
}
}
#endif
Max44007::Max44007(const uint8_t address, const Boolean begin)
{
Max44007::configure(address, &Wire, begin);
}
Max44007::Max44007(const Boolean begin)
{
Max44007::configure(MAX44007_DEFAULT_ADDRESS, &Wire, begin);
}
void Max44007::configure(const uint8_t address, TwoWire *wire, const Boolean begin)
{
_address = address;
_data = 0;
_error = MAX44007_OK;
_wire = wire;
if (begin == Boolean::True)
{
_wire->begin();
}
}
bool Max44007::isConnected()
{
_wire->beginTransmission(_address);
_error = _wire->endTransmission();
return (_error == 0);
}
float Max44007::getLux(void)
{
uint8_t datahigh = read(MAX44007_LUX_READING_HIGH);
if (_error != MAX44007_OK)
{
_error = MAX44007_ERROR_HIGH_BYTE;
return _error;
}
uint8_t datalow = read(MAX44007_LUX_READING_LOW);
if (_error != MAX44007_OK)
{
_error = MAX44007_ERROR_LOW_BYTE;
return _error;
}
uint8_t exponent = datahigh >> 4;
if (exponent == 0x0F)
{
_error = MAX44007_ERROR_OVERFLOW;
return _error;
}
float lux = convertToLux(datahigh, datalow);
return lux;
}
int Max44007::getError()
{
int err = _error;
_error = MAX44007_OK;
return err;
}
bool Max44007::setHighThreshold(const float value)
{
return setThreshold(MAX44007_THRESHOLD_HIGH, value);
}
float Max44007::getHighThreshold(void)
{
return getThreshold(MAX44007_THRESHOLD_HIGH);
}
bool Max44007::setLowThreshold(const float value)
{
return setThreshold(MAX44007_THRESHOLD_LOW, value);
}
float Max44007::getLowThreshold(void)
{
return getThreshold(MAX44007_THRESHOLD_LOW);
}
void Max44007::setThresholdTimer(const uint8_t value)
{
write(MAX44007_THRESHOLD_TIMER, value);
}
uint8_t Max44007::getThresholdTimer()
{
return read(MAX44007_THRESHOLD_TIMER);
}
void Max44007::setConfiguration(const uint8_t value)
{
write(MAX44007_CONFIGURATION, value);
}
uint8_t Max44007::getConfiguration()
{
return read(MAX44007_CONFIGURATION);
}
void Max44007::setAutomaticMode()
{
// CDR & TIM cannot be written in automatic mode
uint8_t config = read(MAX44007_CONFIGURATION);
config &= ~MAX44007_CFG_MANUAL;
write(MAX44007_CONFIGURATION, config);
}
void Max44007::setContinuousMode()
{
uint8_t config = read(MAX44007_CONFIGURATION);
config |= MAX44007_CFG_CONTINUOUS;
write(MAX44007_CONFIGURATION, config);
}
void Max44007::clrContinuousMode()
{
uint8_t config = read(MAX44007_CONFIGURATION);
config &= ~MAX44007_CFG_CONTINUOUS;
write(MAX44007_CONFIGURATION, config);
}
void Max44007::setManualMode(uint8_t CDR, uint8_t TIM)
{
if (CDR !=0) CDR = 1; // only 0 or 1
if (TIM > 7) TIM = 7;
uint8_t config = read(MAX44007_CONFIGURATION);
config |= MAX44007_CFG_MANUAL;
config &= 0xF0; // clear old CDR & TIM bits
config |= CDR << 3 | TIM; // set new CDR & TIM bits
write(MAX44007_CONFIGURATION, config);
}
float Max44007::convertToLux(uint8_t datahigh, uint8_t datalow)
{
uint8_t exponent = datahigh >> 4;
uint32_t mantissa = ((datahigh & 0x0F) << 4) + (datalow & 0x0F);
float lux = ((0x0001 << exponent) * MAX44007_MIN_LUX) * mantissa;
return lux;
}
///////////////////////////////////////////////////////////
//
// PRIVATE
//
bool Max44007::setThreshold(const uint8_t reg, const float value)
{
// CHECK RANGE OF VALUE
if ((value < 0.0) || (value > MAX44007_MAX_LUX)) return false;
uint32_t mantissa = round(value * (1.0 / MAX44007_MIN_LUX)); // compile time optimized.
uint8_t exponent = 0;
while (mantissa > 255)
{
mantissa >>= 1; // bits get lost
exponent++;
};
mantissa = (mantissa >> 4) & 0x0F;
exponent <<= 4;
write(reg, exponent | mantissa);
return true;
}
float Max44007::getThreshold(uint8_t reg)
{
uint8_t datahigh = read(reg);
float lux = convertToLux(datahigh, 0x08); // 0x08 = correction for lost bits
return lux;
}
uint8_t Max44007::read(uint8_t reg)
{
_wire->beginTransmission(_address);
_wire->write(reg);
_error = _wire->endTransmission();
if (_error != MAX44007_OK)
{
return _data; // last value
}
if (_wire->requestFrom(_address, (uint8_t) 1) != 1)
{
_error = MAX44007_ERROR_WIRE_REQUEST;
return _data; // last value
}
_data = _wire->read();
return _data;
}
void Max44007::write(uint8_t reg, uint8_t value)
{
_wire->beginTransmission(_address);
_wire->write(reg);
_wire->write(value);
_error = _wire->endTransmission();
}
// -- END OF FILE --