414 lines
9.7 KiB
C++
Raw Normal View History

2021-01-29 12:31:58 +01:00
//
// FILE: MCP_ADC.cpp
// AUTHOR: Rob Tillaart
2023-09-11 20:03:59 +02:00
// VERSION: 0.2.1
2021-01-29 12:31:58 +01:00
// DATE: 2019-10-24
2023-09-11 20:03:59 +02:00
// PURPOSE: Arduino library for MCP3001, MCP3002, MCP3004, MCP3008, MCP3201, MCP3202, MCP3204, MCP3208
2021-01-29 12:31:58 +01:00
// URL: https://github.com/RobTillaart/MCP_ADC
2021-02-06 15:52:51 +01:00
2021-01-29 12:31:58 +01:00
#include "MCP_ADC.h"
2021-02-06 15:52:51 +01:00
2021-01-29 12:31:58 +01:00
MCP_ADC::MCP_ADC(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
{
_dataIn = dataIn;
_dataOut = dataOut;
_clock = clock;
2021-07-31 17:43:47 +02:00
_select = 0;
2021-02-06 15:52:51 +01:00
_hwSPI = (dataIn == 255) || (dataOut == 255) || (clock == 255);
2021-07-31 17:43:47 +02:00
}
void MCP_ADC::begin(uint8_t select)
{
2021-08-01 15:51:21 +02:00
_select = select;
2021-07-31 17:43:47 +02:00
pinMode(_select, OUTPUT);
digitalWrite(_select, HIGH);
2023-09-11 20:03:59 +02:00
digitalWrite(_select, LOW); // force communication See datasheet)
digitalWrite(_select, HIGH);
2021-07-31 17:43:47 +02:00
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
if (_hwSPI)
{
#if defined(ESP32)
2022-11-17 13:00:17 +01:00
if (_useHSPI) // HSPI
2021-07-31 17:43:47 +02:00
{
mySPI = new SPIClass(HSPI);
2021-08-01 15:51:21 +02:00
mySPI->end();
2022-11-17 13:00:17 +01:00
mySPI->begin(14, 12, 13, select); // CLK=14 MISO=12 MOSI=13
2021-07-31 17:43:47 +02:00
}
2022-11-17 13:00:17 +01:00
else // VSPI
2021-07-31 17:43:47 +02:00
{
mySPI = new SPIClass(VSPI);
2021-08-01 15:51:21 +02:00
mySPI->end();
2022-11-17 13:00:17 +01:00
mySPI->begin(18, 19, 23, select); // CLK=18 MISO=19 MOSI=23
2021-07-31 17:43:47 +02:00
}
2022-11-17 13:00:17 +01:00
#else // generic hardware SPI
2021-07-31 17:43:47 +02:00
mySPI = &SPI;
2021-08-01 15:51:21 +02:00
mySPI->end();
2021-07-31 17:43:47 +02:00
mySPI->begin();
#endif
}
2022-11-17 13:00:17 +01:00
else // software SPI
2021-01-29 12:31:58 +01:00
{
pinMode(_dataIn, INPUT);
pinMode(_dataOut, OUTPUT);
pinMode(_clock, OUTPUT);
digitalWrite(_dataOut, LOW);
digitalWrite(_clock, LOW);
2021-02-06 15:52:51 +01:00
}
2021-01-29 12:31:58 +01:00
}
2021-02-06 15:52:51 +01:00
2021-07-31 17:43:47 +02:00
#if defined(ESP32)
void MCP_ADC::setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)
2021-01-29 12:31:58 +01:00
{
2021-07-31 17:43:47 +02:00
_clock = clk;
_dataIn = miso;
_dataOut = mosi;
_select = select;
2021-08-01 15:51:21 +02:00
pinMode(_select, OUTPUT);
digitalWrite(_select, HIGH);
2022-11-17 13:00:17 +01:00
mySPI->end(); // disable SPI
2021-08-01 15:51:21 +02:00
mySPI->begin(clk, miso, mosi, select);
2021-01-29 12:31:58 +01:00
}
2021-07-31 17:43:47 +02:00
#endif
2021-01-29 12:31:58 +01:00
2021-02-06 15:52:51 +01:00
2021-11-08 13:14:13 +01:00
uint32_t MCP_ADC::count()
{
uint32_t cnt = _count;
_count = 0;
return cnt;
}
2021-01-29 12:31:58 +01:00
int16_t MCP_ADC::analogRead(uint8_t channel)
{
if (channel >= _channels) return 0;
return readADC(channel, true);
}
2021-02-06 15:52:51 +01:00
2023-08-16 10:07:24 +02:00
void MCP_ADC::analogReadMultiple(uint8_t channels[], uint8_t numChannels, int16_t readings[])
{
readADCMultiple(channels, numChannels, readings);
}
2021-01-29 12:31:58 +01:00
int16_t MCP_ADC::differentialRead(uint8_t channel)
{
if (channel >= _channels) return 0;
return readADC(channel, false);
}
2021-02-06 15:52:51 +01:00
2021-01-29 12:31:58 +01:00
int16_t MCP_ADC::deltaRead(uint8_t channel)
{
if (channel >= _channels) return 0;
int16_t val0 = differentialRead(channel & 0xFE);
2022-11-17 13:00:17 +01:00
// int16_t val1 = differentialRead(channel | 0x01);
// no need to read if val0 has a positive value
2021-01-29 12:31:58 +01:00
int16_t val1 = (val0 > 0) ? 0 : differentialRead(channel | 0x01);
if (channel & 0x01) return val1 - val0;
return val0 - val1;
}
2021-07-31 17:43:47 +02:00
void MCP_ADC::setSPIspeed(uint32_t speed)
{
_SPIspeed = speed;
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
};
2021-01-29 12:31:58 +01:00
int16_t MCP_ADC::readADC(uint8_t channel, bool single)
{
if (channel >= _channels) return 0;
2021-11-08 13:14:13 +01:00
_count++;
2021-01-29 12:31:58 +01:00
uint8_t data[3] = { 0,0,0 };
uint8_t bytes = buildRequest(channel, single, data);
2021-02-06 15:52:51 +01:00
digitalWrite(_select, LOW);
2021-01-29 12:31:58 +01:00
if (_hwSPI)
{
2021-07-31 17:43:47 +02:00
mySPI->beginTransaction(_spi_settings);
2021-01-29 12:31:58 +01:00
for (uint8_t b = 0; b < bytes; b++)
{
2021-07-31 17:43:47 +02:00
data[b] = mySPI->transfer(data[b]);
2021-01-29 12:31:58 +01:00
}
2021-07-31 17:43:47 +02:00
mySPI->endTransaction();
2021-01-29 12:31:58 +01:00
}
2023-08-16 10:07:24 +02:00
else // Software SPI
2021-01-29 12:31:58 +01:00
{
for (uint8_t b = 0; b < bytes; b++)
{
data[b] = swSPI_transfer(data[b]);
}
}
2021-02-06 15:52:51 +01:00
digitalWrite(_select, HIGH);
2021-01-29 12:31:58 +01:00
2023-09-11 20:03:59 +02:00
if (bytes == 2)
{
// combine bytes
int16_t raw = 256 * data[0] + data[1];
// patch bit pattern MCP3001
if ((_channels == 1) && (_maxValue == 1023)) raw >>= 3;
// patch bit pattern MCP3201
if ((_channels == 1) && (_maxValue == 4095)) raw >>= 1;
return raw & _maxValue;
}
2021-01-29 12:31:58 +01:00
// data[0]?
return ((256 * data[1] + data[2]) & _maxValue);
}
2021-02-06 15:52:51 +01:00
2023-08-16 10:07:24 +02:00
void MCP_ADC::readADCMultiple(uint8_t channels[], uint8_t numChannels, int16_t readings[])
{
_count += numChannels;
if (_hwSPI) {
mySPI->beginTransaction(_spi_settings);
}
for (uint8_t i = 0; i < numChannels; i++) {
digitalWrite(_select, LOW);
uint8_t data[3] = {0, 0, 0};
uint8_t bytes = buildRequest(channels[i], true, data);
if (_hwSPI) {
for (uint8_t b = 0; b < bytes; b++) {
data[b] = mySPI->transfer(data[b]);
}
} else {
for (uint8_t b = 0; b < bytes; b++) {
data[b] = swSPI_transfer(data[b]);
}
}
if (bytes == 2) {
2023-09-11 20:03:59 +02:00
// combine bytes
int16_t raw = 256 * data[0] + data[1];
// patch bit pattern MCP3001
if ((_channels == 1) && (_maxValue == 1023)) raw >>= 3;
// patch bit pattern MCP3201
if ((_channels == 1) && (_maxValue == 4095)) raw >>= 1;
readings[i] = raw & _maxValue;
2023-08-16 10:07:24 +02:00
} else {
readings[i] = ((256 * data[1] + data[2]) & _maxValue);
}
digitalWrite(_select, HIGH);
}
if (_hwSPI) {
mySPI->endTransaction();
}
}
2022-11-17 13:00:17 +01:00
// MSBFIRST
2021-01-29 12:31:58 +01:00
uint8_t MCP_ADC::swSPI_transfer(uint8_t val)
{
2021-07-31 17:43:47 +02:00
uint8_t clk = _clock;
uint8_t dao = _dataOut;
uint8_t dai = _dataIn;
2021-11-08 13:14:13 +01:00
2021-01-29 12:31:58 +01:00
uint8_t rv = 0;
for (uint8_t mask = 0x80; mask; mask >>= 1)
{
2021-07-31 17:43:47 +02:00
digitalWrite(dao, (val & mask));
digitalWrite(clk, HIGH);
if (digitalRead(dai) == HIGH) rv |= mask;
digitalWrite(clk, LOW);
2021-01-29 12:31:58 +01:00
}
return rv;
}
2023-09-11 20:03:59 +02:00
/////////////////////////////////////////////////////////////////////////////
//
// MCP3001
//
MCP3001::MCP3001(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 1;
_maxValue = 1023;
}
uint8_t MCP3001::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
// P16 fig 6.1 MCP3001
// no specific data needed
// keep build CI compiler (ESP32) happy next statement
data[0] = data[1] = 0;
if ((channel == 0) || (single == false)) return 2;
return 2;
}
2021-01-29 12:31:58 +01:00
/////////////////////////////////////////////////////////////////////////////
//
2022-11-17 13:00:17 +01:00
// MCP3002
2021-11-08 13:14:13 +01:00
//
2021-01-29 12:31:58 +01:00
MCP3002::MCP3002(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 2;
_maxValue = 1023;
}
uint8_t MCP3002::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
2022-11-17 13:00:17 +01:00
// P17 fig 6.1 MCP3002
data[0] = 0x44; // start bit + MSB first bit
if (single) data[0] |= 0x20; // single read | differential
if (channel) data[0] |= (channel << 4); // channel = 0 or 1;
2021-01-29 12:31:58 +01:00
return 2;
}
/////////////////////////////////////////////////////////////////////////////
//
2022-11-17 13:00:17 +01:00
// MCP3004
2021-11-08 13:14:13 +01:00
//
2021-01-29 12:31:58 +01:00
MCP3004::MCP3004(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 4;
_maxValue = 1023;
}
uint8_t MCP3004::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
2022-11-17 13:00:17 +01:00
// P21 fig 6.1 MCP3004/3008
data[0] = 0x01; // start bit
if (single) data[1] = 0x80; // single read | differential
if (channel) data[1] |= (channel << 4); // channel
2021-01-29 12:31:58 +01:00
return 3;
}
/////////////////////////////////////////////////////////////////////////////
//
2022-11-17 13:00:17 +01:00
// MCP3008
2021-11-08 13:14:13 +01:00
//
2021-01-29 12:31:58 +01:00
MCP3008::MCP3008(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 8;
_maxValue = 1023;
}
uint8_t MCP3008::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
2022-11-17 13:00:17 +01:00
// P21 fig 6.1 MCP3004/3008
data[0] = 0x01; // start bit
if (single) data[1] = 0x80; // single read | differential
if (channel) data[1] |= (channel << 4); // channel
2021-01-29 12:31:58 +01:00
return 3;
}
2023-08-16 10:07:24 +02:00
/////////////////////////////////////////////////////////////////////////////
//
// MCP3201
//
MCP3201::MCP3201(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 1;
_maxValue = 4095;
}
uint8_t MCP3201::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
// P21 fig 6.1 MCP3201
// no specific data needed
2023-09-11 20:03:59 +02:00
// keep build CI compiler (ESP32) happy next statements
data[0] = data[1] = 0;
2023-08-16 10:07:24 +02:00
if ((channel == 0) || (single == false)) return 2;
return 2;
}
2021-01-29 12:31:58 +01:00
/////////////////////////////////////////////////////////////////////////////
//
2022-11-17 13:00:17 +01:00
// MCP3202
2021-11-08 13:14:13 +01:00
//
2021-01-29 12:31:58 +01:00
MCP3202::MCP3202(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 2;
_maxValue = 4095;
}
uint8_t MCP3202::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
2022-11-17 13:00:17 +01:00
// P17 fig 6.2 MCP3202
data[0] = 0x01; // start bit
data[1] = 0x20; // MSB first bit
if (single) data[1] |= 0x80; // single read | differential
if (channel) data[1] |= (channel << 6); // channel = 0 or 1;
2021-01-29 12:31:58 +01:00
return 3;
}
/////////////////////////////////////////////////////////////////////////////
//
2022-11-17 13:00:17 +01:00
// MCP3204
2021-11-08 13:14:13 +01:00
//
2021-01-29 12:31:58 +01:00
MCP3204::MCP3204(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 4;
_maxValue = 4095;
}
uint8_t MCP3204::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
2022-11-17 13:00:17 +01:00
// P21 fig 6.1 MCP3204/3208
data[0] = 0x04; // start bit
if (single) data[0] |= 0x02; // single read | differential
if (channel > 3) data[0] |= 0x01; // MSB channel (D2)
if (channel) data[1] |= (channel << 6); // other 2 bits (D1 D0)
2021-01-29 12:31:58 +01:00
return 3;
}
/////////////////////////////////////////////////////////////////////////////
//
2022-11-17 13:00:17 +01:00
// MCP3208
2021-11-08 13:14:13 +01:00
//
2021-01-29 12:31:58 +01:00
MCP3208::MCP3208(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
:MCP_ADC(dataIn, dataOut, clock)
{
_channels = 8;
_maxValue = 4095;
}
uint8_t MCP3208::buildRequest(uint8_t channel, bool single, uint8_t * data)
{
2022-11-17 13:00:17 +01:00
// P21 fig 6.1 MCP3204/3208
data[0] = 0x04; // start bit
if (single) data[0] |= 0x02; // single read | differential
if (channel > 3) data[0] |= 0x01; // MSB channel (D2)
if (channel) data[1] |= (channel << 6); // other 2 bits (D1 D0)
2021-01-29 12:31:58 +01:00
return 3;
}
// -- END OF FILE --
2021-12-21 15:31:31 +01:00