GY-63_MS5611/libraries/MCP_DAC/MCP_DAC.cpp

374 lines
7.4 KiB
C++
Raw Normal View History

2021-05-26 05:38:28 -04:00
//
// FILE: MCP_DAC.cpp
// AUTHOR: Rob Tillaart
2022-10-02 11:19:23 -04:00
// VERSION: 0.1.7
2021-05-26 05:38:28 -04:00
// DATE: 2021-02-03
// PURPOSE: Arduino library for MCP_DAC
// URL: https://github.com/RobTillaart/MCP_DAC
//
// HISTORY
// 0.1.0 2021-02-03 initial version
// 0.1.1 2021-05-26 moved SPI.begin() from constructor to begin()
2021-07-31 11:44:31 -04:00
// 0.1.2 2021-07-29 VSPI / HSPI support for ESP32 (default pins only
// faster software SPI
// minor optimizations / refactor
2021-11-08 10:16:58 -05:00
// 0.1.3 2021-07-31 add increment() and decrement()
2021-08-01 09:52:01 -04:00
// 0.1.4 2021-08-01 fix setGPIOpins() - needs more testing.
2021-11-08 10:16:58 -05:00
// 0.1.5 2021-11-08 update build-CI, badges,
// default parameter 1 for setGain()
// default parameter false for setBufferedMode()
// default parameter 0 for getPercentage()
// extended unit tests
2021-12-21 10:10:14 -05:00
// 0.1.6 2021-12-21 update library.json, license, minor edits
2022-10-02 11:19:23 -04:00
// 0.1.7 2022-10-02 support for RP2040 pico (kudos to Intubin)
// update documentation for RP2040 (kudos to Intubin)
// update build-CI to support RP2040
2021-07-31 11:44:31 -04:00
2021-05-26 05:38:28 -04:00
#include "MCP_DAC.h"
MCP_DAC::MCP_DAC(uint8_t dataOut, uint8_t clock)
{
_dataOut = dataOut;
_clock = clock;
2021-07-31 11:44:31 -04:00
_select = 0;
2021-05-26 05:38:28 -04:00
_hwSPI = (dataOut == 255) || (clock == 255);
_channels = 1;
_maxValue = 255;
reset();
}
void MCP_DAC::reset()
{
_gain = 1;
_value[0] = 0;
_value[1] = 0;
_buffered = false;
_active = true;
}
void MCP_DAC::begin(uint8_t select)
{
_select = select;
pinMode(_select, OUTPUT);
digitalWrite(_select, HIGH);
2021-07-31 11:44:31 -04:00
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
2021-05-26 05:38:28 -04:00
if (_hwSPI)
{
2021-07-31 11:44:31 -04:00
#if defined(ESP32)
if (_useHSPI) // HSPI
{
mySPI = new SPIClass(HSPI);
2021-08-01 09:52:01 -04:00
mySPI->end();
mySPI->begin(14, 12, 13, select); // CLK=14 MISO=12 MOSI=13
2021-07-31 11:44:31 -04:00
}
else // VSPI
{
mySPI = new SPIClass(VSPI);
2021-08-01 09:52:01 -04:00
mySPI->end();
mySPI->begin(18, 19, 23, select); // CLK=18 MISO=19 MOSI=23
2021-07-31 11:44:31 -04:00
}
2022-10-02 11:19:23 -04:00
#elif defined(ARDUINO_ARCH_RP2040)
if (_useSPI1 == true) {
mySPI = &SPI1;
}else{
mySPI = &SPI;
}
mySPI->end();
mySPI->begin();
2021-08-01 09:52:01 -04:00
#else // generic hardware SPI
2021-07-31 11:44:31 -04:00
mySPI = &SPI;
2021-08-01 09:52:01 -04:00
mySPI->end();
2021-07-31 11:44:31 -04:00
mySPI->begin();
#endif
}
else // software SPI
{
pinMode(_dataOut, OUTPUT);
pinMode(_clock, OUTPUT);
digitalWrite(_dataOut, LOW);
digitalWrite(_clock, LOW);
2021-05-26 05:38:28 -04:00
}
}
2022-10-02 11:19:23 -04:00
#if defined(ESP32) or defined(ARDUINO_ARCH_RP2040)
2021-07-31 11:44:31 -04:00
void MCP_DAC::setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)
{
_clock = clk;
_dataOut = mosi;
_select = select;
2021-08-01 09:52:01 -04:00
pinMode(_select, OUTPUT);
digitalWrite(_select, HIGH);
2021-12-21 10:10:14 -05:00
mySPI->end(); // disable SPI
2022-10-02 11:19:23 -04:00
#if defined(ESP32)
2021-08-01 09:52:01 -04:00
mySPI->begin(clk, miso, mosi, select);
2022-10-02 11:19:23 -04:00
#elif defined(ARDUINO_ARCH_RP2040)
mySPI->setCS(select);
mySPI->setSCK(clk);
mySPI->setTX(mosi);
mySPI->begin();
#endif
2021-07-31 11:44:31 -04:00
}
#endif
2021-05-26 05:38:28 -04:00
bool MCP_DAC::setGain(uint8_t gain)
{
2021-11-08 10:16:58 -05:00
if ((0 == gain) || (gain > 2)) return false;
2021-05-26 05:38:28 -04:00
_gain = gain;
return true;
}
bool MCP_DAC::analogWrite(uint16_t value, uint8_t channel)
{
if (channel >= _channels) return false;
// CONSTRAIN VALUE
uint16_t _val = value;
if (_val > _maxValue) _val = _maxValue;
_value[channel] = value;
// PREPARING THE DATA TRANSFER
uint16_t data = 0x1000;
if (channel == 1) data |= 0x8000;
if (_buffered) data |= 0x4000;
if (_gain == 1) data |= 0x2000;
if (_maxValue == 4095) data |= _val;
else if (_maxValue == 1023) data |= (_val << 2);
else data |= (_val << 4);
transfer(data);
return true;
}
void MCP_DAC::fastWriteA(uint16_t value)
{
transfer(0x3000 | value);
}
void MCP_DAC::fastWriteB(uint16_t value)
{
transfer(0xB000 | value);
}
2021-08-01 09:52:01 -04:00
bool MCP_DAC::increment(uint8_t channel)
{
if (channel >= _channels) return false;
if (_value[channel] == _maxValue) return false;
return analogWrite(_value[channel] + 1, channel);
}
bool MCP_DAC::decrement(uint8_t channel)
{
if (channel >= _channels) return false;
if (_value[channel] == 0) return false;
return analogWrite(_value[channel] - 1, channel);
}
2021-05-26 05:38:28 -04:00
void MCP_DAC::setPercentage(float perc, uint8_t channel)
{
if (perc < 0) perc = 0;
if (perc > 100) perc = 100;
analogWrite(perc * _maxValue, channel);
}
float MCP_DAC::getPercentage(uint8_t channel)
{
return (_value[channel] * 100.0) / _maxValue;
}
void MCP_DAC::setLatchPin(uint8_t latchPin)
{
_latchPin = latchPin;
pinMode(_latchPin, OUTPUT);
digitalWrite(_latchPin, LOW);
}
void MCP_DAC::triggerLatch()
{
2021-07-31 11:44:31 -04:00
if (_latchPin != 255)
{
digitalWrite(_latchPin, HIGH);
delayMicroseconds(1); // 100 ns - Page 7
digitalWrite(_latchPin, LOW);
}
2021-05-26 05:38:28 -04:00
}
void MCP_DAC::shutDown()
{
_active = false;
transfer(0x0000); // a write will reset the values..
}
2021-07-31 11:44:31 -04:00
void MCP_DAC::setSPIspeed(uint32_t speed)
{
_SPIspeed = speed;
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
};
//////////////////////////////////////////////////////////////////
2021-11-08 10:16:58 -05:00
//
// PROTECTED
//
2021-05-26 05:38:28 -04:00
void MCP_DAC::transfer(uint16_t data)
{
2021-12-21 10:10:14 -05:00
// DATA TRANSFER
2021-05-26 05:38:28 -04:00
digitalWrite(_select, LOW);
if (_hwSPI)
{
2021-07-31 11:44:31 -04:00
// mySPI->beginTransaction(SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0));
mySPI->beginTransaction(_spi_settings);
mySPI->transfer((uint8_t)(data >> 8));
mySPI->transfer((uint8_t)(data & 0xFF));
mySPI->endTransaction();
2021-05-26 05:38:28 -04:00
}
else // Software SPI
{
swSPI_transfer((uint8_t)(data >> 8));
swSPI_transfer((uint8_t)(data & 0xFF));
}
digitalWrite(_select, HIGH);
}
// MSBFIRST
2021-07-31 11:44:31 -04:00
uint8_t MCP_DAC::swSPI_transfer(uint8_t val)
2021-05-26 05:38:28 -04:00
{
2021-07-31 11:44:31 -04:00
uint8_t clk = _clock;
uint8_t dao = _dataOut;
2021-05-26 05:38:28 -04:00
for (uint8_t mask = 0x80; mask; mask >>= 1)
{
2021-07-31 11:44:31 -04:00
digitalWrite(dao, (val & mask));
digitalWrite(clk, HIGH);
digitalWrite(clk, LOW);
2021-05-26 05:38:28 -04:00
}
return 0;
}
/////////////////////////////////////////////////////////////////////////////
//
// MCP4800 series
2021-12-21 10:10:14 -05:00
//
2021-05-26 05:38:28 -04:00
MCP4801::MCP4801(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 1;
_maxValue = 255;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4802::MCP4802(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 2;
_maxValue = 255;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4811::MCP4811(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 1;
_maxValue = 1023;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4812::MCP4812(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 2;
_maxValue = 1023;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4821::MCP4821(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 1;
_maxValue = 4095;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4822::MCP4822(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 2;
_maxValue = 4095;
};
/////////////////////////////////////////////////////////////////////////////
//
// MCP4900 series
2021-12-21 10:10:14 -05:00
//
2021-05-26 05:38:28 -04:00
MCP4901::MCP4901(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 1;
_maxValue = 255;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4902::MCP4902(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 2;
_maxValue = 255;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4911::MCP4911(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 1;
_maxValue = 1023;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4912::MCP4912(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 2;
_maxValue = 1023;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4921::MCP4921(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 1;
_maxValue = 4095;
};
2021-12-21 10:10:14 -05:00
2021-05-26 05:38:28 -04:00
MCP4922::MCP4922(uint8_t dataOut, uint8_t clock) : MCP_DAC(dataOut, clock)
{
_channels = 2;
_maxValue = 4095;
};
// -- END OF FILE --