2021-01-29 12:31:58 +01:00
|
|
|
//
|
|
|
|
// FILE: PCF8591.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
|
|
|
// DATE: 2020-03-12
|
2023-09-23 17:31:14 +02:00
|
|
|
// VERSION: 0.2.1
|
2021-01-29 12:31:58 +01:00
|
|
|
// PURPOSE: I2C PCF8591 library for Arduino
|
|
|
|
// URL: https://github.com/RobTillaart/PCF8591
|
|
|
|
|
|
|
|
|
|
|
|
#include "PCF8591.h"
|
|
|
|
|
|
|
|
|
2022-11-21 21:33:51 +01:00
|
|
|
// INTERNAL USE ONLY
|
|
|
|
#define PCF8591_DAC_FLAG 0x40
|
|
|
|
#define PCF8591_INCR_FLAG 0x04
|
|
|
|
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
PCF8591::PCF8591(const uint8_t address, TwoWire *wire)
|
|
|
|
{
|
|
|
|
if ((address < 0x48) || (address > 0x4F))
|
|
|
|
{
|
|
|
|
_error = PCF8591_ADDRESS_ERROR;
|
|
|
|
return;
|
2022-11-21 21:33:51 +01:00
|
|
|
}
|
2021-01-29 12:31:58 +01:00
|
|
|
_address = address;
|
|
|
|
_wire = wire;
|
|
|
|
_control = 0;
|
|
|
|
_dac = 0;
|
|
|
|
for (uint8_t i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
_adc[i] = 0;
|
|
|
|
}
|
|
|
|
_error = PCF8591_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if defined (ESP8266) || defined(ESP32)
|
|
|
|
bool PCF8591::begin(uint8_t sda, uint8_t scl, uint8_t val)
|
|
|
|
{
|
|
|
|
if ((sda < 255) && (scl < 255))
|
|
|
|
{
|
|
|
|
_wire->begin(sda, scl);
|
|
|
|
} else {
|
|
|
|
_wire->begin();
|
|
|
|
}
|
|
|
|
if (!isConnected()) return false;
|
|
|
|
analogWrite(val);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2023-02-23 16:01:06 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
bool PCF8591::begin(uint8_t val)
|
|
|
|
{
|
|
|
|
_wire->begin();
|
|
|
|
if (!isConnected()) return false;
|
|
|
|
analogWrite(val);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool PCF8591::isConnected()
|
|
|
|
{
|
|
|
|
_wire->beginTransmission(_address);
|
2023-02-23 16:01:06 +01:00
|
|
|
_error = _wire->endTransmission(); // default == 0 == PCF8591_OK
|
|
|
|
return (_error == PCF8591_OK);
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
|
2023-02-23 16:01:06 +01:00
|
|
|
|
2022-11-21 21:33:51 +01:00
|
|
|
//////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// ADC PART
|
|
|
|
//
|
|
|
|
void PCF8591::enableINCR()
|
|
|
|
{
|
|
|
|
_control |= PCF8591_INCR_FLAG;
|
2023-02-23 16:01:06 +01:00
|
|
|
}
|
2022-11-21 21:33:51 +01:00
|
|
|
|
|
|
|
|
|
|
|
void PCF8591::disableINCR()
|
|
|
|
{
|
|
|
|
_control &= ~PCF8591_INCR_FLAG;
|
2023-02-23 16:01:06 +01:00
|
|
|
}
|
2022-11-21 21:33:51 +01:00
|
|
|
|
|
|
|
|
|
|
|
bool PCF8591::isINCREnabled()
|
|
|
|
{
|
|
|
|
return ((_control & PCF8591_INCR_FLAG) > 0);
|
2023-02-23 16:01:06 +01:00
|
|
|
}
|
2022-11-21 21:33:51 +01:00
|
|
|
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t PCF8591::analogRead(uint8_t channel, uint8_t mode)
|
|
|
|
{
|
|
|
|
if (mode > 3)
|
|
|
|
{
|
|
|
|
_error = PCF8591_MODE_ERROR;
|
|
|
|
return 0;
|
|
|
|
}
|
2023-02-23 16:01:06 +01:00
|
|
|
_control &= 0b01000100; // clear all except flags
|
2021-01-29 12:31:58 +01:00
|
|
|
_control |= (mode << 4);
|
|
|
|
|
|
|
|
_error = PCF8591_CHANNEL_ERROR;
|
|
|
|
switch(mode)
|
|
|
|
{
|
2023-02-23 16:01:06 +01:00
|
|
|
case PCF8591_FOUR_SINGLE_CHANNEL:
|
2021-01-29 12:31:58 +01:00
|
|
|
if (channel > 3) return 0;
|
|
|
|
break;
|
2023-02-23 16:01:06 +01:00
|
|
|
case PCF8591_THREE_DIFFERENTIAL:
|
2021-01-29 12:31:58 +01:00
|
|
|
if (channel > 2) return 0;
|
|
|
|
break;
|
2023-02-23 16:01:06 +01:00
|
|
|
case PCF8591_MIXED:
|
2021-01-29 12:31:58 +01:00
|
|
|
if (channel > 2) return 0;
|
|
|
|
break;
|
2023-02-23 16:01:06 +01:00
|
|
|
case PCF8591_TWO_DIFFERENTIAL:
|
2021-01-29 12:31:58 +01:00
|
|
|
if (channel > 1) return 0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
2023-02-23 16:01:06 +01:00
|
|
|
_control |= channel;
|
2021-01-29 12:31:58 +01:00
|
|
|
_error = PCF8591_OK;
|
|
|
|
|
2023-02-23 16:01:06 +01:00
|
|
|
// NOTE: one must read two values to get an up to date value.
|
|
|
|
// Page 8 datasheet.
|
2021-01-29 12:31:58 +01:00
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(_control);
|
2023-02-23 16:01:06 +01:00
|
|
|
_error = _wire->endTransmission(); // default == 0 == PCF8591_OK
|
2021-01-29 12:31:58 +01:00
|
|
|
if (_error != 0) return PCF8591_I2C_ERROR;
|
|
|
|
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)2) != 2)
|
|
|
|
{
|
|
|
|
_error = PCF8591_I2C_ERROR;
|
2023-02-23 16:01:06 +01:00
|
|
|
return _adc[channel]; // return last known value
|
2021-01-29 12:31:58 +01:00
|
|
|
}
|
|
|
|
_wire->read();
|
|
|
|
_adc[channel] = _wire->read();
|
|
|
|
return _adc[channel];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t PCF8591::analogRead4()
|
|
|
|
{
|
2023-02-23 16:01:06 +01:00
|
|
|
// clear all except flags
|
|
|
|
// MODE == PCF8591_FOUR_SINGLE_CHANNEL.
|
|
|
|
_control &= 0b01000100;
|
2021-01-29 12:31:58 +01:00
|
|
|
uint8_t channel = 0;
|
|
|
|
_control |= channel;
|
2022-11-21 21:33:51 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
enableINCR();
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(_control);
|
2023-02-23 16:01:06 +01:00
|
|
|
_error = _wire->endTransmission(); // default == 0 == PCF8591_OK
|
2022-11-21 21:33:51 +01:00
|
|
|
if (_error != 0)
|
2021-01-29 12:31:58 +01:00
|
|
|
{
|
|
|
|
_error = PCF8591_I2C_ERROR;
|
|
|
|
disableINCR();
|
|
|
|
return _error;
|
|
|
|
}
|
|
|
|
if (_wire->requestFrom(_address, (uint8_t)5) != 5)
|
|
|
|
{
|
|
|
|
_error = PCF8591_I2C_ERROR;
|
|
|
|
disableINCR();
|
|
|
|
return _error;
|
|
|
|
}
|
|
|
|
|
|
|
|
_wire->read();
|
|
|
|
for (uint8_t i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
_adc[i] = _wire->read();
|
|
|
|
}
|
|
|
|
_error = PCF8591_OK;
|
|
|
|
disableINCR();
|
|
|
|
return _error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-21 21:33:51 +01:00
|
|
|
uint8_t PCF8591::lastRead(uint8_t channel)
|
|
|
|
{
|
|
|
|
return _adc[channel];
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2023-02-23 16:01:06 +01:00
|
|
|
// comparator calls need testing.
|
|
|
|
int PCF8591::readComparator_01()
|
|
|
|
{
|
|
|
|
int8_t v = analogRead(0, 3);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int PCF8591::readComparator_23()
|
|
|
|
{
|
|
|
|
int8_t v = analogRead(1, 3);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int PCF8591::readComparator_03()
|
|
|
|
{
|
|
|
|
int8_t v = analogRead(0, 1);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int PCF8591::readComparator_13()
|
|
|
|
{
|
|
|
|
int8_t v = analogRead(1, 1);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2022-11-21 21:33:51 +01:00
|
|
|
//////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// DAC PART
|
|
|
|
//
|
|
|
|
void PCF8591::enableDAC()
|
|
|
|
{
|
|
|
|
_control |= PCF8591_DAC_FLAG;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void PCF8591::disableDAC()
|
|
|
|
{
|
|
|
|
_control &= ~PCF8591_DAC_FLAG;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
bool PCF8591::isDACEnabled()
|
|
|
|
{
|
|
|
|
return ((_control & PCF8591_DAC_FLAG) > 0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
bool PCF8591::analogWrite(uint8_t value)
|
|
|
|
{
|
|
|
|
_wire->beginTransmission(_address);
|
|
|
|
_wire->write(_control);
|
|
|
|
_wire->write(value);
|
|
|
|
_error = _wire->endTransmission();
|
|
|
|
if (_error != 0)
|
|
|
|
{
|
|
|
|
_error = PCF8591_I2C_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_dac = value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-11-21 21:33:51 +01:00
|
|
|
|
|
|
|
uint8_t PCF8591::lastWrite()
|
|
|
|
{
|
|
|
|
return _dac;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// ERROR HANDLING
|
|
|
|
//
|
2021-01-29 12:31:58 +01:00
|
|
|
int PCF8591::lastError()
|
|
|
|
{
|
|
|
|
int e = _error;
|
|
|
|
_error = PCF8591_OK;
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-21 21:33:51 +01:00
|
|
|
// -- END OF FILE --
|
2021-12-23 14:26:13 +01:00
|
|
|
|