Reduce background noise and optimize the speech from an audio clip using ffmpeg

73,381

Solution 1

If you are looking to isolate audible speech try combining a lowpass filter with a high pass filter. For usable audio I have noticed that filtering out 200hz and below then filter out 3000hz and above does a pretty good job of keeping usable voice audio.

ffmpeg -i <input_file> -af "highpass=f=200, lowpass=f=3000" <output_file>

In this example add the high pass filter first to cut the lower frequencies then use the low pass filter to cut the higher frequencies. If needed you could run your file through this more than once to clean up higher db frequencies within the cut frequency ranges.

Solution 2

FFmpeg now has 3 native filters to deal with noise background:

  • afftdn: Denoises audio samples with FFT
  • anlmdn: Reduces broadband noise in audio samples using a Non-Local Means algorithm
  • arnndn: Reduces noise from speech using Recurrent Neural Networks. Examples for model files to load can be found here.

Also, since some time, one can use ladspa (look for noise-supressor) and/or lv2 (look for speech denoiser) filters with FFmpeg.

Solution 3

Update: FFmpeg recently added afftdn which uses the noise threshold per-FFT-bin method described below, with various options for adapting / figuring out appropriate threshold values on the fly.

anlmdn (non-local means) is a technique that works well for video; I haven't tried the audio filter.

Either of these should be much better than highpass / lowpass, unless your only noise is a 60Hz hum or something. (Human speech can still sound ok in a pretty narrow bandpass, but there are much better ways to clean up a broadband noise background hiss.)


ffmpeg doesn't have any decent audio filters for noise-reduction built in. Audacity has a fairly effective NR filter, but it's designed to be used with 2-pass operation with a sample of just the noise, and then the input.

The comments at the top of https://github.com/audacity/audacity/blob/master/src/effects/NoiseReduction.cpp explain how it works. (basically: suppress every FFT bin that's below the threshold. So it only lets signals through when they're louder than the noise floor in that frequency band. It can do amazing things without causing problem. It's like a band-pass filter that adapts to the signal. Since the energy of the noise is spread over the whole spectrum, only letting through a few narrow bands of it will reduce the total noise energy a LOT.

See also Audio noise reduction: how does audacity compare to other options? for more details of how it works, and that thresholding FFT bins in one way or another is the basis of typical commercial noise-reduction filters, too.

Porting that filter to ffmpeg would be a bit awkward. Maybe implementing it as a filter with 2 inputs, instead of a 2-pass filter, would work best. Since it only needs a few seconds to get a noise profile, it's not like it has to read through the whole file. And you SHOULDN'T feed it the whole audio stream as a noise sample, anyway. It needs to see a sample of JUST noise to set thresholds for each FFT bin.

So yeah, a 2nd input, rather than 2pass, would make sense. But that makes it a lot less easy to use than most ffmpeg filters. You'd need a bunch of voodoo with stream split / time-range extract. And of course you need manual intervention, unless you have a noise sample in a separate file that will be appropriate for multiple input files. (one noise sample from the same mic / setup should be fine for all clips from that setup.)

Solution 4

I had a video with vary bad background noise. I managed to fix it in this way: I did two pass with following command:

ffmpeg -i input.mp4 -af "afftdn=nf=-25" file1.mp4

ffmpeg -i file1.mp4 -af "afftdn=nf=-25" file2.mp4

Than I used in order to clarify the speak:

ffmpeg -i file2.mp4 -af "highpass=f=200, lowpass=f=3000" file3.mp4

At the end increased the volume with:

ffmpeg -i file3.mp4 -af "volume=4" finaloutput.mp4

In this way I managed to have a fairly good audio. Anyway sound is something subjective and what is good for me can be not for others. Hope it helps. M.M.

Solution 5

The combination of lowpass and highpass filters with afftdn is quite impressive, I've managed to clean old vhs video from white noise with this configuration:

-af "highpass=200,lowpass=3000,afftdn"

Share:
73,381

Related videos on Youtube

Sudh
Author by

Sudh

Updated on September 18, 2022

Comments

  • Sudh
    Sudh over 1 year

    I extract audio clips from a video file for speech recognition. These videos come from mobile/other handmade devices and hence contain a lot of noise. I want to reduce the background noise of the audio so that the speech that I relay to my speech recognition engine is clear. I am using ffmpeg to do all of this stuff, but am stuck at the noise reduction phase.

    Till now I have tried following filters:

    ffmpeg-20140324-git-63dbba6-win64-static\bin>ffmpeg -i i nput.wav -filter_complex "highpass=f=400,lowpass=f=1800" out2.wav
    
    ffmpeg -i i nput.wav -af "equalizer=f=1000:width_type=h:width=900:g=-10" output.wav
    
    ffmpeg -i i nput.wav -af "bandreject=f=1200:width_type=h:width=900:g=-10" output.wav
    

    But the results are very disappointing. My reasoning was that since speech comes under 300-3000 hz range I can filter out all other frequencies to suppress any background noise. What am I missing?

    Also, I read about weiner filters that could be used for speech enhancements and found this but am not sure how to use it.

    • Totor
      Totor over 2 years
      -af arnndn=m=cb.rnnn' is probably the best noise filter I have used with ffmpeg. No need for frequency band-filters. Trained models (files.rnnn) available here.
  • Angad
    Angad over 8 years
    Sorry, but this seems to do no noticeable noise reduction for me.
  • Iain Collins
    Iain Collins over 7 years
    This works very well to reduce low level of background noise (fans, buzzing, etc) but may compromise the audio quality slightly, though that can be mitigated somewhat by applying other filters afterwards.
  • Eric
    Eric about 7 years
    For my case the original audio was so bad it was almost impossible to hear the voice because of some water fall noice in the background. I used the following. It is not great quality, but 1000x better than the original. -af "highpass=f=200, lowpass=f=1000"
  • shevy
    shevy over 6 years
    I get some error with the above or rather, warning from ffmpeg: [Parsed_highpass_0 @ 0x1524780] clipping 52 times. Please reduce gain.
  • Björn
    Björn about 6 years
    You may preview your filter with ffplay <input file> -af lowpass=3000,highpass=200
  • erandros
    erandros over 5 years
    This increased my file size from 8 MiB to 70 MiB. The file format is flv.
  • Riz
    Riz almost 4 years
    can we use this filter when we are recording windows desktop? Like how to use these filters in this command ffmpeg.exe -threads 4 -rtbufsize 2048m -f dshow -i audio="Microphone (Realtek Audio)" -f gdigrab -offset_x 1 -offset_y 1 -video_size 768x432 -probesize 4096M -i desktop -pix_fmt yuv420p -c:v libx264 -crf 28 -preset ultrafast -tune zerolatency -movflags +faststart -r 15 "test.mp4"
  • Admin
    Admin almost 4 years
    sure, you will need enough fast cpu and also download models files for arnndn filter to work correctly. Basically you would just add -af arnndn=path/to/model_file.rnnn.
  • Riz
    Riz almost 4 years
    thanks but there is a small mistake. Correct syntax is -af arnndn=m=path/to/model_file.rnnn
  • Vijay Prema
    Vijay Prema about 3 years
    Thanks! this command works very well. Additionally, I did a stream copy for video using -c:v copy to make it run way faster and not change the video in any way (I was cleaning audio for a mov file). ffmpeg -i myvideo.mov -af "afftdn=nf=-25" -c:v copy myvideo_clean.mov
  • Geoff Langenderfer
    Geoff Langenderfer almost 3 years
    @VijayPrema so we're avoiding video processing with this?
  • Vijay Prema
    Vijay Prema almost 3 years
    @GeoffLangenderfer Correct, with -c:v copy the video is simply copied over to the output file without processing. Only the audio is filtered with afftdn.
  • dannyman
    dannyman over 2 years
    Bjorn's suggestion is great because you can skim this whole page and quickly experiment with chaining different filters in different orders. This way I find, for example, a nice combined filter: -af lowpass=3000,highpass=200,afftdn=nf=-25
  • Admin
    Admin almost 2 years
    @Matteo use -i for input file ffmpeg file3.mp4 -af "volume=4" finaloutput.mp4