This project aims to provide a simple to build audio-compressor based on an Arduino / Genuino.
Note that we are assuming an ATMega328 or ATMega168 based board, here (such as Arduino Uno, Nano, or Pro Mini), running at 16 MHz and 5V. Circuit and code can clearly be adjusted to other boards, but you'll have to do that, yourself.
If you came here, you probably know, what an audio-compressor is used for: Reducing the dynamic range of an audio signal, often so the more subtle sounds are not affected, but the louders sounds are brought down to levels compatible with your hardware, ears, and/or neighbors.
Achieving a useful compression without undesirable side-effects depends on careful tuning of some parameters, though:
- Attack: Onset response time, i.e. how soon the compressor starts toning down. Should be fast, but not too fast. You want to cut out loudness, not crispiness.
- Release: Offset response time. Usually several times longer than attack, in oder to avoid oscillation.
- Threshold: Signals below a certain level should not be compressed.
- Ratio: Proportion by which to tune down signals exceeding the threshold, e.g. 2, in order to cut excess in half.
Sounds like a perfect job for a microcontroller, right?
Beyond the logic and timing, however, there are two main technical difficulties: First, we need to sense the signal level to work on. Ok, not terribly difficult, but does require some tricks to achieve with an acceptable sampling rate on a microcontroller. Second, we need to modify (or output) the signal, accordingly. Typical approaches involve voltage controlled amplifiers (VCAs), J-FETs, or some other technique to transform a voltage signal into a variable resistance. Well-established, but not quite trivial, if you read up on it. Instead of these, I decided to capitalize on the main advantage of a microprocessor: Doing simple digital things fast. In this case this means switch the audio signal on and off at a rate much higher than audible sounds, in order to achieve a variable resistance very easily.
The following circuit is far from ideal in many ways, but don't worry, it's easy to improve on (even without changing the microcontroller code, for the most part). But we'll start simple (and quite possibly, this may already be enough for your prupose):
Let's start with the bottom half of the circuit: Here, the audio signal is decoupled via a largish capacitor, and biased to a level of roughly 3.3V, in order to bring it to a level suitable for sampling by the arduino. The reason to use 3.3V, here, is simply that a stabilized 3.3V output is already available on many Arduino boards. In principle we could simply use two resistors to form a voltage divider (and probably base that on 2.5V), but a stabilized voltage is clearly preferrable, esp. when powering from USB. Contrary to the schematic, an electrolyte cap is perfectly fine for the decoupling, with the negative side connected to audio in, and the positive side to the Arduino side. The capacitor should be rather large in order to allow for low frequencies to pass. However, the exact value does not matter. Similarly, the resistor value does not matter too much at all.
The biased signal is now fed into the Arduino (pin A0), where it will be sampled at roughly 77kHz. The sampling accuracy is not terribly good, and so one of the requirements of this basic circuit is having a suitably large input signal (line level/headphone level is more than enough). But not a whole lot of accuracy is needed, either: The only point of this part of the circuit is to allow the Arduino to detect the current signal level, and - remember - low signal levels are not of interest in the first place.
The more interesting part of the circuit is the upper half. Two N-FETs are connected back to back (drain of the first connected to source of the second and vice versa), which are both controlled synchronously from Arduino pin D3. These two FETs simply function as a simple analog switch. The good news is that you are pretty likely to have those N-FETs in your part-collection already: A pair of 2N7000 or any other common small signal N-FET will do ok. Importantly, the FET should be far inside the on-region at 4-5 Volts. Also, of course, if should be able to handle whatever current will be flowing. However, also, it should have a rather high bdoy diode forward voltage drop, as that will limit how large voltages we can switch off, reliably. A somewhat better choice than the 2N7000 would be the IRLML2502, and an even better choice will be to use a dedicated analog switch (in this particular schematic you'd want one supporting negative voltage swings!), but again, the 2N7000 will perform ok-ish, and is enough for connecting a headphone, so try that, first. Also, again, the exact resistor values will not matter. The 220 Ohms is for limiting the gate (dis-)charge current, to what the Arduino can safely handle. The 470 Ohms is to provide a bit of isolation from noise in the power supply.
What the switch is doing is simply turning on and off the audio ground[1] at a rate and duty cycle controlled by Arduino digital pin 3 (PWM). The code uses a ~66kHz frequency with duty cycle adjusted between 100% and ~5%. The 66kHz switching will not be audible (your speakers will not even be able to represent such frequencies), but if you are concerned about high frequency artifacts, you could easily add a simple low-pass filter.
That's all folks, nothing else needed. At least if all you need is switching a line level / headphone level mono signal. Need more? Read on / hang on for more sophisticated circuits built around the same idea.
With some luck, the above circuit will simply work for you, but in many cases you will have to do some tuning. You can do that tuning entirely from the sketch, by adjusting the variable values near the top of the code (see the comments in the code, and the following section for details). However, for tuning parameters more comfortably and at runtime, you can easily add some status indicators and controls:
- Connect four LEDs (with appropriate resistors) from pins D10 through D13 to ground. I suggest using green on D10, yellow and D11 and red on D12 and D13.
- Connect a 2 by 4 button matrix to pins D4 through D9. D8 and D9 should be connected to the two row wires, D4 through D7 should be connected to the four column wires.
The two buttons at D4 will be used for tuning attack up and down, the buttons at D5 are for controlling release, D6 is threshold, D7 is ratio. (Pin setup can be customized in the source).
Ok, so how to get started?
- First, slowly turn up the input volume such that the LED on pin D13 will almost, but not quite light up on the loudest signals. (This LED is meant to signify sound levels that are approaching the limits of what the hardware can handle. Don't worry if your signal source does not deliver that much. In this case just turn up the input signal as much as possible).
- Next adjust the threshold value. The green LED on pin D10 will light up as soon as the compressor starts kicking in, i.e. once the threshold is reached. The threshold should be high enough that you can comfortably hear important sounds (such as low speech), but ideally a good deal below sounds that are "definitely too loud". Too low of a threshold may lead to sound artifacts esp. for low signals.
- Adjust the ratio such that the loudest sounds remain inside the acceptable range. Generally, the red LED at D12 should not light up, or only for extreme sounds. It indicates that the compressor is operating close to the limit. Much louder, and sounds cannot be tuned down without considerable distortion. (D12 corresponds to tuning down the signal to roughly 1/12 or -22dB; the maximum the compressor will do is -28dB; the yellow LED at D11 signals tuning down to 1/2 or -6dB).
- For adjusting attack and release, it's hardest to outline a clear procedure, but also these will generally work quite fine at their default values. Note that too small values of attack can lead to artifacts, too small values of release can lead to the compressor "pumping" on certain sounds.
Note: When adjusting parameters, the status LEDs will briefly change their role from indicating signal and compression levels to a very rough indication of the parameter that was changed. No LEDs active signifies a very low value, with LEDs lighting up from D12 to D10 in that order for higher values. If either the low or high end of the scale is reached D13 will light up in addition.
The cuircit is simple, you now know how to adjust parameters, but what exactly is happening in the code?
- Sampling windows
- Moving averages
- The actual volume adjustment is then calculated as follows: If the current signal is ndB above the threshold level, divide n by the ratio, and adjust the signal level to an output of threshold + n/ratio dB.
Now, since decibel is a logarithmic scale, that translates to the following pseudo-code (current is the current output voltage level, threshold is the threshold voltage level):
Now the ATMega is not exactly fast at floating point math, let alone at calculating logs. The above is just prohibitively slow. Fortunately it can be optimized:
target_level = exp((log(current) - log(threshold))/ratio + log(threshold));
Now the ATMega can handle that calculation just fine. As a last step, we simply set the duty-cycle of Pin 3 to be 255 * target_level / current_level.target_level = threshold * exp(log(current)/ratio) / exp(log(threshold)/ratio; // rewriting the division by ratio in the exponent: target_level = threshold * pow(exp(log(current)), 1/ratio) / pow(exp(log(threshold)), 1/ratio); // simplifying: target_level = threshold * pow(current, 1/ratio) / pow (threshold, 1/ratio); // great! The logs are gone. Now: target_level = threshold * pow(current/threshold, 1/ratio);
... but writing this up is more work than it may seem. If you find this useful, consider donating a bread crumb or two via Paypal: thomas.friedrichsmeier@gmx.de
[1] Originally, I was switching the audio signal line, instead of audio ground. Switching on the ground connection has two advantages: First, the N-FETs gate-to-source potential will be (mostly) independent of the current signal level, allowing for more linearity. Second, this way subject to some caveats you can switch on and off a stereo signal with a single N-FET pair (but alternatively, connect a second pair in parallel, also connected to pin D3). Of course, if your next stage is e.g. a high impedance amp, rather than a speaker coil, you should bias "Audio Ground Out" to ground via a largish resistor (some k Ohms).