Log Gabor filter

Source code Author Update time

This example shows how one can apply frequency space kernesl LogGabor and LogGaborComplex using fourier transformation and convolution theorem to extract image features. A similar example is the sptaial space kernel Gabor filter.

using ImageCore, ImageShow, ImageFiltering # or you could just `using Images`
using FFTW
using TestImages

Definition

Mathematically, log gabor filter is defined in spatial space as the composition of its frequency component r and angular component a:

\[r(\omega, \theta) = \exp(-\frac{(\log(\omega/\omega_0))^2}{2\sigma_\omega^2}) \\ a(\omega, \theta) = \exp(-\frac{(\theta - \theta_0)^2}{2\sigma_\theta^2})\]

LogGaborComplex provides a complex-valued matrix with value Complex(r, a), while LogGabor provides real-valued matrix with value r * a.

kern_c = Kernel.LogGaborComplex((10, 10), 1/6, 0)
kern_r = Kernel.LogGabor((10, 10), 1/6, 0)
kern_r == @. real(kern_c) * imag(kern_c)
true
Lazy array

The LogGabor and LogGaborComplex types are lazy arrays, which means when you build the Log Gabor kernel, you actually don't need to allocate any memories. The computation does not happen until you request the value.

using BenchmarkTools
kern = @btime Kernel.LogGabor((64, 64), 1/6, 0); # 1.711 ns (0 allocations: 0 bytes)
@btime collect($kern); # 146.260 μs (2 allocations: 64.05 KiB)

To explain the parameters of Log Gabor filter, let's introduce small helper functions to display complex-valued kernels.

show_phase(kern) = @. Gray(log(abs(imag(kern)) + 1))
show_mag(kern) = @. Gray(log(abs(real(kern)) + 1))
show_abs(kern) = @. Gray(log(abs(kern) + 1))

From left to right are visualization of the kernel in frequency space: frequency r, algular a, sqrt(r^2 + a^2), r * a, and its spatial space kernel.

kern = Kernel.LogGaborComplex((32, 32), 100, 0)
mosaic(
    show_mag(kern),
    show_phase(kern),
    show_abs(kern),
    Gray.(Kernel.LogGabor(kern)),
    show_abs(centered(ifftshift(ifft(kern)))),
    nrow=1
)

Examples

Because the filter is defined on frequency space, we can use the convolution theorem:

\[\mathcal{F}(x \circledast k) = \mathcal{F}(x) \odot \mathcal{F}(k)\]

where $\circledast$ is convolution, $\odot$ is pointwise-multiplication, and $\mathcal{F}$ is the fourier transformation.

Also, since Log Gabor kernel is defined around center point (0, 0), we have to apply ifftshift first before we do pointwise-multiplication.

img = TestImages.shepp_logan(127)
kern = Kernel.LogGaborComplex(size(img), 50, π/4)
# we don't need to call `fft(kern)` here because it's already on frequency space
out = ifft(centered(fft(channelview(img))) .* ifftshift(kern))
mosaic(img, show_abs(kern), show_mag(out); nrow=1)

A filter bank is just a list of filter kernels, applying the filter bank generates multiple outputs:

X_freq = centered(fft(channelview(img)))
filters = vcat(
    [Kernel.LogGaborComplex(size(img), 50, θ) for θ in -π/2:π/4:π/2],
    [Kernel.LogGabor(size(img), 50, θ) for θ in -π/2:π/4:π/2]
)
out = map(filters) do kern
    ifft(X_freq .* ifftshift(kern))
end
mosaic(
    map(show_abs, filters)...,
    map(show_abs, out)...;
    nrow=4, rowmajor=true
)

This page was generated using DemoCards.jl and Literate.jl.