Sunday, June 24, 2018

image processing - 2-d circularly symmetric low-pass filter


For a square pixel grid, the ideal 2-d low-pass filter with a horizontal and a vertical cut-off angular frequency ωc in radians has an impulse response (kernel) h(x,y) that is the product of a horizontal and a vertical stretched and scaled sinc function, with x and y the integer horizontal and vertical pixel coordinates:


h[x,y]=ωcsinc(ωcxπ)π×ωcsinc(ωcyπ)π={ω2cπ2if x=y=0,sin(ωcx)sin(ωcy)π2xyotherwise.


If ωc=π, the kernel of Eq. 1 is simply:


h=[1]ifωc=π.


For real-valued x and y, Eq. 1 would not describe a circularly symmetric kernel, and neither is the square-like frequency response H circularly symmetric. Sometimes an isotropic kernel may be desired, for example if we want to remove the effect of the choice of the direction of the image axes.


What is a kernel h[x,y] such that the frequency response H is 1 inside a circle of radius ωc (angular frequency in radians) and 0 outside it?



Answer




Excerpted from Jae S.Lim 2D signal and image processing ch.1, as an example of 2-D circularly symmetric lowpass filter with a cutoff frequency of ωc radians per sample, whose impulse response is given by: h[n1,n2]=ωc2πn21+n22J1(ωcn21+n22)


where J1 is the Bessel function of the first kind and the first order...


Interested readers may consult the book for a derivation which is not untitive but nevertheless tractable; familiarity with Bessel functions is required but also provided as is.[Derivation is added.]


[Olli below]


At n1=n2=0 the limiting value must be used:


h[0,0]=ω2c4π


A slice through the middle of h[n1,n2] with ωc=π:


enter image description here


Python source for 2-d filter kernel (you may want to apply a 2-d window function):


from scipy import special

import numpy as np

def circularLowpassKernel(omega_c, N): # omega = cutoff frequency in radians (pi is max), N = horizontal size of the kernel, also its vertical size, must be odd.
kernel = np.fromfunction(lambda x, y: omega_c*special.j1(omega_c*np.sqrt((x - (N - 1)/2)**2 + (y - (N - 1)/2)**2))/(2*np.pi*np.sqrt((x - (N - 1)/2)**2 + (y - (N - 1)/2)**2)), [N, N])
kernel[(N - 1)//2, (N - 1)//2] = omega_c**2/(4*np.pi)
return kernel

Example with ωc=π:


import matplotlib.pyplot as plt


kernelN = 11 # Horizontal size of the kernel, also its vertical size. Must be odd.
omega_c = np.pi # Cutoff frequency in radians <= pi
kernel = circularLowpassKernel(omega_c, kernelN)
plt.imshow(kernel, vmin=-1, vmax=1, cmap='bwr')
plt.colorbar()
plt.show()

enter image description here


Example with ωc=π/4:


kernelN = 41  # Horizontal size of the kernel, also its vertical size. Must be odd.

omega_c = np.pi/4 # Cutoff frequency in radians <= pi
kernel = circularLowpassKernel(omega_c, kernelN)
plt.imshow(kernel, vmin=-np.max(kernel), vmax=np.max(kernel), cmap='bwr')
plt.colorbar()
plt.show()

enter image description here


[fat32 below]


Ok, for those who would benefit from a (not so intuitive) derivation, here I make a (n almost) verbatim copy of it from the same book.


First, lets simply write the 2D discrete-time inverse Fourier transform to define the impulse response as:



h[n1,n2]=1(2π)2ω21+ω22<w2c1ej(ω1n1+ω2n2)dω1dω2


Let's make a change of variables ω1=rcos(θ) and ω2=rsin(θ) (effectively write the integral in (1) in the cyclindrical (or polar) coordinates) :


h[n1,n2]=1(2π)2ωcr=0a+2πθ=aejr(cos(θ)n1+sin(θ)n2)r drdθ


Now, make a further change of variables n1=ncos(ϕ) and n2=nsin(ϕ), with n=n21+n22 and obtain


h[n1,n2]=1(2π)2ωcr=0r dra+2πθ=aejrn(cos(θ)cos(ϕ)+sin(θ)sin(ϕ))dθ


which is now: h[n1,n2]=1(2π)2ωcr=0r dra+2πθ=aejrncos(θϕ)dθ


defining f(r)=a+2πθ=aejrncos(θϕ)dθ , then we get:


h[n1,n2]=1(2π)2ωcr=0rf(r)dr


Now, exploring f(r) we can apply the Euler's identity :


f(r)=a+2πθ=acos(r.n.cos(θϕ))dθ+ja+2πθ=asin(r.n.cos(θϕ))dθ



And we can notice that the imaginary part is zero (can check an integral table for that) and with a=ϕ then f(r) becomes


f(r)=2πθ=0cos(r.n.cos(θ))dθ


Now the integral in (7) is recognized as the the Bessel function of order zero, kind one J0(x) which is given as:


J0(x)=12π2πθ=0cos(xcos(θ))dθ


from (7) and (8) we see that f(r)=2πJ0(rn)...


And the last identity is given as is: xJ1(x)|ba=baxJ0(x)dx


Now putting the impulse response into the form


h[n1,n2]=1(2π)2ωcr=0r2πJ0(rn)dr


applying (9) onto (10) with the recognition that x=rn and dr=dx/n and n=n21+n22 yields the result:


h[n1,n2]=ωc2πn21+n22J1(ωcn21+n22)



No comments:

Post a Comment

periodic trends - Comparing radii in lithium, beryllium, magnesium, aluminium and sodium ions

Apparently the of last four, MgX2+ is closest in radius to LiX+. Is this true, and if so, why would a whole larger shell ($\ce{...