import warnings
warnings.filterwarnings("ignore", category=DeprecationWarning)
import seaborn as sns #this can be removed safely if seaborn is not available
sns.mpl.rc("figure", figsize=(7,4)) #idem
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.
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:
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.
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")
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).
Let ωp and ωs denote the edges of the pass-band and of the stop-band.
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).
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).
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
Elements of solution
# 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")
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.
For other kind of filters (namely band-pass, high-pass, band-stop), we can either:
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.
low-pass ωp-- low-pass ω1 z−1→z−1−α1−αz−1 with α=sin(ωp−ω12)sin(ωp+ω12)
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)
Sketch of solution
# 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])
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.
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")
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])