Contents
Processing math: 100%
In [1]:
 



In [2]:
import warnings
warnings.filterwarnings("ignore", category=DeprecationWarning) 
In [3]:
import seaborn as sns                 #this can be removed safely  if seaborn is not available
sns.mpl.rc("figure", figsize=(7,4))   #idem 

1.1  Synthesis of IIR filters by the bilinear transformation method¶

A simple and effective method for designing IIR digital filters with prescribed magnitude response specifications is the bilinear transformation method. The point is that already exists well-known optimal methods for the design of analog filters, such as Butterworth, Chebyshev, or elliptic filter designs. Then, the idea is to map the digital filter into an equivalent analog filter, which can be designed optimally, and map back the design to the digital domain. The key for this procedure is to dispose of a reversible mapping from the analog domain to the digital domain.

1.1.1  The bilinear transform¶

Recall that in the analog domain, the equivalent of the Z-transform is the Laplace transform which associates a signal s(t) with a function S(p) of the complex variable p. When p lives on the imaginary axis of the complex plane, then the Laplace transform reduces to the Fourier transform (for causal signals). For transfer functions, stability is linked to the positions of poles in the complex plane. They must have a negative real part (that is belong to the left half plane) to ensure the stability of the underlying system.

The formula for the bilinear transform comes from a formal analogy between the derivation operator in the Laplace and Z domains.

The bilinear transform makes an association between analog and digital frequencies, as follows:

p=k1−z−11+z−1,

where k is an arbitrary constant. The usual derivation leads to k=2/Ts, where Ts is the sampling period. However, using a general parameter k does not change the methodology and offers a free parameter that enables to simplify the procedure.

The point is that this transform presents some interesting and useful features:

  1. It preserves stability and minimum phase property (the zeros of the transfer function are with negative real part (analog case) or are inside the unit circle (discrete case).
  2. It maps the infinite analog axis into a periodic frequency axis in the frequency domain for discrete signals. That mapping is highly non linear and warp the frequency components, but it recovers the well-known property of periodicity of the Fourier transform of discrete signals.

The corresponding mapping of frequencies is obtained as follows. Letting p=jωa=j2πfa and z=exp(ωd)=exp(j2πfd). Plugging this in (2), we readily obtain

ωa=ktan(ωd2),

or

ωd=2arctan(ωak).

The transformation (3) corresponds to the initial transform of the specifications in the digital domain into analog domain specifications. It is often called a pre-warping . Figure shows the mapping of pulsations from one domain into the other one.

In [4]:
k = 2
xmin = -5 * pi
xmax = -xmin
omegaa = np.arange(xmin, xmax, 0.1)
omegad = 2 * np.arctan(omegaa / k)
plt.plot(omegaa, omegad)
plt.plot([xmin, xmax], [-pi, -pi], '--', color='lightblue')
plt.plot([xmin, xmax], [pi, pi], '--', color='lightblue')
#plt.text(-3.7,0.4,'Fs/2', color='blue',fontsize=14)
plt.xlabel("Analog pulsations $\omega_a$")
plt.ylabel("Digital pulsations $\omega_d$")
_ = plt.xlim([xmin, xmax])
plt.title("Frequency mapping of the bilinear transform")
figcaption("Frequency mapping of the bilinear transform", label="fig:BLT")
Caption: Frequency mapping of the bilinear transform

When designing a digital filter using an analog approximation and the bilinear transform, we follow these steps: Pre-warp the cutoff frequencies Design the necessary analog filter apply the bilinear transform to the transfer function Normalize the resultant transfer function to be monotonic and have a unity passband gain (0dB).

1.1.2  Synthesis of low-pass filters -- procedure¶

Let ωp and ωs denote the edges of the pass-band and of the stop-band.

  1. For the synthesis of the analog filter, it is convenient to work with a normalized filter such that Ωp=1. Therefore, as a first step, we set k=arctan(ωp/2) which ensures that Ωp=1. Then, we compute Ωs=2arctan(ωs/k).

  2. Synthetize the optimum filter in the analog domain, given the type of filter, the frequency and gain constraints. This usually consists in determining the order of the filter such that the gain constraints (ripples, minimum attenuation, etc) are satisfied, and then select the corresponding polynomial. This yields a transfer function Ha(p).

  3. Map back the results to the digital domain, using the bilinear transform (2), that is compute H(z)=Ha(p)|p=k1−z−11+z−1

Exercise 1
We want to synthetize a digital filter with fp=6kHz, with a maximum attenuation of -3dB, and a stop-band frequency of fs=9kHz, with a minimum attenuation of -9dB. The Nyquist rate (sampling frequency) is Fs=36kHz. - Represent the template for this sythesis, - Compute the pulsations in the analog domain (fix Ωp=1). - if we choose a Butterworth filter, the best order is n=2 and the corresponding polynomial is D(p)=p2+√2p+1, and the transfer function is Ha(p)=1/D(p). Compute H(z). - Plot H(f). Use `sig.freqz` for computing the transfer function. We also provide a function `plt_LPtemplate(omega, A, Abounds=None)` which displays the template of the filter. Import it using `from plt_LPtemplate import *`.

Elements of solution

  • k=1/tan(Ï€/6)=√3
  • Ωs=ktan(Ï€/4)=√3
  • H(z)=1+2z−1+z−2(4+√6)−4z−1+(4−√6)z−2
In [5]:
# compute the transfer function using freqz 
w, H = sig.freqz([1, 2, 1], [4 + sqrt(6), -4, 4 - sqrt(6)], whole=True)
# plot the result --w-pi corresponds to a shift of the pulsation 
#axis associated with the fftshift of the transfer function. 
plt.plot(w - pi, 20 * np.log10(fftshift(np.abs(H))))
# plot the filter template
from plt_LPtemplate import *
plt_LPtemplate([pi / 3, pi / 2], [-3, -9], Abounds=[5, -35])
plt.title("Realized filter and its template")
figcaption("Realized filter and its template")
/home/bercherj/.local/lib/python3.5/site-packages/ipykernel_launcher.py:5: RuntimeWarning: divide by zero encountered in log10
  """
Caption: Realized filter and its template

In practice, the transfer function can be generated by transforming the poles and zeros of the analog filter into the poles and zeros of the digital filter. This is simply done using the transformation z=1+p/k1−p/k. It remains a global gain that can be fixed using a value at H(1) (ω=0).This dramatically simplifies the synthesis in practice.

1.1.3  Synthesis of other type of filters¶

For other kind of filters (namely band-pass, high-pass, band-stop), we can either:

  • use the bilinear transform and the pre-warping to obtain a filter of the same type in the analog domain; then transferring the problem to the analog designer.
  • use an auxiliary transform that converts a digital low-pass filter into another type of filter. Using the second option, we see that the low-pass synthesis procedure has to be completed with
  1. Transform the specifications of the digital filter into specifications for a low-pass digital filter.

1.1.3.1  Transposition from a low-pass filter (ωp) to another type¶

Let ω1 and ω2 denote the low and high cut-off frequencies (only ω1 for the transposition of a low-pass into a high-pass). These transformations preserve the unit circle. That is, z=ejθ is transformed into z=ejθ′. There is an additional frequency warping, but the notion of frequency is preserved.

  1. low-pass ωp-- low-pass ω1 z−1→z−1−α1−αz−1 with α=sin(ωp−ω12)sin(ωp+ω12)

  2. low-pass ωp-- band-pass ω1,ω2 z−1→−z−2−2αββ+1z−1+β−1β+1β−1β+1z−2−2αββ+1z−1+1 with α=cos(ωp+ω12)cos(ωp−ω12) and β=tan(ωp2)tan(ω2−ω12)

Exercise 2
We want to synthetize an high-pass digital filter with edge frequencies 2,4,12 and 14 kHz, with a maximum attenuation of 3dB in the band-pass, and a minimum attenuation of -12dB in the stop-band. The Nyquist rate (sampling frequency) is Fs=32kHz. - Represent the template for this sythesis, - Compute the pulsations in the analog domain (fix Ωp=1). - if we choose a Butterworth filter, the best order is n=2 and the corresponding polynomial is D(p)=p2+√2p+1, and the transfer function is Ha(p)=1/D(p). Compute H(z). - Plot H(f). Use `sig.freqz` for computing the transfer function.

Sketch of solution

  • normalized pulsations: ω0=Ï€8,ω1=Ï€4,ω2=3Ï€4,ω3=7Ï€8
  • transposition into a digital low-pass α=0,β=tan(ωp/2). Choosing β=1, we get ωp=Ï€/2
  • the transform is thus z−1→−z−2
  • In the BLT, we take k=1; thus ωs=1 and ωs=tan(3Ï€/8)
  • Finally, we obtain H(z)=1+2z−1+z−2(2+√2)+(2−√2)z−2 for the digital low-pass, which after the transform z−1→−z−2 gives H(z)=1−2z−2+z−4(2+√2)+(2−√2)z−4
In [6]:
# compute the transfer function using freqz 
w, H = sig.freqz(
    [1, 0, -2, 0, 1], [2 + sqrt(2), 0, 0, 0, 2 - sqrt(2)], whole=True)
# plot the result --w-pi corresponds to a shift of the pulsation 
#axis associated with the fftshift of the transfer function. 
plt.plot((w - pi) / (2 * pi) * 32, 20 * np.log10(fftshift(np.abs(H))))
plt.xlabel("Frequencies")
_ = plt.ylim([-15, 0])
/home/bercherj/.local/lib/python3.5/site-packages/ipykernel_launcher.py:6: RuntimeWarning: divide by zero encountered in log10
  

1.1.4  Numerical results¶

Finally, we point out that these procedures have been systematized into computer programs. Two functions are available in scipy to design an IIR filter: sig.iirfilter that computes the coefficients of a filter for a given order, and sig.iirdesign that even determine a correct order given constraint on maximum and/or minimum attenuation. It is instructive to consult the help of these functions (e.g. help(sig.iirdesign)) and try to reproduce the results we obtained analytically above. Possible solutions are provided below.

In [7]:
b, a = sig.iirfilter(
    2, [1 / (pi)],
    rp=None,
    rs=None,
    btype='lowpass',
    analog=False,
    ftype='butter',
    output='ba')
# compute the ttransfer function using freqz 
w, H = sig.freqz(b, a, whole=True)
# plot the result --w-pi corresponds to a shift of the pulsation 
#axis associated with the fftshift of the transfer function. 
plt.plot(w - pi, 20 * np.log10(fftshift(np.abs(H))))
# plot the filter template
from plt_LPtemplate import *
plt_LPtemplate([pi / 3, pi / 2], [-3, -9], Abounds=[12, -35])
plt.title("Realized filter and its template")
figcaption("Realized filter and its template")
/home/bercherj/.local/lib/python3.5/site-packages/ipykernel_launcher.py:13: RuntimeWarning: divide by zero encountered in log10
  del sys.path[0]
Caption: Realized filter and its template
In [8]:
b, a = sig.iirdesign(
    [4 / 16, 12 / 16], [2 / 16, 14 / 16],
    3,
    12,
    analog=False,
    ftype='butter',
    output='ba')
# compute the ttransfer function using freqz 
w, H = sig.freqz(b, a, whole=True)
# plot the result --w-pi corresponds to a shift of the pulsation 
#axis associated with the fftshift of the transfer function. 
plt.plot((w - pi) / (2 * pi) * 32, 20 * np.log10(fftshift(np.abs(H))))
_ = plt.ylim([-20, 0])
/home/bercherj/.local/lib/python3.5/site-packages/ipykernel_launcher.py:12: RuntimeWarning: divide by zero encountered in log10
  if sys.path[0] == '':