This is an example application for the esp32-freqcount component.
It is written and tested for v3.3 of the ESP-IDF environment, using the xtensa-esp32-elf toolchain (gcc version 5.2.0).
Ensure that submodules are cloned:
$ git clone --recursive https://github.com/DavidAntliff/esp32-freqcount-example.git
Build the application with:
$ cd esp32-freqcount-example.git
$ idf.py menuconfig # set your serial configuration and IO configuration
$ idf.py -p (PORT) flash monitor
The program should monitor pulses on the configured input and display the measured frequency in the console.
This application makes use of the following components (included as submodules):
- components/esp32-freqcount
To run this demo application, you will need to supply a pulse-based signal to an ESP32 input pin. Use idf.py menuconfig
to set this GPIO.
In addition, you will need to allocate a GPIO for the internal RMT->PCNT window signal. This is a signal generated by the RMT peripheral to provide a signalling window to PCNT (pulse counter), during which PCNT will count both rising and falling edges of the signal of interest. Once the window has finished, the counter value is used to determine the average frequency over the entire sampling window.
An optional LED can be connected to a GPIO (configured by make menuconfig
) and this will be used to visually show the incoming frequency
signal. For fast signals this may result in the LED simply glowing rather than blinking.
In order to accurately measure a frequency signal of interest, it is important to consider the expected frequency range of the signal. The general rule is: the slower the expected frequency, the longer the sampling window needs to be to get an accurate measurement.
For example, a ~10 Hz input signal measured over 10 seconds will only be accurate to +/- 0.05 Hz. However a ~10,000 Hz signal measured over a 0.1 second window will be accurate to +/- 5 Hz.
In addition, the onboard peripheral that generates the sampling window (RMT) is constrained by the system clock and a programmable divider. The duration of the window is specified by an array of "items" that are processed in order. Each item describes the number of RMT periods (after the divider) to hold the window signal high or low. Describing RMT operation is outside of the scope of this document, however it is sufficient to know that:
- RMT programming consists of a block of instructions, called "items",
- Each block contains 64 items,
- There are up to 8 blocks of items,
- Each item can encode up to 2 * 32767 = 65534 periods of the window
Therefore, for a APB clock of 80 MHz and an RMT divider of 160, the RMT period will be 1 / (80 MHz / 160) = 1 / 500,000 seconds. A fully-employed item will therefore represent 65534 / 500,000 = 0.1311 seconds of window. So it follows that a 10 second sampling window will need 10.0 / 0.1311 = ~76 items. Since each block contains 64 items, this will require two blocks.
The user specifies the maximum number of blocks that this component is allowed to use. If the sampling window requirements exceed this maximum, an assertion will fire.
In addition to the sampling window, there is also a filter that can be configured to reduce glitches on the counter input. The length of this filter must be carefully specified otherwise it can mask genuine edges. The filter length is specified as the number of periods at APB clock rate (80 MHz) that changes in the PCNT input signal will be ignored. For example, a value of 1023 will limit square-wave inputs to a maxmum frequency of 80 MHz / 1023 / 2 (because there are two edges per cycle!) = 39,100 Hz.
The source is available from GitHub.
The code in this project is licensed under the MIT license - see LICENSE for details.
Thank you to Chris Morgan for converting the original demo application into a reusable IDF component, which is used by this example.