Files
scikit-image/skimage/deconvolution/uft.py
T
2013-12-10 22:45:09 +01:00

430 lines
11 KiB
Python

# -*- coding: utf-8 -*-
# uft.py --- Unitary fourier transform
# Copyright (c) 2011, 2012, 2013 Franois Orieux <orieux@iap.fr>
# Permission is hereby granted, free of charge, to any person
# obtaining a copy of this software and associated documentation files
# (the "Software"), to deal in the Software without restriction,
# including without limitation the rights to use, copy, modify, merge,
# publish, distribute, sublicense, and/or sell copies of the Software,
# and to permit persons to whom the Software is furnished to do so,
# subject to the following conditions:
# The above copyright notice and this permission notice shall be
# included in all copies or substantial portions of the Software.
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.
# Commentary:
"""Function of unitary fourier transform and utilities
This module implement unitary fourier transform, that is ortho-normal
transform. They are specially usefull for convolution [1]: they
respect the parseval equality, the value of the null frequency is
equal to
.. math:: \frac{1}{\sqrt{n}} \sum_i x_i.
You must keep in mind that the transform are applied from the last
axes. this is a fftw convention for performance reason (c order
array). If you want more sofisticated use, you must use directly the
numpy.fft module.
References
----------
.. [1] B. R. Hunt "A matrix theory proof of the discrete convolution
theorem", IEEE Trans. on Audio and Electroacoustics,
vol. au-19, no. 4, pp. 285-288, dec. 1971
"""
# code:
import numpy as np
__copyright__ = "Copyright scikit-image team"
__license__ = "mit"
__version__ = "0.1.0"
__maintainer__ = "Francois Orieux"
__email__ = "orieux@iap.fr"
__url__ = ""
__keywords__ = "fft"
def _circshift(inarray, shifts):
"""Shift array circularly.
Circularly shifts the values in the array `a` by `s`
elements. Return a copy.
Parameters
----------
a : ndarray
The array to shift.
s : tuple of int
A tuple of integer scalars where the N-th element specifies the
shift amount for the N-th dimension of array `a`. If an element
is positive, the values of `a` are shifted down (or to the
right). If it is negative, the values of `a` are shifted up (or
to the left).
Returns
-------
y : ndarray
The shifted array (elements are copied)
Examples
--------
>>> circshift(np.arange(10), 2)
array([8, 9, 0, 1, 2, 3, 4, 5, 6, 7])
"""
# Initialize array of indices
idx = []
# Loop through each dimension of the input matrix to calculate
# shifted indices
for dim in range(inarray.ndim):
length = inarray.shape[dim]
try:
shift = shifts[dim]
except IndexError:
shift = 0 # no shift if not specify
# Lets start for fancy indexing. First we build the shifted
# index for dim k. It will be broadcasted to other dim so
# ndmin is specified
index = np.mod(np.array(range(length),
ndmin=inarray.ndim) - shift,
length)
# Shape adaptation
shape = np.ones(inarray.ndim)
shape[dim] = inarray.shape[dim]
index = np.reshape(index, shape)
idx.append(index.astype(int))
# Perform the actual conversion by indexing into the input matrix
return inarray[idx]
def ufftn(inarray, dim=None):
"""N-dim unitary Fourier transform
Parameters
----------
inarray : ndarray
The array to transform.
dim : int, optional
The `dim` last axis along wich to compute the transform. All
axes by default.
Returns
-------
outarray : array-like (same shape than inarray)
"""
if not dim:
dim = inarray.ndim
outarray = np.fft.fftn(inarray, axes=range(-dim, 0))
return outarray / np.sqrt(np.prod(inarray.shape[-dim:]))
def uifftn(inarray, dim=None):
"""N-dim unitary inverse Fourier transform
Parameters
----------
inarray : ndarray
The array to transform.
dim : int, optional
The `dim` last axis along wich to compute the transform. All
axes by default.
Returns
-------
outarray : array-like (same shape than inarray)
"""
if not dim:
dim = inarray.ndim
outarray = np.fft.ifftn(inarray, axes=range(-dim, 0))
return outarray * np.sqrt(np.prod(inarray.shape[-dim:]))
def urfftn(inarray, dim=None):
"""N-dim real unitary Fourier transform
This transform consider the Hermitian property of the transform on
real input
Parameters
----------
inarray : ndarray
The array to transform.
dim : int, optional
The `dim` last axis along wich to compute the transform. All
axes by default.
Returns
-------
outarray : array-like (the last dim as N / 2 + 1 lenght)
"""
if not dim:
dim = inarray.ndim
outarray = np.fft.rfftn(inarray, axes=range(-dim, 0))
return outarray / np.sqrt(np.prod(inarray.shape[-dim:]))
def uirfftn(inarray, dim=None):
"""N-dim real unitary Fourier transform
This transform consider the Hermitian property of the transform
from complex to real real input.
Parameters
----------
inarray : ndarray
The array to transform.
dim : int, optional
The `dim` last axis along wich to compute the transform. All
axes by default.
Returns
-------
outarray : array-like (the last dim as (N - 1) *2 lenght)
"""
if not dim:
dim = inarray.ndim
outarray = np.fft.irfftn(inarray, axes=range(-dim, 0))
return outarray * np.sqrt(np.prod(inarray.shape[-dim:-1]) *
(inarray.shape[-1] - 1) * 2)
def ufft2(inarray):
"""2-dim unitary Fourier transform
Compute the Fourier transform on the last 2 axes.
Parameters
----------
inarray : ndarray
The array to transform.
Returns
-------
outarray : array-like (same shape than inarray)
See Also
--------
uifft2, ufftn, urfftn
"""
return ufftn(inarray, 2)
def uifft2(inarray):
"""2-dim inverse unitary Fourier transform
Compute the inverse Fourier transform on the last 2 axes.
Parameters
----------
inarray : ndarray
The array to transform.
Returns
-------
outarray : array-like (same shape than inarray)
See Also
--------
uifft2, uifftn, uirfftn
"""
return uifftn(inarray, 2)
def urfft2(inarray):
"""2-dim real unitary Fourier transform
Compute the real Fourier transform on the last 2 axes. This
transform consider the Hermitian property of the transform from
complex to real real input.
Parameters
----------
inarray : ndarray
The array to transform.
Returns
-------
outarray : array-like (the last dim as (N - 1) *2 lenght)
See Also
--------
ufft2, ufftn, urfftn
"""
return urfftn(inarray, 2)
def uirfft2(inarray):
"""2-dim real unitary Fourier transform
Compute the real inverse Fourier transform on the last 2 axes.
This transform consider the Hermitian property of the transform
from complex to real real input.
Parameters
----------
inarray : ndarray
The array to transform.
Returns
-------
outarray : array-like (the last dim as (N - 1) *2 lenght)
See Also
--------
urfft2, uifftn, uirfftn
"""
return uirfftn(inarray, 2)
def image_quad_norm(inarray):
"""Return quadratic norm of images in Fourier space
This function detect if the image suppose the hermitian property.
Parameters
----------
inarray : array-like
The images are supposed to be in the last two axes
Returns
-------
norm : float
"""
# If there is an hermitian symmetry
if inarray.shape[-1] != inarray.shape[-2]:
return 2 * np.sum(np.sum(np.abs(inarray)**2, axis=-1), axis=-1) - \
np.sum(np.abs(inarray[..., 0])**2, axis=-1)
else:
return np.sum(np.sum(np.abs(inarray)**2, axis=-1), axis=-1)
def ir2tf(imp_resp, shape, dim=None, real=True):
"""Compute the transfer function of IR
This function make the necessary correct zero-padding, zero
convention, correct fft2 etc... to compute the transfer function
of IR. To use with unitary Fourier transform for the signal (ufftn
or equivalent).
Parameters
----------
imp_resp : ndarray
The impulsionnal responses.
shape : tuple of int
A tuple of integer corresponding to the target shape of the
tranfert function.
dim : int, optional
The `dim` last axis along wich to compute the transform. All
axes by default.
real : boolean (optionnal, default True)
If True, imp_resp is supposed real and the hermissian property
is used with rfftn Fourier transform.
Returns
-------
y : complex ndarray
The tranfert function of shape `shape`.
See Also
--------
ufftn, uifftn, urfftn, uirfftn
Notes
-----
The input array can be composed of multiple dimentionnal IR with
an arbitraru number of IR. The individual IR must be accesed
through first axes. The last `dim` axes of space definition. The
`dim` parameter must be specified to compute the transform only
along these last axes.
"""
if not dim:
dim = imp_resp.ndim
# Zero padding and fill
irpadded = np.zeros(shape)
irpadded[tuple([slice(0, s) for s in imp_resp.shape])] = imp_resp
# Circshift fo zero convention of the fft to avoid the phase
# problem. Work with odd and even size.
irpadded = _circshift(irpadded,
[-np.floor(s / 2)
if i >= imp_resp.ndim - dim
else 0
for i, s in enumerate(imp_resp.shape)])
if real:
return np.fft.rfftn(irpadded, axes=range(-dim, 0))
else:
return np.fft.fftn(irpadded, axes=range(-dim, 0))
def laplacian(ndim, shape):
"""Return the transfert function of the laplacian
Laplacian is the second order difference, on line and column.
Parameters
----------
ndim : int
The dimension of the laplacian
shape : tuple, shape
The support on which to compute the transfert function
Returns
-------
tf : array_like, complex
The transfert function
impr : array_like, real
The laplacian
"""
impr = np.zeros([3] * ndim)
for dim in range(ndim):
idx = tuple([slice(1, 2)] * dim +
[slice(None)] +
[slice(1, 2)] * (ndim - dim - 1))
impr[idx] = np.array([-1.0,
0.0,
-1.0]).reshape([-1 if i == dim else 1
for i in range(ndim)])
impr[([slice(1, 2)] * ndim)] = 2.0 * ndim
return ir2tf(impr, shape), impr