427 lines
8.0 KiB
C++
Raw Normal View History

2021-01-29 12:31:58 +01:00
//
// FILE: AD985X.cpp
// AUTHOR: Rob Tillaart
2023-10-16 15:40:00 +02:00
// VERSION: 0.3.6
2021-01-29 12:31:58 +01:00
// DATE: 2019-02-08
// PURPOSE: Class for AD9850 and AD9851 function generator
2022-10-25 19:47:34 +02:00
// URL: https://github.com/RobTillaart/AD985X
2021-01-29 12:31:58 +01:00
#include "AD985X.h"
2022-10-25 19:47:34 +02:00
// UNO HARDWARE SPI PINS
2023-10-16 15:40:00 +02:00
#define SPI_CLOCK 13 // not portable.
2021-01-29 12:31:58 +01:00
#define SPI_MISO 12
#define SPI_MOSI 11
#define AD985X_POWERDOWN 0x04
////////////////////////////////////////////////////////
//
2022-10-25 19:47:34 +02:00
// AD9850
2021-01-29 12:31:58 +01:00
//
AD9850::AD9850()
{
}
2021-06-06 19:57:20 +02:00
void AD9850::begin(uint8_t select, uint8_t resetPin, uint8_t FQUDPin, uint8_t dataOut , uint8_t clock)
2021-01-29 12:31:58 +01:00
{
2021-08-25 09:52:59 +02:00
_select = select;
_reset = resetPin;
_fqud = FQUDPin;
_dataOut = dataOut;
_clock = clock;
2022-10-25 19:47:34 +02:00
// following 3 are always set.
2021-01-29 12:31:58 +01:00
pinMode(_select, OUTPUT);
pinMode(_reset, OUTPUT);
pinMode(_fqud, OUTPUT);
2022-10-25 19:47:34 +02:00
// device select = HIGH See - https://github.com/RobTillaart/AD985X/issues/13
2023-01-11 16:49:14 +01:00
digitalWrite(_select, LOW);
2021-01-29 12:31:58 +01:00
digitalWrite(_reset, LOW);
digitalWrite(_fqud, LOW);
2021-08-25 09:52:59 +02:00
_hwSPI = ((dataOut == 0) || (clock == 0));
_spi_settings = SPISettings(2000000, LSBFIRST, SPI_MODE0);
if (_hwSPI)
2021-01-29 12:31:58 +01:00
{
2021-08-25 09:52:59 +02:00
#if defined(ESP32)
2022-10-25 19:47:34 +02:00
if (_useHSPI) // HSPI
2021-08-25 09:52:59 +02:00
{
mySPI = new SPIClass(HSPI);
mySPI->end();
2023-10-16 15:40:00 +02:00
mySPI->begin(14, 12, 13, select); // CLK = 14 MISO = 12 MOSI = 13
2021-08-25 09:52:59 +02:00
}
2022-10-25 19:47:34 +02:00
else // VSPI
2021-08-25 09:52:59 +02:00
{
mySPI = new SPIClass(VSPI);
mySPI->end();
2023-10-16 15:40:00 +02:00
mySPI->begin(18, 19, 23, select); // CLK = 18 MISO = 19 MOSI = 23
2021-08-25 09:52:59 +02:00
}
2022-10-25 19:47:34 +02:00
#else // generic hardware SPI
2021-08-25 09:52:59 +02:00
mySPI = &SPI;
mySPI->end();
mySPI->begin();
#endif
2021-01-29 12:31:58 +01:00
}
2022-10-25 19:47:34 +02:00
else // software SPI
2021-01-29 12:31:58 +01:00
{
2021-08-25 09:52:59 +02:00
pinMode(_dataOut, OUTPUT);
pinMode(_clock, OUTPUT);
digitalWrite(_dataOut, LOW);
digitalWrite(_clock, LOW);
2021-01-29 12:31:58 +01:00
}
reset();
}
2021-08-25 09:52:59 +02:00
#if defined(ESP32)
void AD9850::setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)
{
_clock = clk;
_dataOut = mosi;
_select = select;
pinMode(_select, OUTPUT);
digitalWrite(_select, LOW);
2023-01-11 16:49:14 +01:00
mySPI->end(); // disable SPI
2021-08-25 09:52:59 +02:00
mySPI->begin(clk, miso, mosi, select);
}
#endif
2021-01-29 12:31:58 +01:00
void AD9850::reset()
{
2023-01-11 16:49:14 +01:00
// be sure to select the correct device
2021-08-25 09:52:59 +02:00
digitalWrite(_select, HIGH);
2021-01-29 12:31:58 +01:00
pulsePin(_reset);
2021-12-10 16:56:06 +01:00
if (_hwSPI)
{
#if defined(ESP32)
2022-10-25 19:47:34 +02:00
if (_useHSPI) pulsePin(14); // HSPI magic number clock
else pulsePin(18); // VSPI magic number clock
2023-01-11 16:49:14 +01:00
#else
2021-12-10 16:56:06 +01:00
// UNO hardware SPI
pulsePin(SPI_CLOCK);
#endif
}
2021-01-29 12:31:58 +01:00
else pulsePin(_clock);
2021-06-06 19:57:20 +02:00
digitalWrite(_select, LOW);
2021-01-29 12:31:58 +01:00
2022-10-25 19:47:34 +02:00
_config = 0; // 0 phase no power down
2021-01-29 12:31:58 +01:00
_freq = 0;
_factor = 0;
_offset = 0;
_autoUpdate = true;
writeData();
}
void AD9850::powerDown()
{
2022-10-25 19:47:34 +02:00
_config |= AD985X_POWERDOWN; // keep phase and REFCLK as is.
2021-01-29 12:31:58 +01:00
writeData();
}
void AD9850::powerUp()
{
_config &= ~AD985X_POWERDOWN;
writeData();
}
2023-01-11 16:49:14 +01:00
bool AD9850::setPhase(uint8_t phase)
2021-01-29 12:31:58 +01:00
{
2023-01-11 16:49:14 +01:00
if (phase > 31) return false;
2023-10-16 15:40:00 +02:00
_config &= 0x07; // keep last three bits
_config |= (phase << 3); // set new phase
2021-01-29 12:31:58 +01:00
writeData();
2023-01-11 16:49:14 +01:00
return true;
}
uint8_t AD9850::getPhase()
{
return (_config >> 3);
2021-01-29 12:31:58 +01:00
}
void AD9850::pulsePin(uint8_t pin)
{
digitalWrite(pin, HIGH);
digitalWrite(pin, LOW);
}
2021-08-25 09:52:59 +02:00
void AD9850::setSPIspeed(uint32_t speed)
{
_SPIspeed = speed;
_spi_settings = SPISettings(_SPIspeed, LSBFIRST, SPI_MODE0);
};
2021-01-29 12:31:58 +01:00
void AD9850::writeData()
{
2022-10-25 19:47:34 +02:00
// Serial.println(_factor, HEX);
// Serial.println(_config, HEX);
2021-01-29 12:31:58 +01:00
uint32_t data = _factor;
2023-10-16 15:40:00 +02:00
// used for multi device configuration only
// see https://github.com/RobTillaart/AD985X/issues/13
2023-01-11 16:49:14 +01:00
digitalWrite(_select, HIGH);
2021-08-25 09:52:59 +02:00
if (_hwSPI)
2021-01-29 12:31:58 +01:00
{
2021-08-25 09:52:59 +02:00
mySPI->beginTransaction(_spi_settings);
mySPI->transfer(data & 0xFF);
2021-01-29 12:31:58 +01:00
data >>= 8;
2021-08-25 09:52:59 +02:00
mySPI->transfer(data & 0xFF);
2021-01-29 12:31:58 +01:00
data >>= 8;
2021-08-25 09:52:59 +02:00
mySPI->transfer(data & 0xFF);
mySPI->transfer(data >> 8);
2022-10-25 19:47:34 +02:00
mySPI->transfer(_config & 0xFC); // mask factory test bit
2021-08-25 09:52:59 +02:00
mySPI->endTransaction();
2021-01-29 12:31:58 +01:00
}
else
{
swSPI_transfer(data & 0xFF);
data >>= 8;
swSPI_transfer(data & 0xFF);
data >>= 8;
swSPI_transfer(data & 0xFF);
swSPI_transfer(data >> 8);
2022-10-25 19:47:34 +02:00
swSPI_transfer(_config & 0xFC); // mask factory test bit
2021-01-29 12:31:58 +01:00
}
digitalWrite(_select, LOW);
2022-10-25 19:47:34 +02:00
// update frequency + phase + control bits.
// should at least be 4 ns delay - P14 datasheet
2021-01-29 12:31:58 +01:00
if (_autoUpdate) update();
}
2022-10-25 19:47:34 +02:00
// simple one mode version
2021-08-25 09:52:59 +02:00
void AD9850::swSPI_transfer(uint8_t val)
2021-01-29 12:31:58 +01:00
{
2021-08-25 09:52:59 +02:00
uint8_t clk = _clock;
uint8_t dao = _dataOut;
2022-10-25 19:47:34 +02:00
// for (uint8_t mask = 0x80; mask; mask >>= 1) // MSBFIRST
for (uint8_t mask = 0x01; mask; mask <<= 1) // LSBFIRST
2021-01-29 12:31:58 +01:00
{
2021-08-25 09:52:59 +02:00
digitalWrite(dao, (val & mask));
digitalWrite(clk, HIGH);
digitalWrite(clk, LOW);
2021-01-29 12:31:58 +01:00
}
}
2023-01-11 16:49:14 +01:00
bool AD9850::setFrequency(uint32_t freq)
2021-01-29 12:31:58 +01:00
{
2023-01-11 16:49:14 +01:00
bool rv = true;
2022-10-25 19:47:34 +02:00
// freq OUT = (Δ Phase × CLKIN)/2^32
// 64 bit math to keep precision to the max
2023-01-11 16:49:14 +01:00
if (freq > AD9850_MAX_FREQ)
{
rv = false;
freq = AD9850_MAX_FREQ;
}
2023-10-16 15:40:00 +02:00
// _factor = round(freq * 34.359738368); // 4294967296 / 125000000
2021-01-29 12:31:58 +01:00
_factor = (147573952589ULL * freq) >> 32;
_freq = freq;
_factor += _offset;
writeData();
2023-01-11 16:49:14 +01:00
return rv;
2021-01-29 12:31:58 +01:00
}
2022-10-25 19:47:34 +02:00
// especially for lower frequencies (with decimals)
2023-01-11 16:49:14 +01:00
bool AD9850::setFrequencyF(float freq)
2021-01-29 12:31:58 +01:00
{
2023-01-11 16:49:14 +01:00
bool rv = true;
2022-10-25 19:47:34 +02:00
// freq OUT = (Δ Phase × CLKIN)/2^32
// 64 bit math to keep precision to the max
2023-01-11 16:49:14 +01:00
if (freq > AD9850_MAX_FREQ)
{
rv = false;
freq = AD9850_MAX_FREQ;
}
2022-10-25 19:47:34 +02:00
_factor = round(freq * 34.359738368); // 4294967296 / 125000000
2021-01-29 12:31:58 +01:00
_freq = freq;
_factor += _offset;
writeData();
2023-01-11 16:49:14 +01:00
return rv;
}
float AD9850::getFrequency()
{
return _freq;
}
uint32_t AD9850::getMaxFrequency()
{
return AD9850_MAX_FREQ;
2021-01-29 12:31:58 +01:00
}
void AD9850::update()
{
2023-01-11 16:49:14 +01:00
digitalWrite(_select, HIGH);
2021-01-29 12:31:58 +01:00
pulsePin(_fqud);
2023-01-11 16:49:14 +01:00
digitalWrite(_select, LOW);
2021-01-29 12:31:58 +01:00
}
////////////////////////////////////////////////////////
//
2022-10-25 19:47:34 +02:00
// AD9851
2021-01-29 12:31:58 +01:00
//
2023-01-11 16:49:14 +01:00
// bit is a 6x multiplier bit P.14 datasheet
#define AD9851_REFCLK 0x01
2021-01-29 12:31:58 +01:00
2023-01-11 16:49:14 +01:00
bool AD9851::setFrequency(uint32_t freq)
2021-01-29 12:31:58 +01:00
{
2023-01-11 16:49:14 +01:00
bool rv = true;
2022-10-25 19:47:34 +02:00
// PREVENT OVERFLOW
2023-01-11 16:49:14 +01:00
if (freq > AD9851_MAX_FREQ)
{
rv = false;
freq = AD9851_MAX_FREQ;
}
2022-10-25 19:47:34 +02:00
// AUTO SWITCH REFERENCE FREQUENCY
2021-01-29 12:31:58 +01:00
if (_autoRefClock)
{
if (freq > _ARCCutOffFreq)
{
_config |= AD9851_REFCLK;
}
else
{
_config &= ~AD9851_REFCLK;
}
}
2022-10-25 19:47:34 +02:00
if (_config & AD9851_REFCLK) // 6x 30 = 180 MHz
2021-01-29 12:31:58 +01:00
{
_factor = (102481911520ULL * freq) >> 32; // (1 << 64) / 180000000
}
2022-10-25 19:47:34 +02:00
else // 1x 30 = 30 MHz
2021-01-29 12:31:58 +01:00
{
_factor = (614891469123ULL * freq) >> 32; // (1 << 64) / 30000000
}
_freq = freq;
_factor += _offset;
writeData();
2023-01-11 16:49:14 +01:00
return rv;
2021-01-29 12:31:58 +01:00
}
2022-10-25 19:47:34 +02:00
// especially for lower frequencies (with decimals)
2023-01-11 16:49:14 +01:00
bool AD9851::setFrequencyF(float freq)
2021-01-29 12:31:58 +01:00
{
2023-01-11 16:49:14 +01:00
bool rv = true;
2022-10-25 19:47:34 +02:00
// PREVENT OVERFLOW
2023-01-11 16:49:14 +01:00
if (freq > AD9851_MAX_FREQ)
{
rv = false;
freq = AD9851_MAX_FREQ;
}
2021-01-29 12:31:58 +01:00
2022-10-25 19:47:34 +02:00
// AUTO SWITCH REFERENCE FREQUENCY
2021-01-29 12:31:58 +01:00
if (_autoRefClock)
{
if (freq > _ARCCutOffFreq)
{
_config |= AD9851_REFCLK;
}
else
{
_config &= ~AD9851_REFCLK;
}
}
2022-10-25 19:47:34 +02:00
if (_config & AD9851_REFCLK) // 6x 30 = 180 MHz
2021-01-29 12:31:58 +01:00
{
_factor = uint64_t(102481911520ULL * freq) >> 32; // (1 << 64) / 180000000
}
2022-10-25 19:47:34 +02:00
else // 1x 30 = 30 MHz
2021-01-29 12:31:58 +01:00
{
_factor = (6148914691ULL * uint64_t (100 * freq)) >> 32;
}
_freq = freq;
_factor += _offset;
writeData();
2023-01-11 16:49:14 +01:00
return rv;
2021-01-29 12:31:58 +01:00
}
2023-01-11 16:49:14 +01:00
uint32_t AD9851::getMaxFrequency()
{
return AD9851_MAX_FREQ;
};
2021-01-29 12:31:58 +01:00
////////////////////////////////////////////////////////
//
2022-10-25 19:47:34 +02:00
// AD9851 - AUTO REFERENCE CLOCK
2021-01-29 12:31:58 +01:00
//
void AD9851::setAutoRefClock(bool arc)
{
_autoRefClock = arc;
setFrequency(_freq);
};
2023-01-11 16:49:14 +01:00
bool AD9851::getAutoRefClock()
{
return _autoRefClock;
};
2021-01-29 12:31:58 +01:00
void AD9851::setRefClockHigh()
{
_config |= AD9851_REFCLK;
setFrequency(_freq);
}
void AD9851::setRefClockLow()
{
_config &= ~AD9851_REFCLK;
setFrequency(_freq);
}
uint8_t AD9851::getRefClock()
{
return (_config & AD9851_REFCLK) ? 180 : 30;
}
2023-01-11 16:49:14 +01:00
2021-01-29 12:31:58 +01:00
void AD9851::setARCCutOffFreq(uint32_t Hz)
{
if (Hz > 30000000UL) Hz = 30000000;
_ARCCutOffFreq = Hz;
};
2023-01-11 16:49:14 +01:00
uint32_t AD9851::getARCCutOffFreq()
{
return _ARCCutOffFreq;
};
// -- END OF FILE --
2022-10-25 19:47:34 +02:00