Real-Time Implementation of IIR Filters
Real-Time Implementation of IIR Filters
Real-Time Implementation of IIR Filters
b0 + b1z 1 + m + bN z M . 1 + a1z 1 + m + a N z N
y (n ) = b0 x ( x ) + b1 x ( n 1) + m + bM x ( n M ) a1 y ( n 1) m a N y (n N ) Single-Pole Filter
As an initial check, you should implement the single-pole low-pass filter H ( z ) =
b0 . The 1 + a1 z 1
impulse response of this filter is a negative exponential and for a time constant of , you need to set a1 = exp( T / ) where T = 1 / 8000 is the sample period. Implement this filter with
= 1 ms and choose b0 to give a DC gain of unity. By driving the input with a low-frequency
squarewave, verify that the impulse response is correct. Verify also that the frequency response is as expected and has the correct corner frequency.
This can be designed using the MATLAB function ellip(). You should write a MATLAB function that calculates the coefficients for such a filter and writes them into a text file called coef.txt in a format suitable for inclusion in a C program. For example, for a third-order filter, the file might contain: float a[] = { 1, -1.76, 1.1829, -0.2781,}; float b[] = { 0.0181, 0.0543, 0.0543, 0.0181,}; The comma following the final value in each line is optional but makes your MATLAB output routine easier. Note that the ellip() function requires you to specify frequencies normalized to the Nyquist frequency and to specify the order as half the desired value. Write a C program to implement an IIR filter in Direct Form II as shown in the Figure below. Your program should work for any filter order, but you can assume that a[] and b[] are the same
length. You can determine the filter order (which is one less than the length of a[] ) and allocate the required temporary storage with the following statements: order = sizeof(a)/sizeof(a[0]) - 1; w = (float *) calloc(order, sizeof(float)); You may require the w[] array to be of length order or order+1 according to the nature of your algorithm. Verify that the filter frequency response agrees with the MATLAB prediction.
xin
+ a1 a2 a3 D D D
b0 b1 b2 b3
yout
Use the profiler to determine how many instruction cycles per sample are needed for are needed for a filter of order n in the form A + Bn . You should include only the instructions between the calls to AD535_HWI_read() and AD535_HWI_write(). Now recompile your program but using the Compiler option o2 to optimise the program for speed. See what difference this makes to the number of instruction cycles required.
xin b3 + a3 D b2 + a2 D b1 + a1 D b0 + yout
Verify that the filter response is unchanged. Determine the cycle count with and without optimisation.
xin
a12 D
a22 D
H ( z) = g
Note that you only need a single gain coefficient, g, for the whole filter and do not need individual b0 coefficients for each stage. You may find it easiest to have four separate coefficient arrays for the a*1 , a*2 , b*1 , b*2 values. In MATLAB, you can convert a direct-form filter into cascaded second-order sections using the function tf2sos(). Verify that your bandpass filter works correctly. The main reason for using a cascaded biquad implementation is that it is much less sensitive to coefficient errors. To see this, design the following enhanced version of your filter and try it out using both your direct form and your biquad versions: Order Passband Passband ripple Stopband attenuation 12th 100 Hz to 500 Hz 0.5 dB 40 dB
You will probably find that your direct form filter doesnt work: to see what is going wrong, use the watch window to examine the signal values. Even for a comparatively uncomplicated filter such as this, the coefficients need to be fantastically precise. MATLAB works to a precision of 52 significant bits but we can artificially reduce the precision to n bits with the following routine:
function y=bitsprec(x,n) [x,e]=log2(x); y=pow2(round(pow2(x,n)),e-n); end
By evaluating abs(roots(bitsprec(a,n))) for various values of n, determine how many bits of precision are required to ensure that the 12th order filter is stable (a decimal digit corresponds to 3.3 bits). Do the same for the biquad filter sections.