Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/arminjo/arduino-frequencydetector

Fast audio frequency detector without fft for plain Arduino and Attiny85. Whistle switch example included.
https://github.com/arminjo/arduino-frequencydetector

arduino arduino-library frequency-detection pitch-detection whistle-detector

Last synced: 3 months ago
JSON representation

Fast audio frequency detector without fft for plain Arduino and Attiny85. Whistle switch example included.

Awesome Lists containing this project

README

        

# [Frequency Detector](https://github.com/ArminJo/Arduino-FrequencyDetector) Library for Arduino and ATtinys

Detects frequency **from 38 Hz to 9612 Hz** and works even on an ATTiny85 with 1 MHz up to 4806 Hz.

The input signal can be plotted to the Arduino Serial Plotter resulting in a **simple Oscilloscope** to test the internal signal.

Only tested on ATtiny85 and ATmega328P.

[![Badge License: GPLv3](https://img.shields.io/badge/License-GPLv3-brightgreen.svg)](https://www.gnu.org/licenses/gpl-3.0)
   
[![Badge Version](https://img.shields.io/github/v/release/ArminJo/Arduino-FrequencyDetector?include_prereleases&color=yellow&logo=DocuSign&logoColor=white)](https://github.com/ArminJo/Arduino-FrequencyDetector/releases/latest)
   
[![Badge Commits since latest](https://img.shields.io/github/commits-since/ArminJo/Arduino-FrequencyDetector/latest?color=yellow)](https://github.com/ArminJo/Arduino-FrequencyDetector/commits/master)
   
[![Badge Build Status](https://github.com/ArminJo/Arduino-FrequencyDetector/workflows/LibraryBuildWithAction/badge.svg)](https://github.com/ArminJo/Arduino-FrequencyDetector/actions)
   
![Badge Hit Counter](https://visitor-badge.laobi.icu/badge?page_id=ArminJo_Arduino-FrequencyDetector)




[![Stand With Ukraine](https://raw.githubusercontent.com/vshymanskyy/StandWithUkraine/main/badges/StandWithUkraine.svg)](https://stand-with-ukraine.pp.ua)

Available as [Arduino library "Arduino-FrequencyDetector"](https://www.arduinolibraries.info/libraries/frequency-detector).

[![Button Install](https://img.shields.io/badge/Install-brightgreen?logoColor=white&logo=GitBook)](https://www.ardu-badge.com/Arduino-FrequencyDetector)
   
[![Button Changelog](https://img.shields.io/badge/Changelog-blue?logoColor=white&logo=AzureArtifacts)](https://github.com/ArminJo/Arduino-FrequencyDetector#revision-history)

#### If you find this library useful, please give it a star.

🌎 [Google Translate](https://translate.google.com/translate?sl=en&u=https://github.com/ArminJo/Arduino-FrequencyDetector)


YouTube video of whistle switch example in action.

[![Demonstration of 3 whistle switches in one room](https://i.ytimg.com/vi/_e2mElB8zJs/hqdefault.jpg)](https://www.youtube.com/watch?v=_e2mElB8zJs)


# Internal operation
This library analyzes a (microphone) signal and outputs the detected frequency. It simply counts zero crossings and **it does not use FFT**.
The ADC sample data is **not** stored in RAM, only the period lengths (between triggers) are stored in the `PeriodLength[]` array,
which is a byte array and has the size of `NUMBER_OF_SAMPLES / 8`.

It is like in the [Arduino Simple Audio Frequency Meter](https://www.arduino.cc/en/Tutorial/SimpleAudioFrequencyMeter) but includes additional noise check.

The **timer 0 interrupt**, which counts the milliseconds, **is disabled during reading** and enabled afterwards!
The value of millis() is adjusted after reading.

The alternative to disabling the interrupt is getting partially invalid results!

There are 3 detection ranges available:
- `FREQUENCY_RANGE_HIGH` -> 13 µs/sample -> 300 to **9612** Hz with 1024 samples and 600 to 9612 Hz with 512 samples.
- `FREQUENCY_RANGE_DEFAULT` -> 52 µs/sample -> **75 to 2403 Hz with 1024 samples** and 150 to 2403 Hz with 512 samples.
- `FREQUENCY_RANGE_LOW` -> 104 µs/sample -> **38** to 1202 Hz with 1024 samples and 75 to 1202 Hz with 512 samples.

### `readSignal()` is the ADC read routine, which reads 1024 samples (512 for ATtinies) and computes the following values:
1. Frequency of signal `uint16_t FrequencyRaw`
2. Amplitude = (MaxValue - MinValue) `uint16_t SignalDelta`
3. Average = (SumOfSampleValues / NumberOfSamples) `uint16_t AverageLevel`
4. The length of each period (between 2 trigger conditions) in the `PeriodLength[]` array.

### `doEqualDistributionPlausi()` checks if the signal in the `PeriodLength[]` array is valid / not noisy.
It checks if at maximum 1/8 of the periods are greater than 1.5 or less than 0.75 of the average period.
If not, the value of `FrequencyRaw` is overwritten with the error code `SIGNAL_DISTRIBUTION_PLAUSI_FAILED`.

### `computeDirectAndFilteredMatch()` waits for n matches within a given frequency range (FrequencyMatchLow - FrequencyMatchHigh)
and also low pass filters the result for smooth transitions between the 3 match states (lower, match, greater). It computes the following values:
1. Low pass filtered frequency of signal `uint16_t FrequencyFiltered;`
2. Match result `MatchStateEnum FrequencyMatchDirect;`
3. Low pass filtered match result `MatchStateEnum FrequencyMatchFiltered`


# Compile options / macros for this library
To customize the library to different requirements, there are some compile options / macros available.

These macros must be defined in your program **before** the line `#include "FrequencyDetector.hpp"` to take effect.

Modify them by enabling / disabling them, or change the values if applicable.

| Name | Default value | Description |
|-|-:|-|
| `PRINT_INPUT_SIGNAL_TO_PLOTTER` | disabled | Signal input data is stored and can be printed together with trigger levels using `printInputSignalValuesForArduinoPlotter()` like in the *SimpleFrequencyDetector* example to implement a simple digital oscilloscope using the Arduino 1.x Serial Plotter. |

### Arduino Plotter output of SimpleFrequencyDetector example with PRINT_INPUT_SIGNAL_TO_PLOTTER enabled
![Arduino Plotter output of SimpleFrequencyDetector example with `PRINT_INPUT_SIGNAL_TO_PLOTTER` enabled](https://github.com/ArminJo/Arduino-FrequencyDetector/blob/master/extras/SimpleFrequencyDetectorPlotterOutput.png)

### Arduino Plotter output of SimpleFrequencyDetector example with PRINT_RESULTS_TO_SERIAL_PLOTTER enabled
Above you see the `FrequencyRaw` I whistled with all the dropouts and the `FrequencyFiltered` without dropouts but with a slight delay if the `FrequencyRaw` changes.

The WhistleSwitch example uses the `FrequencyMatchFiltered` value, to decide if a match happens. The value for a valid match is **200**. At 80 and 140 you see 2 short and at 300 you see a long valid match.

![Arduino Plotter output of whistle switch in action](https://github.com/ArminJo/Arduino-FrequencyDetector/blob/master/extras/WhistleSwitchPlotterOutput.png)


# Dependencies
The [ATtinySerialOut library](https://github.com/ArminJo/ATtinySerialOut) is required for [print functions](src/FrequencyDetector.h#L312) on ATtinies.


# SimpleFrequencyDetector example
This example reads analog signal e.g. from MAX9814 Module at A1 and computes the frequency.
If frequency is in the range of 1400 to 1700 Hz, the Arduino built in LED will light up.
It prints the detected frequency as well as plausibility errors.
For frequency below 500 Hz it might be good to change `FREQUENCY_RANGE_DEFAULT` to `FREQUENCY_RANGE_LOW`.

By enabling `PRINT_INPUT_SIGNAL_TO_PLOTTER` you can convert the example to a simple DSO.

By enabling `PRINT_RESULTS_TO_SERIAL_PLOTTER` you can watch the [generated output](https://github.com/ArminJo/Arduino-FrequencyDetector#arduino-plotter-output-of-whistleswitch-in-action) of the library.

SimpleFrequencyDetector on breadboard with MAX9814 Module
![SimpleFrequencyDetector on breadboard with MAX9814 Module](https://github.com/ArminJo/Arduino-FrequencyDetector/blob/master/extras/SimpleFrequencyDetector_MAX9814.jpg)
YouTube Demonstration of SimpleFrequencyDetector with MAX9812 Module

[![Demonstration of SimpleFrequencyDetector](https://img.youtube.com/vi/tsxfSx0iY5s/0.jpg)](https://www.youtube.com/watch?v=tsxfSx0iY5s)


# WhistleSwitch example
The WhistleSwitch example analyzes a microphone signal (I use a MAX9814 module from Adafruit) and toggles an output pin, if the main frequency is for a specified duration in a specified range.
It works as a frequency detector for a whistle pitch which operates a mains relay. By using different pitches it is possible to control multiple relays in a single room.

If the pitch is lower than the specified frequency, the feedback LED blinks slowly, if the pitch is higher it blinks fast.

If the (low pass filtered) match from the FrequencyDetector library holds for `MATCH_TO_LONG_MILLIS` (1 second) after switching output,
the output switches again, to go back to the former state.
This can be useful if a machine generated signal (e.g. from a vacuum cleaner) matches the range.

**This example is mainly created to run on an ATtiny85 at 1 MHz, but will work also on a plain Arduino.**


## PREDEFINED RANGES
After power up or reset, the feedback LED echoes the range number. Range number 10 indicates an individual range, programmed by advanced selecting.
The timeout state is signaled by short LED pulses after the range number feedback (no short pulse -> no timeout enabled).

the following pitch ranges are predefined for easy selection:
1. 1700 - 2050 Hz -> 350 Hz A6-C6
2. 1500 - 1680 Hz -> 180 Hz FS6-GS6
3. 1300 - 1480 Hz -> 180 Hz E6-FS6
4. 1150 - 1280 Hz -> 130 Hz D6-DS6
5. 1000 - 1130 Hz -> 130 Hz C6-CS6
6. 900 - 990 Hz -> 90 Hz A5-B5

7. 1550 - 1900 Hz -> 350 Hz G6-AS6
8. 1250 - 1530 Hz -> 380 Hz DS6-G6
9. 1000 - 1230 Hz -> 230 Hz C6-DS6

10. Dummy range, if chosen, disables "relay on" timeout handling.
11. Dummy range, if chosen, sets "relay on" timeout to 2 hours.
12. Dummy range, if chosen, sets "relay on" timeout to 4 hours.
13. Dummy range, if chosen, sets "relay on" timeout to 8 hours.

## SELECTING the RANGE
Selecting is started by a long press of the button.
After `BUTTON_PUSH_ENTER_PROGRAM_SIMPLE_MILLIS` (1.5 seconds), the feedback LED blinks once for signaling simple programming mode.
After `BUTTON_PUSH_ENTER_PROGRAM_ADVANCED_MILLIS` (4 seconds), the feedback LED blinks twice for signaling advanced programming mode.
After releasing the button, the selected programming mode is entered.

### SIMPLE PROGRAMMING MODE
Press the button once for range 1, twice for range 2 etc. Each button press is echoed by the feedback LED.
Waiting for `PROGRAM_MODE_SIMPLE_END_DETECT_MILLIS` (1.5 seconds) ends the programming mode
and the feedback LED echoes the number of button presses recognized.
The duration of signal match to toggle the relay is fixed at `MATCH_MILLIS_NEEDED_DEFAULT` (1.2 seconds).

### ADVANCED PROGRAMMING MODE
Whistle the pitch you want to detect, then press the button again.
While you press the button, the pitch range is measured. i.e. the minimum and maximum of the pitch you are whistling is stored.

If you press the button again before the `PROGRAM_MODE_ADVANCED_END_DETECT_MILLIS` (3 seconds) timeout
the duration of this second press is taken as the required duration for the signal match to toggle the relay.
Otherwise the `MATCH_MILLIS_NEEDED_DEFAULT` (1.2 seconds) are taken.
After timeout of `PROGRAM_MODE_TIMEOUT_MILLIS` (5 seconds) the advanced programming mode is ended
and the effective duration is echoed by the feedback LED.

## TIMEOUT
After a timeout of `TIMEOUT_RELAY_ON_SIGNAL_MINUTES`_(1 to 3) (2, 4 or 8 hours) the relay goes OFF for 1 second.
In the next `TIMEOUT_RELAY_SIGNAL_TO_OFF_MINUTES` (3) minutes you must then press the button or whistle the pitch to cancel the timeout, otherwise the relay will switch OFF afterwards.
Cancellation of timeout is acknowledged by the LED blinking 5 times for 1 second on and off. Timeout can be switched on by selecting the dummy ranges 11 to 13 and off by selecting the dummy range 10.
The setting is stored in EEPROM. Default is `TIMEOUT_RELAY_ON_SIGNAL_MINUTES_3` (8 hours).

## RESET
A reset can be performed by power off/on or by pressing the button two times, each time shorter than `RESET_ENTER_BUTTON_PUSH_MILLIS` (0.12 seconds) within a `RESET_WAIT_TIMEOUT_MILLIS` (0.3 seconds) interval.


# SCHEMATIC for external components of FrequencyDetector / WhistleSwitch
```
Discrete microphone amplifier with LM308

+ 5V _____ o--O PIN REF
| o-|_____|--o |
_ | 1M | _
| | | | | |
| | 2k2 o---|\ | | | 1M
|_| | 2| \____| |_|
| _____ _____ | | /6 | ____ | | |
o---|_____|----o----|_____|-------|/ o--|____|--| |--o--O PIN A1
| 2k2 | 10k | 3 10k | | |
--- |O MICROPHONE _ LM308 10-100nF _
--- 1 uF | | | | |
| | | | 10k | | 1M
_|_ _|_ |_| |_|
| |
| |
--- _|_
--- 100 nF
|
_|_
```

```
External circuit for 1x amplification configuration on a Digispark board.

+ CPU 5V - * Schottky-diode
o------------------------------------ o-----|<|--o-- USB 5V
| | - |
_ | |
| | o / |
470k | | /=| Push button
|_| / |
1n | ____ ____ o----------o
>- | |--o--|____|--o--|____|--O PB4 550 mV |
500Hz | 3k3 | 10k to enable USB _
High _ | programming | |
Pass | | --- | | * 1k5 pullup
100k | | --- 22n 2kHz Low |_|
|_| | Pass |
| | _____ |
o----------o PB3 O---|_____|--o
| * 68/22 |
| __
| /\` * 3V6 Z-diode
| --
| | * = assembled USB circuit on Digispark
| |
_|_ _|_
```

```
External circuit for 20x amplification configuration on a Digispark board.

+ CPU 5V - * Schottky-diode
o------------------------------------ o-----|<|--o-- USB 5V
| | - |
_ | |
| | o / |
680k | | /=| Push button
|_| / |
100n | ____ ____ o----------o
>- | |--o--|____|--o--|____|--O PB4 44 mV |
500Hz | 3k3 | 10k to enable USB _
High _ | programming | |
Pass | | --- | | * 1k5 pullup
3k3 | | --- 22n 2kHz Low |_|
|_| | Pass |
| | _____ |
o----------o--O PB3 22 mV----|_____|--o
| * 68/22 |
_ __
| | /\` * 3V6 Z-diode
3k3 | | --
|_| | * = assembled USB circuit on Digispark
| |
_|_ _|_

PB2 O-- Serial out 115200 baud
PB1 O-- Feedback LED
PB0 O-- Relay
```

# Revision History
### Version 2.1.0
- Updated SimpleFrequencyDetector example.
- Renamed `printSignalValuesForArduinoPlotter()` to `printInputSignalValuesForArduinoPlotter()`,
`printLegendForArduinoPlotter()` to `printResultLegendForArduinoPlotter()`
and `printDataForArduinoPlotter()` to `printResultDataForArduinoPlotter()`.

### Version 2.0.0
- Added plotter output of input signal.
- Renamed `doPlausi()` to `doEqualDistributionPlausi()`.
- Changed [error values](src/FrequencyDetector.h#L170) and computation.
- Added documentation.
- Added [`MEASURE_READ_SIGNAL_TIMING`](src/FrequencyDetector.h#L64) capability.
- Refactored [WhistleSwitch example](examples/WhistleSwitch) and adapted to [`EasyButtonAtInt01`](https://github.com/ArminJo/EasyButtonAtInt01) library.
- Removed blocking wait for ATmega32U4 Serial in examples.

### Version 1.1.1
- Moved libraries for WhistleSwitch example.

### Version 1.1.0
- Corrected formula for compensating millis().
- New field PeriodOfOneReadingMillis.
- Now accept dropout values in milliseconds.
- New functions `printResultLegendForArduinoPlotter()` and `printResultDataForArduinoPlotter()`.

# CI
The library examples are tested with GitHub Actions for the following boards:

- arduino:avr:uno
- digistump:avr:digispark-tiny1
- ATTinyCore:avr:attinyx5:chip=85,clock=1internal