134 lines
2.8 KiB
C++
Raw Normal View History

2021-05-30 14:16:15 +02:00
//
// FILE: TSL235R.cpp
// AUTHOR: Rob Tillaart
2023-02-18 15:26:18 +01:00
// VERSION: 0.1.4
// DATE: 2020-05-29
2021-12-29 13:37:09 +01:00
// PURPOSE: library for the TSL235R light to frequency convertor
2023-02-18 15:26:18 +01:00
// URL: https://github.com/RobTillaart/TSL235R
2021-05-30 14:16:15 +02:00
#include "TSL235R.h"
TSL235R::TSL235R(float voltage)
{
2023-02-18 15:26:18 +01:00
_waveLength = 635;
_waveLengthFactor = 1.0;
_voltageFactor = 1.0;
_voltage = voltage;
2021-05-30 14:16:15 +02:00
calculateFactor();
}
float TSL235R::irradiance(uint32_t Hz)
{
return Hz * _factor;
}
float TSL235R::irradiance(uint32_t pulses, uint32_t milliseconds)
{
return (pulses * 1000.0 * _factor) / milliseconds;
}
2023-02-18 15:26:18 +01:00
2021-06-04 15:58:39 +02:00
float TSL235R::irradiance_HS(uint32_t pulses, uint32_t microseconds)
{
return (pulses * 1000000.0 * _factor) / microseconds;
}
2021-05-30 14:16:15 +02:00
2023-02-18 15:26:18 +01:00
float TSL235R::getFactor()
{
return _factor;
}
2021-05-30 14:16:15 +02:00
void TSL235R::setWavelength(uint16_t wavelength)
{
_waveLength = wavelength;
calculateFactor();
}
2023-02-18 15:26:18 +01:00
uint16_t TSL235R::getWavelength()
{
return _waveLength;
}
float TSL235R::getWaveLengthFactor()
{
return _waveLengthFactor;
}
2021-05-30 14:16:15 +02:00
void TSL235R::setVoltage(float voltage)
{
_voltage = voltage;
calculateFactor();
}
2021-12-29 13:37:09 +01:00
2023-02-18 15:26:18 +01:00
float TSL235R::getVoltage()
{
return _voltage;
}
float TSL235R::getVoltageFactor()
{
return _voltageFactor;
}
2021-05-30 14:16:15 +02:00
void TSL235R::calculateFactor()
{
2022-11-26 17:40:57 +01:00
// figure 1 datasheet
// 1 KHz crosses the line at 35/230 between 1 and 10.
2023-02-18 15:26:18 +01:00
// so the correction factor is 10^0.15217 = 1.419659
2022-11-26 17:40:57 +01:00
// as the graph is in kHz we need to correct a factor 1000
// as the irradiance function gets Hz
2023-02-18 15:26:18 +01:00
const float cf = 0.001419659;
2022-11-26 17:40:57 +01:00
_waveLengthFactor = calculateWaveLengthFactor(_waveLength);
2021-05-30 14:16:15 +02:00
_voltageFactor = 0.988 + (_voltage - 2.7) * (0.015 / 2.8);
_factor = cf * _waveLengthFactor * _voltageFactor;
}
2022-11-26 17:40:57 +01:00
float TSL235R::calculateWaveLengthFactor(uint16_t _waveLength)
2021-05-30 14:16:15 +02:00
{
2022-11-26 17:40:57 +01:00
// figure 2 datasheet
// 635 nm is reference 1.000
// remaining is linear interpolated between points in the graph
2021-05-30 14:16:15 +02:00
float in[] = { 300, 350, 400, 500, 600, 635, 700, 750, 800, 850, 900, 1000, 1100};
float out[] = { 0.1, 0.35, 0.5, 0.75, 0.93, 1.00, 1.15, 1.20, 1.15, 1.10, 0.95, 0.40, 0.10};
return 1.0 / multiMap(_waveLength, in, out, 13);
}
2023-02-18 15:26:18 +01:00
// from https://github.com/RobTillaart/MultiMap
2021-12-29 13:37:09 +01:00
float TSL235R::multiMap(float value, float * _in, float * _out, uint8_t size)
2021-05-30 14:16:15 +02:00
{
2022-11-26 17:40:57 +01:00
// take care the value is within range
// value = constrain(value, _in[0], _in[size-1]);
2021-12-29 13:37:09 +01:00
if (value <= _in[0]) return _out[0];
if (value >= _in[size-1]) return _out[size-1];
2021-05-30 14:16:15 +02:00
2022-11-26 17:40:57 +01:00
// search right interval
2023-02-18 15:26:18 +01:00
uint8_t pos = 1; // _in[0] already tested
2021-12-29 13:37:09 +01:00
while(value > _in[pos]) pos++;
2021-05-30 14:16:15 +02:00
2022-11-26 17:40:57 +01:00
// this will handle all exact "points" in the _in array
2021-12-29 13:37:09 +01:00
if (value == _in[pos]) return _out[pos];
2021-05-30 14:16:15 +02:00
2022-11-26 17:40:57 +01:00
// interpolate in the right segment for the rest
2021-12-29 13:37:09 +01:00
uint8_t pos1 = pos - 1;
return (value - _in[pos1]) * (_out[pos] - _out[pos1]) / (_in[pos] - _in[pos1]) + _out[pos1];
2021-05-30 14:16:15 +02:00
}
2022-11-26 17:40:57 +01:00
// -- END OF FILE --
2021-12-29 13:37:09 +01:00