The pFIRE Algorithm¶
Algorithm Description¶
pFIRE operates by iteratively solving a matrix equation to find updates to the deformation map that minimizes the difference between the fixed and the moved image. After each iteration the original moved image is warped using the updated map to produce a new moved image that is closer to the fixed image. The original moved image is used each time to minimize the accumulation of errors from warping the same image multiple times.
Initially the displacement map that is used is coarse, having only 3 nodes per dimension, but as the registration converges at each resolution the map is then refined, doubling the number of nodes per dimension until the desired resolution is reached.
The Registration Equation¶
The basic equation that pFIRE attempts to solve relates the fixed image \(\vec{f}\) to the moved image \(\vec{m}\) via the displacement map \(\vec{a}\). This can be written in one of two forms, depending on if we are “pushing” the moved image onto the fixed image or pulling the fixed onto the moved image.
Taylor expanding these expressions and truncating to first order allows us to separate the displacement vector from the source image by introducing a gradient term:
Note that these expressions are valid only in the case that the displacement \(\vec{a}(\vec{x})\) is small, however, the scheme will be applied iteratively to converge upon the correct result.
Combining these two expressions yields a single registration equation to be solved for \(\vec{a}(\vec{x})\) which includes the information from both images:
The above registration equation has the images and map in the form on continuous functions, however, when solving numerically we have images composed of pixels and a map represented as a series of node points. Information between these points can then be found by (linear) interpolation. Typically there will also be far fewer map nodes than image pixels both for practical reasons of computation time and mathematical reasons of resistance to noise and simply having sufficient information to fully constrain the problem.
The net result of this is that we require some kind of interpolation method to map contributions from image pixels to their corresponding map nodes. pFIRE uses linear interpolation, with each pixel contributing to its nearest 4 (or 8 in 3D) map nodes, with linear interpolation coefficients given by
Here upper latin indices represent the image pixel indices and the greek indices represent the map node indices. The object \(x^n_{ijk}\) is the position in spatial dimension \(n\) of the pixel at position (i,j,k) in the image, and similarly \(x^n_{\mu\nu\omega}\) describes the position of a map node.
This can then be used, along with a gradient information tensor \(I\) and a displacement (map) tensor \(A\), to construct a discrete form of the registration equation (using Einstein summation convention):
where the index \(\alpha\) refers to the four degrees of freedom (\(x, y, z, s\)), the latin index triplets (\(x y z\) and \(i j k\)) refer to image pixel indices and the greek index triplet (\(\mu \nu \omega\)) refers to map node indices.
The form of the gradient information tensor \(I\) is diagonal, with elements given by
where \(\nabla_\alpha\) indicates differentiation with respect to \(x, y\ \text{or}\ z\) for \(\alpha=1,2,3\) respectively. Note that this is essentially a concatenation of four diagonal tensors, and in the subsequent steps will become a concatentation of four diagonal matrices.
The complexity of implementation may be eased by electing to explicitly flatten the indices \(x,y,z\) and \(\mu,\nu,\omega\). In the case of pFIRE indexing is performed such that the x index varies fastest - this is the same indexing scheme as used by PETSc:
where \((X,Y,Z)\) is the size of the mesh.
This allows the problem to be expressed in the form of a matrix equation:
where \(\bar{F}\) and \(\bar{M}\) are column vectors containing the fixed and moved image intensities, \(\mathbf{{G}}_\alpha\) are the square, diagonal submatrices for each dimension of the gradient information, \(\mathbf{{\phi}}\) is the same submatrix for each dimension, and \(A_\alpha\) are column matrices containing the nodal displacements for each dimension of the map.
The matrix \(T\) is defined here as the product of the matrices \(\mathbf{{G}}\) and \(\mathbf{{\Phi}}\), and it is, in general, non-square. We therefore create a soluable matrix equation from this by multiplying both sides by the transpose \(\mathbf{{T}}^t\),
yielding an equation with square matrix \(\mathbf{{T}}^t\mathbf{{T}}\) which may be solved by standard numerical methods.
Unfortunately \(\mathbf{{T}}^t\mathbf{{T}}\) is typically singular and so we must apply some kind of regularisation to make the problem soluable. In this case we use the method of Tikhonov regularisation, which includes an additional smoothing term in the problem. In this case the Laplacian matrix is chosen, adding minimization of the second derivative of the map as the extra constraint (\(\mathbf{{L}}\bar{A} = 0\) and the problem becomes,
This simplifies to
where \(\lambda\) is a free parameter which is chosen to minimize the condition number of the regularised matrix. Adding this additional Laplacian constraint ensures that the smoothest solution to the problem is chosen.
Finally, because the algorithm is iterative, we calculate a new mapping at each step and add it to the previously accumulated displacements to refine the overall solution. In many cases, we may want to constrain the overall displacement to be maximally smooth, thus our additional constraint becomes
or
So the equation becomes
This simplifies to
This form of the equation “remembers” the value of the map from the previous iteration and attempts to enforce global smoothing on the final result and is referred to as the “memory term”. This method of smoothing is useful for registering images where the displacement is expected to be continuous and smooth, for example in the case of registration of multimodal images of the same structure. In constrast, this option should be disabled in the case that image features are expected to move relative to each other, for example in cell-tracking applications.
Implementation Details¶
Calculating \(\mathbf{{T}}^t\mathbf{{T}}\)¶
The naive implementation of registration equation requires explicitly constructing the matrix \(\mathbf{{T}}\) before calculating the product \(\mathbf{{T}}^t\mathbf{{T}}\). This is not an efficient use of memory, however, especially since the matrix \(\mathbf{{T}}\) is much larger than the final product and is only an intermediate value in computing \(\mathbf{{T}}^t(\bar{F} - \bar{M})\) and \(\begin{bmatrix}\mathbf{{T}}^t\mathbf{{T}} + \lambda\mathbf{{L}}^t\mathbf{{L}}\end{bmatrix}\).
Because of this, pFIRE computes the matrix \(\mathbf{{T}}^t\mathbf{{T}}\) and the vector \(\mathbf{{T}}^t(\bar{F} - \bar{M})\) directly. The structure of the final matrix and vector are distributed between the various ranks such that each rank computes an equal number of components of the final data structure. For each matrix or vector element, the location of all the required image or gradient pixels is determiend and then required values that are not local to the rank are communicated using MPI_Alltoall with all ranks communicating at once.
Implementation of the computation can be made more efficient by understanding the structure of the \(\mathbf{{G}}\) and \(\mathbf{{\Phi}}\) matrices, as prior knowledge of the zero-patterns in these matrices can make calculation of the final matrix \(\mathbf{{T}}^t\mathbf{{T}}\) much more efficient. Further, since the \(\mathbf{{G}}\) matrix is diagonal we know that the zero pattern of its product with any matrix will match that of the other matrix, therefore we need only consider the zero-pattern of the interpolation matrix \(\mathbf{{\Phi}}\).
Revisiting the definition of \(\mathbf{\phi}\) (for flattened indices) and a single dimension:
with \(\mu\) the map indices and \(i\) the image indices. So when calculating
the \(\mathbf{G}\) matrices function as Kronecker deltas for the purposes of discerning the zero-pattern, and so we can determine a zero-pattern matrix \(\mathbf{Z}\) for the matrix product \(\mathbf{\phi}^t\mathbf{\phi}\)
From this we see that there will only be a non-zero entry in the map if the image pixel at point \(i\) is a contributor to the map node at both points \(\mu\) and \(\nu\). Because we are using nearest neighbor interpolation, this means there are two classes of non-zero entry in \(\mathbf{\phi}\): diagonal entries, for which \(\mu = \nu\); and off diagonal entries, for which \(\mu = \nu \pm \{1,U,U*V\}\) (where \(U\) and \(U*V\) are the two multiplicative factors used in flattening the indexing). This gives a total of 9 non-zero entries per row. If the domain splitting is carefully chosen, this can lead to an optimal communication pattern where information need only be exchanged with nearest neighbour ranks, rather than requiring all-to-all communication. In practice this optimal scheme is not implemented by pFIRE since it requires fixing the problem size per processor. Instead dynamic communication patterns are used to communicate only the required data regardless of its location.
Improving Matrix Conditioning¶
The matrix as constructed, even with the addition of the laplacian is suboptimally conditioned. In order to improve the conditioning further, a preconditioning step is applied to the calculated matrix \(\mathbf{{T}}^t\mathbf{{T}}\), predicated on the fact that the matrix is symmetric and positive definite, and that the intensity values calculated for the map will not be used to update the map, but are relevant purely for the correct calculation of the displacement components.
The average of the diagonal elements is calculated for the displacement components and for the intensity components, and the intensity components rescaled by multiplying with a diagonal matrix with appropriate values on the diagonal, such that the average of the diagonal intensity elements is equal to the average for the displacement elements.
Calculating Lambda¶
Prior to calculating \(\lambda\) a “premultiplication factor” \(\chi\) is calculated and applied to rescale the laplacian matrix to make the average of the diagonal elements of \(\mathbf{{L}}^t\mathbf{{L}}\) equal to the average of the diagonal elements of \(\mathbf{{T}}^t\mathbf{{T}}\).
The optimal value of \(\lambda\) is then found by minimising the condition number of the matrix
In general this could be done by various optimisation strategies, however, in pFIRE the approach taken is to minimize the required number of condition number calculations as they are computationall expensive. It has been empirically observed that the behaviour of the condition number with increasing \(\lambda\) is to initially decrease to a wide minimum region before increasing again. Therefore, a reasonable value for lambda can be found by evaluating the condition number at three points, one expected to be to well below the optimum value, one expected to be close to it and one expected to be well above. Provided the middle value produces a lower condition number than the edges, a quadratic fit to these points will then provide a reasonable estimation of the optimum lambda. In the case that the middle point is not lower, the gradient of the fitted line will inform whether to increase or decrease the values of the search points and try again.
Solving the Registration Equation¶
The registration equation is solved using the Krylov subspace solver routines provided by PETsC.
Warping the Image¶
The image is warped by the “pulling method”. For a given pixel in the target image, the map value is interpolated to the pixel location and used to determine the location of the source pixel in the source image. The value at this point is then calculated, using MPI_AllToAll to communicate the data from other ranks as needed, and inserted into the target image.