311 lines
10 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/AD985X/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-08-25 09:52:59 +02:00
[![JSON check](https://github.com/RobTillaart/AD985X/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/AD985X/actions/workflows/jsoncheck.ym)l
[![Arduino-lint](https://github.com/RobTillaart/AD985X/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/AD985X/actions/workflows/arduino-lint.yml)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/AD985X/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/AD985X.svg?maxAge=3600)](https://github.com/RobTillaart/AD985X/releases)
2021-08-25 09:52:59 +02:00
2021-01-29 12:31:58 +01:00
# AD985X
Arduino library for AD9850 and AD9851 function generators.
## Description
Library for the AD9850 and AD9851 function generators.
2022-10-25 19:47:34 +02:00
These devices can produce a square and a sine wave
| type | max frequency | phase (stepsize) |
|:---------|---------------:|-------------------:|
| AD9850 | 40 MHz | 0..31 x 11.25<EFBFBD> |
| AD9851 | 70 MHz | 0..31 x 11.25<EFBFBD> |
Note that at the max frequency the devices do not give a nice sine anymore.
You need to check what is acceptable for your project.
2021-08-25 09:52:59 +02:00
The library has a AD9850 as base class that implements the commonalities.
2021-01-29 12:31:58 +01:00
The AD9851 is derived and has its own **setFrequency()** methods.
Furthermore the AD9851 also has function to select the reference clock,
2022-10-25 19:47:34 +02:00
a feature the AD9850 does not have.
This feature improves the tuning for both low and high frequencies.
2021-01-29 12:31:58 +01:00
**Warning**
The library is not suitable for AD9852 as that is a function generator with
way more functionality.
2021-12-10 16:56:06 +01:00
Note: mainly tested on Arduino UNO. Tweaking for other platforms is expected.
2021-01-29 12:31:58 +01:00
## Connection
2022-10-25 19:47:34 +02:00
Schema break-out
2021-01-29 12:31:58 +01:00
```
TOP VIEW
+-----------+
| X-TAL |
| L |
VCC | o o | VCC
CLK | o o | D0
PUFD | o o | D1
DATA | o o | D2
RESET | o o | D3
GND | o CCC o | D4
QOUT1 | o CCC o | D5
QOUT2 | o o | D6
ZOUT1 | o o | D7 ----- SELECT SERIAL LOW
ZOUT2 | o PP o | GND
| PP |
+-----------+
XTAL = crystal
L = LED
C = chip
2021-12-10 16:56:06 +01:00
P = potentiometer => for duty cycle square wave
2021-01-29 12:31:58 +01:00
```
2022-10-25 19:47:34 +02:00
2021-12-10 16:56:06 +01:00
### Multi device
2021-01-29 12:31:58 +01:00
2022-10-25 19:47:34 +02:00
See **Multi_AD985X_devices.pdf**
2021-01-29 12:31:58 +01:00
2021-06-06 19:57:20 +02:00
Discussion leading to the document see - https://github.com/RobTillaart/AD985X/issues/13
2021-01-29 12:31:58 +01:00
The AD985X board can be connected with a SPI bus like interface.
However there is **no Chip Select pin (CS)** so one must take other measures to control multiple AD985X devices.
#### Trivial solution
The trivial implementation is to give each device a set of unique pins.
If you have pins to spare this is the perfect solution.
#### Shared line solution
A more common SPI solution is to share the data and clock lines.
However that would typical set all AD985X devices simultaneously.
So extra hardware is needed to prevent this.
A possible solution is to put all needed lines behind an AND port that allows only
communication when the **SELECT** is HIGH.
2021-06-06 19:57:20 +02:00
```
Arduino AND AD985X
--------------------------------------------------
+--------+
SELECT ----| A |
| Y |------- DATA
DATA -----| B |
+--------+
+--------+
SELECT ----| A |
| Y |------- CLOCK
CLOCK ----| B |
+--------+
+--------+
SELECT ----| A |
| Y |------- FQ_UD
FQ_UD ----| B |
+--------+
+--------+
SELECT ----| A |
| Y |------- RESET
RESET ----| B |
+--------+
```
2021-01-29 12:31:58 +01:00
The **DATA** line of the device is connected to the output of an **AND** port.
The inputs if the **AND** port are (a) the SPI bus **DATA** line and (b) the **SELECT** pin.
Strictly for the **DATA** this is not needed as data will only clock in if there is a **CLOCK**.
The **CLOCK** pin of the device is connected to the output of an **AND** port.
The inputs if the **AND** port are (a) the SPI bus **CLOCK** line and (b) the **SELECT** pin.
The **FQ_UD** pin of the device is connected to the output of an **AND** port.
The inputs if the **AND** port are (a) the MCU **FQ_UD** line and (b) the **SELECT** pin.
See FQ_UD note below.
The **RESET** pin of the device is connected to the output of an **AND** port.
The inputs if the **AND** port are (a) the MCU **RESET** line and (b) the **SELECT** pin.
A typical IC to use is the **74HC08** which has 4 AND ports in it.
In short this setup makes the lines 'switchable' pass through, with the **SELECT** line.
It allows to have multiple AD985X devices, and even to share the SPI bus **DATA** and **CLOCK**
lines with other SPI devices.
### FQ_UD note
It might be possible to connect a single FQ_UD line to multiple AD985X devices directly.
The FQ_UD pulse would update the frequency and as this register is not changed, the FQ_UD
pulse might just have no changing effect. To be investigated to confirm this.
If confirmed this would change the above Shared line solution a bit.
If the FQ_UD line can be shared directly it offers a way to start / change multiple
devices at the same time.
## Interface
### Constructors
- **AD9850()** 40 MHz signal generator
- **AD9851()** 70 MHz signal generator, derived from AD9850 with some extra options.
### Common interface
2021-12-10 16:56:06 +01:00
- **void begin(uint8_t selectPin, uint8_t resetPin, uint8_t FQUDPin, uint8_t dataOut = 0, uint8_t clock = 0)**
2021-01-29 12:31:58 +01:00
For hardware SPI only use the first three parameters,
for SW SPI you need to define the data and clock pin too.
- selectPin = chip select. The library uses HIGH as active and LOW as not selected.
- resetPin = reset
- FQUD = Frequency UpDate Pin
2021-08-25 09:52:59 +02:00
- **void reset()** resets the function generator.
- **void powerDown()** idem
- **void powerUp()** idem
- **void setFrequency(uint32_t freq)** SetFrequency sets the frequency and is limited by the
2021-01-29 12:31:58 +01:00
MaxFrequency of the class used. For the AD9850 => 40 MHz, for the AD9851 => 70 MHz.
- Note that the quality of the signal gets less at higher frequencies.
- Note setFrequency is affected by the autoUpdateFlag.
2021-08-25 09:52:59 +02:00
- **void setFrequencyF(float freq)** SetFrequencyF sets the frequency with a float with a maximum of **two** decimals.
- Note that a float only has a mantissa of 6-7 digits so for frequencies above above ~1.000.000 = 1MHz all decimals are lost.
2021-01-29 12:31:58 +01:00
- Note setFrequencyF is affected by the autoUpdateFlag.
2021-08-25 09:52:59 +02:00
- **uint32_t getMaxFrequency()** returns the maximum frequency that can be set. For the AD9850 this is 20 MHz.
2021-01-29 12:31:58 +01:00
For the AD9851 this is 70 MHz.
2021-08-25 09:52:59 +02:00
- **float getFrequency()** returns the frequency set. As it returns a float it might loose some accuracy at higher frequencies.
- **void setPhase(uint8_t phase = 0)** set the phase in units of 11.25<EFBFBD> 0..31 allowed.
2021-01-29 12:31:58 +01:00
Default it sets the phase to 0.
2021-08-25 09:52:59 +02:00
- **uint8_t getPhase()** returns the phase set, 0 by default. One need to multiply by 11.25<EFBFBD> to get the actual angle.
2021-01-29 12:31:58 +01:00
### Calibration
**Warning:** use with care.
- **void setCalibrationOffset(int32_t offset = 0)** sets an offset to calibrate the frequency.
- **uint32_t getCalibrationOffset()** reads back the offset set.
- **uint32_t getFactor()** internal factor, for debugging
2021-12-10 16:56:06 +01:00
Note: **reset()** resets the offset to 0..
2021-01-29 12:31:58 +01:00
Note: setting the offset reduces the range of frequencies (at the ends of scale).
### Auto update / manual update
(new since 0.2.2)
**Warning:** use with care.
2021-08-25 09:52:59 +02:00
- **void setAutoUpdate(bool update)** sets the autoUpdate flag, default set to true.
2021-01-29 12:31:58 +01:00
- **bool getAutoUpdate()** reads the autoUpdate flag.
- **void update()** manually toggle the FQ_UD flag to update the frequency.
Manual updating allows one to prepare the frequency, and actually apply
it at a later moment.
2021-12-10 16:56:06 +01:00
Note: The default of the autoUpdate flag is true.
Note: **reset()** resets the autoUpdateFlag to true.
2021-01-29 12:31:58 +01:00
2021-08-25 09:52:59 +02:00
### Hardware SPI
To be used only if one needs a specific speed.
2021-12-10 16:56:06 +01:00
- **void setSPIspeed(uint32_t speed)** set SPI transfer rate.
- **uint32_t getSPIspeed()** returns SPI transfer rate.
2021-08-25 09:52:59 +02:00
- **bool usesHWSPI()** returns true if HW SPI is used.
### ESP32 specific
This functionality is new in 0.3.1.
- **void selectHSPI()** in case hardware SPI, the ESP32 has two options HSPI and VSPI.
- **void selectVSPI()** see above.
- **bool usesHSPI()** returns true if HSPI is used.
- **bool usesVSPI()** returns true if VSPI is used.
The **selectVSPI()** or the **selectHSPI()** needs to be called
BEFORE the **begin()** function.
#### ESP32 experimental
2021-12-10 16:56:06 +01:00
- **void setGPIOpins(clk, miso, mosi, select)** overrule GPIO pins of ESP32 for hardware SPI.
Needs to be called AFTER the **begin()** function.
2021-08-25 09:52:59 +02:00
```cpp
void setup()
{
freqGen.selectVSPI();
freqGen.begin(15);
freqGen.setGPIOpins(CLK, MISO, MOSI, SELECT); // SELECT should match the param of begin()
...
}
```
2021-01-29 12:31:58 +01:00
### AD9850 specific
The AD9850 has no specific functions.
2021-08-25 09:52:59 +02:00
2021-01-29 12:31:58 +01:00
### AD9851 specific
2021-08-25 09:52:59 +02:00
- **void setRefClockHigh()** set reference clock to 180 Mhz.
- **void setRefClockLow()** set reference clock to 30 Mhz.
- **uint8_t getRefClock()** returns 30 or 180.
- **void setAutoRefClock(bool arc)** sets a flag so the library switches automatically
2021-01-29 12:31:58 +01:00
to the reference clock of 180 MHz when the frequency is set above 10 MHz and
to 30 MHz when the frequency is set to 10 MHz or lower.
The initial value is **false** == OFF for backwards compatibility.
2021-12-10 16:56:06 +01:00
- **bool getAutoRefClock()** returns true if autoRefClock is set.
2021-01-29 12:31:58 +01:00
- **void setARCCutOffFreq(uint32_t Hz = 10000000UL )** set cut off frequency
2021-12-10 16:56:06 +01:00
for the auto reference clock. Maximum value is 30 MHz, typical 10 MHz.
2021-01-29 12:31:58 +01:00
- **uint32_t getARCCutOffFreq()** returns cut off frequency set.
- Note: the autoRefClock mode does **NOT** automatically adjust the calibration offset.
- Note: **reset()** does **NOT** reset the autoRefClock flag.
## Operation
2021-12-10 16:56:06 +01:00
See examples.
2021-01-29 12:31:58 +01:00
### Operational notes
- The quality of the signal becomes less at higher frequencies.
2021-08-25 09:52:59 +02:00
Switch the reference clock to find your optimal quality.
2021-01-29 12:31:58 +01:00
- If the calibration offset is not 0, it needs to be set by the user after every startup,
and after switching the reference clock.
The user is also responsible to store it e.g. in EEPROM to make it persistent.
- Experimental parts may change or removed in the future.
2021-08-25 09:52:59 +02:00
## Future
- examples for ESP32 HWSPI interface
2021-12-10 16:56:06 +01:00
- do tests on ESP32
2021-08-25 09:52:59 +02:00
- performance measurements
2022-10-25 19:47:34 +02:00
- move code to .cpp
2021-08-25 09:52:59 +02:00