Introduction#
This package contains a large collection of forward operators appearing in imaging applications. The acquisition models are of the form
where \(x\in\xset\) is an image, \(y\in\yset\) are the measurements, \(A:\xset\mapsto \yset\) is a deterministic (linear or non-linear) operator capturing the physics of the acquisition and \(N:\yset\mapsto \yset\) is a mapping which characterizes the noise affecting the measurements.
All forward operators inherit the structure of the deepinv.physics.Physics
class.
They are torch.nn.Module
which can be called with the forward
method.
>>> import torch
>>> import deepinv as dinv
>>> # load an inpainting operator that masks 50% of the pixels and adds Gaussian noise
>>> physics = dinv.physics.Inpainting(mask=.5, tensor_size=(1, 28, 28),
... noise_model=dinv.physics.GaussianNoise(sigma=.05))
>>> x = torch.rand(1, 1, 28, 28) # create a random image
>>> y = physics(x) # compute noisy measurements
>>> y2 = physics.A(x) # compute the A operator (no noise)
Linear operators#
Linear operators \(A:\xset\mapsto \yset\) inherit the structure of the deepinv.physics.LinearPhysics
class.
They have important specific properties such as the existence of an adjoint \(A^{\top}:\yset\mapsto \xset\).
Linear operators with a closed-form singular value decomposition are defined via deepinv.physics.DecomposablePhysics
,
which enables the efficient computation of their pseudo-inverse and regularized inverse.
Composition and linear combinations of linear operators is still a linear operator.
>>> import torch
>>> import deepinv as dinv
>>> # load a CS operator with 300 measurements, acting on 28 x 28 grayscale images.
>>> physics = dinv.physics.CompressedSensing(m=300, img_shape=(1, 28, 28))
>>> x = torch.rand(1, 1, 28, 28) # create a random image
>>> y = physics(x) # compute noisy measurements
>>> y2 = physics.A(x) # compute the linear operator (no noise)
>>> x_adj = physics.A_adjoint(y) # compute the adjoint operator
>>> x_dagger = physics.A_dagger(y) # compute the pseudo-inverse operator
>>> x_prox = physics.prox_l2(x, y, .1) # compute a regularized inverse
More details can be found in the doc of each class.
Parameter-dependent operators#
Many (linear or non-linear) operators depend on (optional) parameters \(\theta\) that describe the imaging system, ie
\(y = \noise{\forw{x, \theta}}\) where the forward
method can be called with a dictionary of parameters as an extra input.
The explicit dependency on \(\theta\) is often useful for blind inverse problems, model identification,
imaging system optimization, etc. The following example shows how operators and their parameter can be instantiated and called as:
>>> import torch
>>> from deepinv.physics import Blur
>>> x = torch.rand((1, 1, 16, 16))
>>> theta = torch.ones((1, 1, 2, 2)) / 4 # a basic 2x2 averaging filter
>>> # default usage
>>> physics = Blur(filter=theta) # we instantiate a blur operator with its convolution filter
>>> y = physics(x)
>>> theta2 = torch.randn((1, 1, 2, 2)) # a random 2x2 filter
>>> physics.update_parameters(filter=theta2)
>>> y2 = physics(x)
>>>
>>> # A second possibility
>>> physics = Blur() # a blur operator without convolution filter
>>> y = physics(x, filter=theta) # we define the blur by specifying its filter
>>> y = physics(x) # now, the filter is well-defined and this line does the same as above
>>>
>>> # The same can be done by passing in a dictionary including 'filter' as a key
>>> physics = Blur() # a blur operator without convolution filter
>>> dict_params = {'filter': theta, 'dummy': None}
>>> y = physics(x, **dict_params) # # we define the blur by passing in the dictionary
Physics Generators#
We provide some parameters generation methods to sample random parameters’ \(\theta\).
Physics generators inherit from the deepinv.physics.generator.PhysicsGenerator
class:
>>> import torch
>>> import deepinv as dinv
>>>
>>> x = torch.rand((1, 1, 8, 8))
>>> physics = dinv.physics.Blur(filter=dinv.physics.blur.gaussian_blur(.2))
>>> y = physics(x) # compute with Gaussian blur
>>> generator = dinv.physics.generator.MotionBlurGenerator(psf_size=(3, 3))
>>> params = generator.step(x.size(0)) # params = {'filter': torch.tensor(...)}
>>> y1 = physics(x, **params) # compute with motion blur
>>> assert not torch.allclose(y, y1) # different blurs, different outputs
>>> y2 = physics(x) # motion kernel is stored in the physics object as default kernel
>>> assert torch.allclose(y1, y2) # same blur, same output
If we want to generate both a new physics and noise parameters, it is possible to sum generators as follows:
>>> mask_generator = dinv.physics.generator.SigmaGenerator() \
... + dinv.physics.generator.RandomMaskGenerator((32, 32))
>>> params = mask_generator.step(batch_size=4)
>>> print(sorted(params.keys()))
['mask', 'sigma']
Tip
It is also possible to mix generators of physics parameters through the
deepinv.physics.generator.GeneratorMixture
class.
Combining Physics#
It is possible to stack and compose multiple physics operators into a single operator.
Stacking operators \(A_1\) and \(A_2\) into a single operator
can be done with deepinv.physics.stack()
. The stacked operator is
>>> import torch
>>> import deepinv as dinv
>>> x = torch.rand((1, 1, 8, 8))
>>> physics1 = dinv.physics.BlurFFT(img_size=(1, 8, 8), filter=dinv.physics.blur.gaussian_blur(.2))
>>> physics2 = dinv.physics.Downsampling(img_size=(1, 8, 8), factor=2)
>>> physics3 = dinv.physics.stack(physics1, physics2)
>>> physics3 = physics1.stack(physics2) # equivalent to the previous line
>>> y = physics3(x) #
>>> print(y[0].shape)
torch.Size([1, 1, 8, 8])
>>> print(y[1].shape)
torch.Size([1, 1, 4, 4])
>>> physics4 = physics3.stack(physics1) # add a new operator to the stack
>>> len(physics4)
3
The measurements are stored as deepinv.utils.TensorList
objects, which can be accessed by index
(see the TensorList user guide for more details).
The resulting stacked operator is a deepinv.physics.StackedPhysics
object, and has some useful
methods:
>>> print(physics3[0](x).shape) # access the first operator only
torch.Size([1, 1, 8, 8])
>>> print(physics3[1](x).shape) # access the second operator only
torch.Size([1, 1, 4, 4])
Tip
See also the custom classes deepinv.optim.StackedPhysicsDataFidelity
and deepinv.loss.StackedPhysicsLoss
provide easy ways to build data fidelity terms and self-supervised losses with stacked operators.
Composing operators \(A_1\) and \(A_2\) into a single operator
can be done by multiplying the operators:
>>> import torch
>>> import deepinv as dinv
>>> x = torch.rand((1, 1, 8, 8))
>>> physics1 = dinv.physics.Downsampling(img_size=(1, 8, 8), factor=2)
>>> physics2 = dinv.physics.BlurFFT(img_size=(1, 4, 4), filter=dinv.physics.blur.gaussian_blur(.2))
>>> physics = physics2 * physics1
>>> y = physics(x) # equivalent to y = physics2(physics1.A(x))
>>> print(y.shape)
torch.Size([1, 1, 4, 4])