SignalGenerator-Teensy/README.md

4.5 KiB

Teensy 3.5 DDS Signal Generator

Using the integrated DAC of a Teensy 3.5 to generate arbitrary periodic waveforms
This Signal Generator has been done as a project for my final High School Exam (Emanuele Coletta, ITIS Galileo Galilei Roma, a.s. 2020-2021)

How it works

The Teensy 3.5 is a powerful Arduino-compatible microcontroller running an ARM Cortex-M4 at 120 MHz
For the purpose of this project, it has been overclocked to 168MHz, running with the Fastest Pure Code+LTO optimizer
It has two integrated DACs whose output can vary between 0v and 3.3V
Using classical techniques from DDS (Direct Digital Synthesis) the Teensy can become a fully functional function generator
In particular, the technique used here is the classic Phase Accumulator, together with a Lookup Table. This allows to precompute arbitrary periodic waveforms and then output them at the wanted frequency.
Square waves of variable duty cycles can be generated by simply toggling a digital pin high and low and the correct time. Despite this, it's been chosen to use the DAC1 pin to generate square waves, which opens the possibility of using both DACs for arbitrary wave generation. Despite both sine and square wave being output by the teensy are quite clean and seem to have low noise and spurs levels at naked eye, advanced filtering techniques, like Chebyshev, Butterworth or Bessel filters with multiple stages are required. The desired waveform, frequency and duty cycle can be selected using the I2C LCD Display and an incremental encoder with pushbutton

Why two different microcontrollers

Each and every clock cycle of the Teensy 3.5 is meticulously used to generate waveforms and have them to be precise in frequency over time. Therefore, no clock cycle can be wasted reading external input or driving the LCD Display. For this purpose, an AtMega32U4 handles the I2C LCD Display, the incremental encoder and the analog inputs from the output stage.

Amplification and offset

Amplification is done using an OpAmp inverting configuration. The non-inverting pin is wired to a voltage divider made with a potentiometer which varies between -12V and +12V to offset the wave. Using a peak detector the peak-to-peak voltage of the wave can be measured. An OpAmp in non-inverting summer configuration is used to sum the offset and the wave before the peak-detector: this allows to measure the peak no matter the offset condition. Furthermore, the use of opamps provides impeadence decoupling, isolating the measuring circuits from the output. The peak-to-peak and offset voltage of the wave are then remapped in the range 0-5V using an inverting OpAmp with a ratio of 1/5 whose non-inverting input is fixed at 2.5V. Those two signals are then fed back to an analogic input of the AtMega 32U4 to be shown on the LCD Display

Filters

Filters have been generated using The Filter Wizard by Analog Devices.
A 1st order RC passive low-pass filter has been choosen to smooth out the transitory of the square wave. Futhermore, this allows the use of DAC1 for arbitrary wave generation, even different from a square wave.
A 6th order Butterworth low-pass filter has been choosen to smooth out the wave generated by the DAC.

alt text

Credits

A lot of choices come from following this Degree dissertation paper (Direct Digital Synthesizers: Theory, Design and Applications by Jouko Vankka, Helsinki University of Technology, November 2000): http://lib.tkk.fi/Diss/2000/isbn9512253186/isbn9512253186.pdf

Analog Devices offers a lot of cool guides and papers about DDS, this is one of them: https://www.analog.com/media/en/training-seminars/design-handbooks/Technical-Tutorial-DDS/Section8.pdf

The website ZipCPU offers a variety of articles about sine wave generation using FPGAs


Stackoverflow and the rest of the internet of course