FIR Filter
A FIR Filter (Finite impulse response filter) is a type of filter used on signal processing. It works by taking the sum of last Nth samples multiplied by a value, called FIR taps. It's finite because if you pass a FIR filter in an impulse response, the impulse will fade out after passing though the N taps. That's easy to notice since the FIR filter never uses itself as feed unlike the IIR Filter.
On the SNES, the FIR filter has 8 taps which are 1.7 fixed point values. The filter is applied on the echo output so it has direct influence to the sound output and can be used to archive different effects which is more detailed on the following topics.
Mathematical Definition
The FIR filter can be defined in the following mathematical formula:
For the current output sample Y[n], take the sum of previous N samples from source (including current), multiplied by the FIR coefficient, which is:
The SNES has eight FIR taps, which limits N to 8:
However, the samples are processed from the oldest to the newest sample. That means for the first FIR tap is applied to the oldest sample while the 8th tap is applied to the newest sample:
Which in other words, it yields to the equivalent pseudo-code:
y[n] = FIR[0] * x[n - 7] + FIR[1] * x[n - 6] + FIR[2] * x[n - 5] + FIR[3] * x[n - 4] + FIR[4] * x[n - 3] + FIR[5] * x[n - 2] + FIR[6] * x[n - 1] + FIR[7] * x[n - 0];
Where FIR is a array containing the eight taps.
S-DSP Implementation
The FIR taps are located on the DSP registers $0F though $7F, where $0F is the first tap (FIR[0]) and $7F is the last tap (FIR[7]).
Because the S-DSP doesn't have floating point capability, the tap values are actually in the 1.7 fixed point, signed format. This means that values between $00-$7F is positive (0 to 127) and $80-$FF is negative (-128 to -1) and after doing the multiplication the value then is divided by 128.
In addition, the tap sum is done on a 16-bit integer type, which means that if an overflow occur the value gets clipped. The only exception for this is the last multiplication (last FIR tap multiplied by the first sample). For this case, the number gets clamped instead of clipped, e.g.: 18623 + 16888 will yield to 32767 instead of overflowing to -30025.
An accurate implementation of the FIR filter using the above rules would be:
S = (FIR[0] * x[n - 7] >> 6) + (FIR[1] * x[n - 6] >> 6) + (FIR[2] * x[n - 5] >> 6) + (FIR[3] * x[n - 4] >> 6) + (FIR[4] * x[n - 3] >> 6) + (FIR[5] * x[n - 2] >> 6) + (FIR[6] * x[n - 1] >> 6);
The ">>" operator is the arithmetic right shift operation, which on this context is the equivalent by dividing the value by 128 (without decimal places).
With the first 7 taps calculated, we clip it so it's always within the -32768 to 32767 range:
S = S & 0xFFFF;
Now we add the last tap which is multiplied by the current sample, x[n]:
S = S + (FIR[0] * x[n] >> 6);
clamp it to the -32768 to 32767:
S = S > 32767 ? 32767 : S; S = S < -32768 ? -32768 : S;
It's important to remember that the Echo buffer uses 15-bit samples instead of 16-bit and it's left-aligned, so the binary format is: "seee eeee eeee eee0" and not "ssee eeee eeee eeee". However, when the sample is placed on the sample history (x[]), it end ups right shifted so it gets to the "ssee eeee eeee eeee" format.
So we finally take the last bit and we have our FIR value done:
Result = S & 0xFFFE;
Each sound channel (left and right) is processed separately.
Because of the integer clipping behavior on the first seven taps, it's important to your FIR filter DC gain never exceed 0 dB (or the absolute sum of all FIR taps never exceed 128) for avoiding audio clicks, specially with the first seven taps. Not all games obey that, so the most important is checking out the frequency response of the filter and checking how the gain behaves for the frequency range.
Once done, the FIR is multiplied by the L/R echo volume and it's output together the main volume. It's also multiplied by the echo feedback value and fed back to the echo buffer together the sound output.
Timing Information
The S-DSP generates one stereo sample every 32 SPC700 clocks. During these 32 clocks, some operations are done by the S-DSP regarding echo and FIR filter. These are the following:
Each time a new sample is generated on the S-DSP, one sample is taken from the echo buffer and inserted on the sample history (x[]), then the sample history is used to generate the FIR filter output:
- The left echo channel is inserted on the sample history at cycle 22 and written on cycle 29.
- The right echo channel is inserted on the sample history at cycle 23 and written on cycle 30.
The FIR taps are not read immediately, but gradually read while the left and right echo channel are being processed:
- FIR[0] is accessed during cycle 22.
- FIR[1] and FIR[2] are accessed during cycle 23.
- FIR[3], FIR[4] and FIR[5] are accessed during cycle 24.
- FIR[6] and FIR[7] are accessed during cycle 25.
Uses
The most useful thing that you can do with FIR filters is amplifying or attenuating a frequency range. Effectively, you can create your own sound equalizer depending on the echo settings and main volume combinations.
The most common usages involve:
- Using FIR as a low or high pass filters, normally using 4.5 kHz as the cut-off point:
- Low-pass to filter out high frequencies from the echo sound output.
- High-pass to filter out low frequencies from the echo sound output.
- Identity filter ($7F on the first tap and $00 on others): output original produced song from the echo buffer.
Frequency response
It's possible to view the frequency response of a FIR filter and therefore visualize which frequencies are amplified or attenuated. The calculations involves Discrete-time Fourier transform (DFFT) and are considerably complex. The formula looks like this:
Where:
- is the output power for given frequency . It is a complex number, therefore you will end up having two components, which is the magnitude and phase .
- Magnitude is the output volume e.g. 1.0 is 100% volume.
- Phase means how late or early is the signal relative to the original, in radians (). Humans normally don't perceive phase changes, but combining two close and oppose phased-signals can generate some odd effects like an oscillating signal used on Square SPC engines.
- is the FIR filter taps.
- is the amount of taps.
Disregarding SNES clipping and clamping effects, the frequency response of its 8-tap FIR would be calculated as the following:
Expanding the sum yields:
When doing a visualization, it's often common to see ripples or signal ringing on certain regions. Even with careful values choice, it's not possible to attenuate them because of the low amount of taps the SNES FIR filter has, but regardless of the amount of taps they are often common on modern DSP applications and other kind of filters.
VilelaBot has a command for viewing FIR filters, using the !vb fir <FIR filter taps in hex> command.
Examples
Low-pass filters
Low-pass filters attenuates high frequencies (high pitched sound).
Game | FIR Filter | Cut-off frequency | Maximum Gain | Notes |
---|---|---|---|---|
Kyuuyaku Megami Tensei | $0B $21 $28 $28 $18 $FC $FB $F7 | 4 kHz | +0.42 dB | High frequencies were reduced to -20 dB average. |
Band-stop filters
Band-stop filters attenuates frequencies between a frequency region. It has a "U" like aspect.
Game | FIR Filter | Left Freq. | Right Freq. | Max. Gain | Notes |
---|---|---|---|---|---|
Chaos Seed | $FF $40 $20 $20 $00 $00 $00 $00 | 5 kHz | 14 kHz | -0.07 dB | It has na "U"-like with 10 kHz at center. |
Do-Re-Mi Fantasy - Milon no Dokidoki Daibouken | $00 $00 $00 $00 $48 $00 $30 $00 | 4 kHz | 12 kHz | -0.56 dB | It has a "V"-like curve with 8 kHz at center. |
Many games | $0C $21 $2B $2B $F3 $FE $F3 $F9 | 6 kHz | 12 kHz | -0.14 dB | It looks like a lying "Z". <6 kHz with -2dB and >12 kHz with -11dB. At 9 kHz, -14 dB. |
Popful Mail | $08 $FF $E0 $B8 $04 $80 $09 $FF | 4 kHz | 10 kHz | +4.42 dB | It has an "U"-like curve with 6.7 kHz at center. |
Super Bomberman 3 | $00 $00 $00 $28 $00 $28 $00 $FE | 5 kHz | 11 kHz | -4.30 dB | It has an even "V"-like curve with 8 kHz at center. |
Super Bomberman 3 | $00 $00 $00 $40 $00 $28 $00 $FE | 5 kHz | 11 kHz | -1.97 dB | It has an even "V"-like curve with 8 kHz at center. |
Tengai Makyou Zero | $00 $00 $00 $00 $30 $00 $48 $00 | 4 kHz | 12 kHz | -0.56 dB | It has a "V"-like curve with 8 kHz at center. |
Unusual filters
Game | FIR Filter | Maximum Gain | What it Does |
---|---|---|---|
Alien vs. Predator | $0C $2B $2B $2B $28 $FF $F8 $F9 | +2.21 dB | Band-stop between 6 kHz and 8.5 kHz (-15 dB). Smooth Low-pass cut-off at 13 kHz (-18 dB). |
Angelique Voice Fantasy | $0D $F5 $FA $12 $12 $FA $F5 $0D | -5.40 dB | Eliminates 12 kHz and 16 kHz. |
Donkey Kong Country | $00 $01 $00 $00 $00 $00 $00 $00 | -42.14 dB | Makes echo almost inaudible. |
Doom | $FA $0A $1A $13 $13 $1A $0A $FA | -2.32 dB | Eliminates 6.4 kHz, 12.8 kHz and 16 kHz. |
Dragon Guest VI | $0D $10 $10 $24 $3D $F4 $F4 $F4 | -0.76 dB | It looks like a twisted "W" at ~6 kHz (-5 dB) and ~12.5 kHz (-10 dB). |
Front Mission, BS Koi ha Balance, BS Dynani Tracer | $XX $00 $00 $00 $00 $00 $00 $00 | -0.14 dB | $XX fades in from $00 to $7E at beginning, essentially controlling the echo volume and feedback. |
Jim Power | $FF $FC $FD $FE $FF $FE $FD $FC | -16.12 dB | Many ripples at 4 kHz (-50 dB), 9 kHz (-45 dB), 13.5 kHz (-42 dB), gradually attenuating higher frequencies. |
Kamaitachi no Yoru | $FF $0A $1E $32 $32 $1E $0A $FF | +2.86 dB | Eliminates 9.5 kHz, 11.5 kHz and 16 kHz. |
Mario is Missing! | $0A $0A $0A $0A $0A $0A $0A $0A | -4.08 dB | Eliminates 4, 8, 12 and 16 kHz in a way that echo doesn't explode on the song. |
Popful Mail | $06 $B0 $07 $FF $EB $B8 $0E $00 | +1.93 dB | It looks like a twisted "W" at ~4.5 kHz (-22 dB) and ~12.5 kHz (-16 dB). |
Prince of Persia II | $0F $08 $86 $08 $0B $09 $90 $09 | +6.17 dB | It looks like a "W" at ~4 kHz and ~12 kHz. |
Star Ocean, Tales of Phantasia | $0D $22 $22 $24 $11 $F0 $03 $FF | -0.56 dB | Band-stop between 5 kHz and 10 kHz (-16 dB). Smooth Low-pass cut-off at 13 kHz (-17 dB). Echo slowly overflows without the FIR. |
Super Genjin 2 | $00 $00 $00 $36 $00 $28 $00 $41 | +1.88 dB | It looks like a "W" at ~5 kHz and ~11 kHz. |
Super Tekkyuu Fight! | $0E $49 $4B $46 $5F $08 $DE $08 | +7.66 dB | It looks like a twisted "W" at ~7 kHz (-5 dB) and ~15 kHz (-30dB). |
The Flintstones | $XX $33 $00 $D9 $E5 $01 $FC $EB | +4.34 dB | $XX ranges from $00 to $7F, essentially a light high-pass filter with volume fade. |
The Mask | $FA $02 $18 $28 $2A $F4 $0A $1E | +0.14 dB | Slight attenuates at 4 kHz (-7 dB), then starting at 9 kHz it gradually starts attenuating (-15 dB). |
The Simpsons - Bart's Nightmare | $00 $08 $10 $10 $10 $10 $08 $00 | -4.08 dB | Eliminates 6.8 kHz and 13.6 kHz in a way that echo doesn't explode on the song. |
External links
- Romhacking.net - Documents - Anomie's S-DSP Doc
- Fullsnes - Nocash SNES Specs
- Design of FIR Filters - Dr. Elena Punskaya
- What is FIR Filter? - FIR Filters for Digital Signal Processing
- ECE 2610 Signal and Systems - FIR Filters
- FIR Digital Filter Design | Spectral Audio Signal Processing
- FIR filter FAQ - dspGuru
- The relationship between decibels, volume and power - sengpielaudio