ADC -> DSP -> DAC Example
Written by Ross Wolin - last updated 2014.01.19
This project creates a chain which:
Build Environment: This project was built on Ubuntu Linux 12.04 LTS, compiled with GNU Tools for ARM Embedded Processors 4.8-2013q4, and programmed into an STM32F4 Discovery Kit using the 2014.01.18 commit of STLINK. If you need to install and set up the toolchain, refer to my previous blog posting: STM32 Discovery Development on Linux
The project files are hosted on Github, you can clone the repository with:
cd ~ git clone https://github.com/rowol/adc_dsp_dac
The ADC_DSP_DAC example source is now in ~/adc_dsp_dac/project
I have also included a STM's library source, needed to build the project, in the ~/adc_dsp_dac/STM32F4-Discovery_FW_V1.1.0 directory. If you want the complete STM library, utility, and examples source code, download that from STM's website
If you don't wish to clone the repository with Git, you may also retreive the project
as a zipfile.
Build the code by running:
Program the STM32F4 Discovery with STLINK by running:
The STM32F4 can do all sorts of fancy things with the ADCs, like making the ADCs run continuously and store the results with DMA, generating an interrupt when a conversion is complete, etc. In our realtime DSP application we would like to process each ADC sample when it's available, so we disable the ADC continuous conversions, DMA, and interrupts, and run the ADC in single-shot mode.
In the adc.c::ADC_init() function ADC1 channel 15 is set up in single-shot mode with all DMA and interrupt features disabled. The ADC does a conversion when you tell it do by calling ADC_SoftwareStartConv() and sets a flag, which you can monitor with ADC_GetSoftwareStartConvStatus() to determine when the conversion is complete.
I'm running the ADC with a divided by 8 clock to reduce the 84Mhz APB2 clock down to 10.5Mhz so it falls into the 600Khz-30Mhz range allowed for the ADC clock in the STM32F4 datasheet. I have also set the sampling time (how long the ADC "measures/averages" the signal before starting a conversion) to 144 cycles. Assuming the ADC takes about 16 cycles for a conversion, this means sampling+conversion will take about 160 cycles or 15.2uS with our 10.5Mhz ADC clock. Our timer interrupt generated sampling rate of 44Khz gives us 22.7uS between samples, so there is plenty of time for the ADC conversion to complete between each sample. The 44Khz sampling rate was chosen because the Nyguist-Shannon sampling theorem states you want to sample at twice the highest frequency in your input signal: audio CDs are sampled at approximately 44Khz.
DAC2 is also configured with DMA disabled. When you write to the DAC with dac.c::DAC2_set(), it changes the DAC2 value using the DAC_SetChannel2Data() library function.
In the tmr_sample module, the tmr_sample::tmr_sample_init() function configures basic timer TIM6 to generate interrupts at a rate of 44Khz, meaning main::TIM6_DAC_IRQHandler() is called approximately every 22.7uS. This IRQ handler does a couple of things:
My example simple digital filter is a notch filter with the peak centered at 880Hz. The graph shows the response from 0Hz to 5500Hz, with unity gain at the top of the peak. I designed this filter with FIVIEW, using the command "fiview 44000 -i BpRe/50/880"
IO Pin Summary
By connecting PC.4 to PC.5 with a jumper, you can run an 880Hz square wave into the ADC, through the digital filter, out the DAC, and observe the results with an oscillscope at PA.5. Square waves are composed of a sequence of sine waves of the fundamental plus odd harmonics: running a square wave through a good notch filter centered around the fundamental frequency will yield a sine wave.
These events happen in our project with PC.4 jumpered to PC.5:
The STM32F4 Discovery is a $15 development board, featuring a 168Mhz ARM Cortex M4 (STM32F407VGT6) The ARM is programmed via an STLINK/V2 interface connected to a PC's USB port (i.e. requires no JTAG plug, ICD, BDM, proprietary dongle, etc.)
Send comments, questions, money in large denominations, etc to eng at mysticengineering.com