2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
[![Arduino CI](https://github.com/RobTillaart/FunctionGenerator/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
|
2021-11-02 11:27:01 +01:00
|
|
|
[![Arduino-lint](https://github.com/RobTillaart/FunctionGenerator/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/FunctionGenerator/actions/workflows/arduino-lint.yml)
|
|
|
|
[![JSON check](https://github.com/RobTillaart/FunctionGenerator/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/FunctionGenerator/actions/workflows/jsoncheck.yml)
|
2021-01-29 12:31:58 +01:00
|
|
|
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/FunctionGenerator/blob/master/LICENSE)
|
|
|
|
[![GitHub release](https://img.shields.io/github/release/RobTillaart/FunctionGenerator.svg?maxAge=3600)](https://github.com/RobTillaart/FunctionGenerator/releases)
|
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
|
2020-11-27 11:16:22 +01:00
|
|
|
# FunctionGenerator
|
|
|
|
|
2021-12-18 16:52:23 +01:00
|
|
|
Arduino library to generate (numeric) wave forms for a DAC.
|
2021-11-02 11:27:01 +01:00
|
|
|
|
2020-11-27 11:16:22 +01:00
|
|
|
|
|
|
|
## Description
|
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
This library presents a class for a function generator in software.
|
|
|
|
It is typical used to control one or more DAC's.
|
|
|
|
To maximize signal quality one has to apply all (or most) processor power
|
|
|
|
to calculate new values over and over again to get enough resolution.
|
|
|
|
In practice the generator is useful for low frequencies,
|
|
|
|
0.01 - 25 Hz, depending on waveform and processor and number of DAC's.
|
|
|
|
(see indication below).
|
|
|
|
|
|
|
|
Note: this class generates float values, performance wise this can be optimized,
|
|
|
|
to achieve higher speeds at cost of accuracy / precision.
|
|
|
|
|
|
|
|
|
|
|
|
## Performance
|
|
|
|
|
|
|
|
Indication of what performance can be expected (based upon 0.2.1 version).
|
|
|
|
Note that the values need to be transported to a DAC or serial port too.
|
|
|
|
Numbers based on performance example, for one single signal.
|
|
|
|
|
2021-12-18 16:52:23 +01:00
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
| Processor | Clock | Waveform | usec/calls | max freq |
|
|
|
|
|:-------------|---------:|:---------|-----------:|---------:|
|
2021-12-18 16:52:23 +01:00
|
|
|
| Arduino UNO | 16 MHz | sawtooth | 62 | 60 Hz |
|
|
|
|
| Arduino UNO | 16 MHz | triangle | 74 | 50 Hz |
|
2021-11-02 11:27:01 +01:00
|
|
|
| Arduino UNO | 16 MHz | square | 53 | 1000 Hz |
|
2021-12-18 16:52:23 +01:00
|
|
|
| Arduino UNO | 16 MHz | sinus | 164 | 25 Hz |
|
|
|
|
| Arduino UNO | 16 MHz | stair | 81 | 50 Hz |
|
2021-11-02 11:27:01 +01:00
|
|
|
| Arduino UNO | 16 MHz | random | 37 | 1000 Hz |
|
|
|
|
| | | | | |
|
|
|
|
| ESP32 | 240 MHz | sawtooth | 3.8 | 1000 Hz |
|
|
|
|
| ESP32 | 240 MHz | triangle | 3.9 | 1000 Hz |
|
|
|
|
| ESP32 | 240 MHz | square | 2.8 | 1000 Hz |
|
2021-12-18 16:52:23 +01:00
|
|
|
| ESP32 | 240 MHz | sinus | 13.6 | 250 Hz |
|
|
|
|
| ESP32 | 240 MHz | stair | 4.8 | 800 Hz |
|
2021-11-02 11:27:01 +01:00
|
|
|
| ESP32 | 240 MHz | random | 1.3 | 1000 Hz |
|
|
|
|
|
|
|
|
|
|
|
|
- assumption minimal 250 samples per period to get a smooth signal.
|
|
|
|
if a rougher signal is OK, higher frequencies are possible.
|
|
|
|
- ESP32 can do more calculations however 1000 Hz seems to be a nice
|
|
|
|
upper limit for a software based function generator.
|
2021-12-18 16:52:23 +01:00
|
|
|
- if one wants to control multiple DAC's one need to divide the numbers
|
|
|
|
and round down.
|
2021-11-02 11:27:01 +01:00
|
|
|
|
|
|
|
|
|
|
|
Note: hardware function generator https://github.com/RobTillaart/AD985X
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
## Interface
|
|
|
|
|
|
|
|
### Constructor
|
|
|
|
|
|
|
|
- **funcgen(float period = 1.0, float amplitude = 1.0, float phase = 0.0, float yShift = 0.0)**
|
2020-11-27 11:16:22 +01:00
|
|
|
All parameters can be set in the constructor but also later in configuration.
|
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
### Configuration
|
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
- **void setPeriod(float period = 1.0)** set the period of the wave in seconds.
|
|
|
|
- **float getPeriod()** returns the set period.
|
|
|
|
- **void setFrequency(float frequency = 1.0)** set the frequency of the wave in Hertz (1/s).
|
2021-12-18 16:52:23 +01:00
|
|
|
- **float getFrequency()** returns the set frequency in Hertz.
|
2021-11-02 11:27:01 +01:00
|
|
|
- **void setAmplitude(float amplitude = 1.0)** sets the amplitude of the wave. TODO point to point?
|
|
|
|
Setting the amplitude to 0 gives ?what?
|
|
|
|
- **float getAmplitude()** returns the set amplitude.
|
|
|
|
- **void setPhase(float phase = 0.0)** shifts the phase of the wave. Will only be noticeable when
|
|
|
|
compared with other waves.
|
|
|
|
- **float getPhase()** returns the set phase.
|
|
|
|
- **void setYShift(float yShift = 0.0)** sets an Y-shift in amplitude, allows to set some zero point.
|
|
|
|
- **float getYShift()** returns the set Y-shift.
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
|
|
|
### Wave forms
|
|
|
|
|
2020-11-27 11:16:22 +01:00
|
|
|
t is time in seconds
|
2021-11-02 11:27:01 +01:00
|
|
|
|
|
|
|
- **float sawtooth(float t, uint8_t mode = 0)** mode == 0 (default) ==> sawtooth /|. mode == 1 ==> sawtooth |\.
|
|
|
|
- **float triangle(float t)** triangle form, duty cycle 50%.
|
|
|
|
- **float square(float t)** square wave with duty cycle 50%.
|
|
|
|
- **float sinus(float t)** sinus wave, has no duty cycle.
|
|
|
|
- **float stair(float t, uint16_t steps = 8, uint8_t mode = 0)** mode = 0 ==> steps up, mode = 1 steps down.
|
|
|
|
- **float random()** noise generation.
|
2021-12-18 16:52:23 +01:00
|
|
|
- **float line()** constant voltage line. Depends on the set YShift and amplitude.
|
2021-11-02 11:27:01 +01:00
|
|
|
- **float zero()** constant zero.
|
|
|
|
|
|
|
|
Line() and zero() are functions that can be used to drive a constant voltage from a DAC
|
|
|
|
and can be used to calibrate the generator / DAC combination.
|
|
|
|
|
2020-11-27 11:16:22 +01:00
|
|
|
|
|
|
|
## Operational
|
|
|
|
|
|
|
|
See examples.
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
## Future
|
|
|
|
|
|
|
|
|
|
|
|
#### waveforms
|
|
|
|
|
|
|
|
- square duty-cycle (will be slower!)
|
|
|
|
- triangle duty-cycle (makes sawtooth a special triangle)
|
|
|
|
- trapezium wave (would merge square and triangle and sawtooth)
|
2021-12-18 16:52:23 +01:00
|
|
|
- Bezier curve?
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2021-11-02 11:27:01 +01:00
|
|
|
#### investigate
|
|
|
|
|
|
|
|
- duty-cycle for sinus what does it mean. move peak.
|
|
|
|
- white noise, pink noise etc.
|
|
|
|
- investigate algorithms for performance gains (DAC specific values 10-12-16 bit)
|
|
|
|
- Amplitude modulation ?
|
|
|
|
- external clock to synchronize two or more sw function generators.
|
|
|
|
- RC function curve.
|
|
|
|
- heartbeat curve?
|
|
|
|
- record a signal and play back
|
|
|
|
|
|
|
|
#### misc
|
|
|
|
|
|
|
|
- stand-alone functions in separate .h?
|
|
|
|
- mapping to voltage function.
|
|
|
|
- check for synergy with https://github.com/RobTillaart/AD985X
|
|
|
|
|
|
|
|
#### examples
|
|
|
|
|
|
|
|
- example ESP32 version as separate task...
|
2021-01-29 12:31:58 +01:00
|
|
|
- example with DAC. 8 12 16 bit.
|
2021-11-02 11:27:01 +01:00
|
|
|
- example with potentiometers for 4 parameters
|
|
|
|
|
2021-01-29 12:31:58 +01:00
|
|
|
|
2016-12-17 20:57:37 +01:00
|
|
|
|