mirror of
https://github.com/RobTillaart/Arduino.git
synced 2024-10-03 18:09:02 -04:00
374 lines
8.5 KiB
C++
374 lines
8.5 KiB
C++
//
|
|
// FILE: MCP_ADC.cpp
|
|
// AUTHOR: Rob Tillaart
|
|
// VERSION: 0.1.9
|
|
// DATE: 2019-10-24
|
|
// PURPOSE: Arduino library for MCP3002, MCP3004, MCP3008, MCP3202, MCP3204, MCP3208
|
|
// URL: https://github.com/RobTillaart/MCP_ADC
|
|
|
|
|
|
#include "MCP_ADC.h"
|
|
|
|
|
|
MCP_ADC::MCP_ADC(uint8_t dataIn, uint8_t dataOut, uint8_t clock)
|
|
{
|
|
_dataIn = dataIn;
|
|
_dataOut = dataOut;
|
|
_clock = clock;
|
|
_select = 0;
|
|
_hwSPI = (dataIn == 255) || (dataOut == 255) || (clock == 255);
|
|
}
|
|
|
|
|
|
void MCP_ADC::begin(uint8_t select)
|
|
{
|
|
_select = select;
|
|
pinMode(_select, OUTPUT);
|
|
digitalWrite(_select, HIGH);
|
|
|
|
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
|
|
|
|
if (_hwSPI)
|
|
{
|
|
#if defined(ESP32)
|
|
if (_useHSPI) // HSPI
|
|
{
|
|
mySPI = new SPIClass(HSPI);
|
|
mySPI->end();
|
|
mySPI->begin(14, 12, 13, select); // CLK=14 MISO=12 MOSI=13
|
|
}
|
|
else // VSPI
|
|
{
|
|
mySPI = new SPIClass(VSPI);
|
|
mySPI->end();
|
|
mySPI->begin(18, 19, 23, select); // CLK=18 MISO=19 MOSI=23
|
|
}
|
|
#else // generic hardware SPI
|
|
mySPI = &SPI;
|
|
mySPI->end();
|
|
mySPI->begin();
|
|
#endif
|
|
}
|
|
else // software SPI
|
|
{
|
|
pinMode(_dataIn, INPUT);
|
|
pinMode(_dataOut, OUTPUT);
|
|
pinMode(_clock, OUTPUT);
|
|
digitalWrite(_dataOut, LOW);
|
|
digitalWrite(_clock, LOW);
|
|
}
|
|
}
|
|
|
|
|
|
#if defined(ESP32)
|
|
void MCP_ADC::setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)
|
|
{
|
|
_clock = clk;
|
|
_dataIn = miso;
|
|
_dataOut = mosi;
|
|
_select = select;
|
|
pinMode(_select, OUTPUT);
|
|
digitalWrite(_select, HIGH);
|
|
|
|
mySPI->end(); // disable SPI
|
|
mySPI->begin(clk, miso, mosi, select);
|
|
}
|
|
#endif
|
|
|
|
|
|
uint32_t MCP_ADC::count()
|
|
{
|
|
uint32_t cnt = _count;
|
|
_count = 0;
|
|
return cnt;
|
|
}
|
|
|
|
|
|
int16_t MCP_ADC::analogRead(uint8_t channel)
|
|
{
|
|
if (channel >= _channels) return 0;
|
|
return readADC(channel, true);
|
|
}
|
|
|
|
|
|
void MCP_ADC::analogReadMultiple(uint8_t channels[], uint8_t numChannels, int16_t readings[])
|
|
{
|
|
readADCMultiple(channels, numChannels, readings);
|
|
}
|
|
|
|
|
|
int16_t MCP_ADC::differentialRead(uint8_t channel)
|
|
{
|
|
if (channel >= _channels) return 0;
|
|
return readADC(channel, false);
|
|
}
|
|
|
|
|
|
int16_t MCP_ADC::deltaRead(uint8_t channel)
|
|
{
|
|
if (channel >= _channels) return 0;
|
|
|
|
int16_t val0 = differentialRead(channel & 0xFE);
|
|
// int16_t val1 = differentialRead(channel | 0x01);
|
|
// no need to read if val0 has a positive value
|
|
int16_t val1 = (val0 > 0) ? 0 : differentialRead(channel | 0x01);
|
|
|
|
if (channel & 0x01) return val1 - val0;
|
|
return val0 - val1;
|
|
}
|
|
|
|
|
|
void MCP_ADC::setSPIspeed(uint32_t speed)
|
|
{
|
|
_SPIspeed = speed;
|
|
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
|
|
};
|
|
|
|
|
|
int16_t MCP_ADC::readADC(uint8_t channel, bool single)
|
|
{
|
|
if (channel >= _channels) return 0;
|
|
|
|
_count++;
|
|
|
|
uint8_t data[3] = { 0,0,0 };
|
|
uint8_t bytes = buildRequest(channel, single, data);
|
|
|
|
digitalWrite(_select, LOW);
|
|
if (_hwSPI)
|
|
{
|
|
mySPI->beginTransaction(_spi_settings);
|
|
for (uint8_t b = 0; b < bytes; b++)
|
|
{
|
|
data[b] = mySPI->transfer(data[b]);
|
|
}
|
|
mySPI->endTransaction();
|
|
}
|
|
else // Software SPI
|
|
{
|
|
for (uint8_t b = 0; b < bytes; b++)
|
|
{
|
|
data[b] = swSPI_transfer(data[b]);
|
|
}
|
|
}
|
|
digitalWrite(_select, HIGH);
|
|
|
|
if (bytes == 2) return ((256 * data[0] + data[1]) & _maxValue);
|
|
// data[0]?
|
|
return ((256 * data[1] + data[2]) & _maxValue);
|
|
}
|
|
|
|
|
|
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) {
|
|
readings[i] = ((256 * data[0] + data[1]) & _maxValue);
|
|
} else {
|
|
readings[i] = ((256 * data[1] + data[2]) & _maxValue);
|
|
}
|
|
|
|
digitalWrite(_select, HIGH);
|
|
}
|
|
|
|
if (_hwSPI) {
|
|
mySPI->endTransaction();
|
|
}
|
|
}
|
|
|
|
|
|
// MSBFIRST
|
|
uint8_t MCP_ADC::swSPI_transfer(uint8_t val)
|
|
{
|
|
uint8_t clk = _clock;
|
|
uint8_t dao = _dataOut;
|
|
uint8_t dai = _dataIn;
|
|
|
|
uint8_t rv = 0;
|
|
for (uint8_t mask = 0x80; mask; mask >>= 1)
|
|
{
|
|
digitalWrite(dao, (val & mask));
|
|
digitalWrite(clk, HIGH);
|
|
if (digitalRead(dai) == HIGH) rv |= mask;
|
|
digitalWrite(clk, LOW);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MCP3002
|
|
//
|
|
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)
|
|
{
|
|
// 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;
|
|
return 2;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MCP3004
|
|
//
|
|
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)
|
|
{
|
|
// 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
|
|
return 3;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MCP3008
|
|
//
|
|
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)
|
|
{
|
|
// 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
|
|
return 3;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// 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
|
|
// keep build CI compiler (ESP32) happy next statement
|
|
if ((channel == 0) || (single == false)) return 2;
|
|
return 2;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MCP3202
|
|
//
|
|
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)
|
|
{
|
|
// 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;
|
|
return 3;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MCP3204
|
|
//
|
|
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)
|
|
{
|
|
// 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)
|
|
return 3;
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MCP3208
|
|
//
|
|
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)
|
|
{
|
|
// 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)
|
|
return 3;
|
|
}
|
|
|
|
|
|
// -- END OF FILE --
|
|
|