Skip to content

Commit

Permalink
Merge pull request #302 from nipreps/enh/more-verbose-debug-mode
Browse files Browse the repository at this point in the history
MAINT: Housekeeping and more verbose debugging outputs
  • Loading branch information
oesteban authored Nov 17, 2022
2 parents 69abb77 + dacfb63 commit fd53565
Show file tree
Hide file tree
Showing 4 changed files with 77 additions and 11 deletions.
23 changes: 23 additions & 0 deletions sdcflows/interfaces/bspline.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ class _BSplineApproxInputSpec(BaseInterfaceInputSpec):
usedefault=True,
desc="generate a field, extrapolated outside the brain mask",
)
debug = traits.Bool(False, usedefault=True, desc="generate extra assets for debugging")


class _BSplineApproxOutputSpec(TraitedSpec):
Expand Down Expand Up @@ -128,11 +129,22 @@ def _run_interface(self, runtime):
# Load in the fieldmap
fmapnii = nb.load(self.inputs.in_data)
data = fmapnii.get_fdata(dtype="float32")

# Generate the output naming base
out_name = fname_presuffix(
self.inputs.in_data, suffix="_field", newpath=runtime.cwd
)
# Create a copy of the header for use below
hdr = fmapnii.header.copy()
hdr.set_data_dtype("float32")

mask = (
nb.load(self.inputs.in_mask).get_fdata() > 0
if isdefined(self.inputs.in_mask)
else np.ones_like(data, dtype=bool)
)

# Massage bs_spacing input
bs_spacing = [np.array(sp, dtype="float32") for sp in self.inputs.bs_spacing]

# Recenter the fieldmap
Expand Down Expand Up @@ -334,6 +346,7 @@ class _TransformCoefficientsInputSpec(BaseInterfaceInputSpec):
)
fmap_ref = File(exists=True, mandatory=True, desc="the fieldmap reference")
transform = File(exists=True, mandatory=True, desc="rigid-body transform file")
fmap_target = File(exists=True, desc="the distorted EPI target (feed to set debug mode on)")


class _TransformCoefficientsOutputSpec(TraitedSpec):
Expand All @@ -356,6 +369,10 @@ def _run_interface(self, runtime):
level,
self.inputs.fmap_ref,
self.inputs.transform,
fmap_target=(
self.inputs.fmap_target if isdefined(self.inputs.fmap_target)
else None
),
)
out_file = fname_presuffix(
level, suffix="_space-target", newpath=runtime.cwd
Expand Down Expand Up @@ -512,6 +529,12 @@ def _fix_topup_fieldcoeff(in_coeff, fmap_ref, pe_dir, out_file=None):
header = coeffnii.header.copy()
header.set_qform(newaff, code=1)
header.set_sform(newaff, code=1)
header["cal_max"] = max((
abs(np.asanyarray(coeffnii.dataobj).min()),
np.asanyarray(coeffnii.dataobj).max(),
))
header["cal_min"] = - header["cal_max"]
header.set_intent("estimate", tuple(), name="B-Spline coefficients")

# Write out fixed (generalized) coefficients
coeffnii.__class__(coeffnii.dataobj, newaff, header).to_filename(out_file)
Expand Down
54 changes: 44 additions & 10 deletions sdcflows/transform.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,6 @@
from bids.utils import listify

from niworkflows.interfaces.nibabel import reorient_image
from sdcflows.utils.tools import ensure_positive_cosines


def _clear_mapped(instance, attribute, value):
Expand All @@ -48,7 +47,7 @@ class B0FieldTransform:

coeffs = attr.ib(default=None)
"""B-Spline coefficients (one value per control point)."""
xfm = attr.ib(default=nt.linear.Affine(), on_setattr=_clear_mapped)
xfm = attr.ib(default=None, on_setattr=_clear_mapped)
"""A rigid-body transform to prepend to the unwarping displacements field."""
mapped = attr.ib(default=None, init=False)
"""
Expand All @@ -74,6 +73,9 @@ def fit(self, spatialimage):
if isinstance(spatialimage, (str, bytes, Path)):
spatialimage = nb.load(spatialimage)

# Initialize xform (or set identity)
xfm = self.xfm if self.xfm is not None else nt.linear.Affine()

if self.mapped is not None:
newaff = spatialimage.affine
newshape = spatialimage.shape
Expand All @@ -88,9 +90,9 @@ def fit(self, spatialimage):

# Generate tensor-product B-Spline weights
for level in listify(self.coeffs):
self.xfm.reference = spatialimage
xfm.reference = spatialimage
moved_cs = level.__class__(
level.dataobj, self.xfm.matrix @ level.affine, level.header
level.dataobj, xfm.matrix @ level.affine, level.header
)
wmat = grid_bspline_weights(spatialimage, moved_cs)
weights.append(wmat)
Expand All @@ -103,9 +105,12 @@ def fit(self, spatialimage):
)

# Cache
self.mapped = nb.Nifti1Image(vsm, spatialimage.affine, None)
self.mapped.header.set_intent("estimate", name="Voxel shift")
self.mapped.header.set_xyzt_units(*spatialimage.header.get_xyzt_units())
hdr = spatialimage.header.copy()
hdr.set_intent("estimate", name="Voxel shift")
hdr.set_data_dtype("float32")
hdr["cal_max"] = max((abs(vsm.min()), vsm.max()))
hdr["cal_min"] = - hdr["cal_max"]
self.mapped = nb.Nifti1Image(vsm, spatialimage.affine, hdr)
return True

def apply(
Expand Down Expand Up @@ -153,6 +158,8 @@ def apply(
The data imaged after resampling to reference space.
"""
from sdcflows.utils.tools import ensure_positive_cosines

# Ensure the fmap has been computed
if isinstance(spatialimage, (str, bytes, Path)):
spatialimage = nb.load(spatialimage)
Expand All @@ -177,6 +184,7 @@ def apply(
).reshape(3, -1)
else:
# Map coordinates from reference to time-step
self.xfm.reference = spatialimage
hmc_xyz = self.xfm.map(self.xfm.reference.ndcoords.T)
# Convert from RAS to voxel coordinates
voxcoords = (
Expand Down Expand Up @@ -323,6 +331,11 @@ def disp_to_fmap(xyz_nii, ro_time, pe_dir, itk_format=True):
fmap_nii = nb.Nifti1Image(vsm / scale_factor, xyz_nii.affine)
fmap_nii.header.set_intent("estimate", name="Delta_B0 [Hz]")
fmap_nii.header.set_xyzt_units("mm")
fmap_nii.header["cal_max"] = max((
abs(np.asanyarray(fmap_nii.dataobj).min()),
np.asanyarray(fmap_nii.dataobj).max(),
))
fmap_nii.header["cal_min"] = - fmap_nii.header["cal_max"]
return fmap_nii


Expand Down Expand Up @@ -413,12 +426,33 @@ def grid_bspline_weights(target_nii, ctrl_nii):
return kron(kron(wd[0], wd[1]), wd[2])


def _move_coeff(in_coeff, fmap_ref, transform):
def _move_coeff(in_coeff, fmap_ref, transform, fmap_target=None):
"""Read in a rigid transform from ANTs, and update the coefficients field affine."""
xfm = nt.linear.Affine(
nt.io.itk.ITKLinearTransform.from_filename(transform).to_ras(),
reference=fmap_ref,
)
coeff = nb.load(in_coeff)
newaff = xfm.matrix @ coeff.affine
return coeff.__class__(coeff.dataobj, newaff, coeff.header)
hdr = coeff.header.copy()

if fmap_target is not None: # Debug mode: generate fieldmap reference
nii_target = nb.load(fmap_target)
debug_ref = (~xfm).apply(fmap_ref, reference=nii_target)
debug_ref.header.set_qform(nii_target.affine, code=1)
debug_ref.header.set_sform(nii_target.affine, code=1)
debug_ref.to_filename(Path() / "debug_fmapref.nii.gz")

# Generate a new transform
newaff = np.linalg.inv(np.linalg.inv(coeff.affine) @ (~xfm).matrix)

# Prepare output file
hdr.set_qform(newaff, code=1)
hdr.set_sform(newaff, code=1)

# Make it easy on viz software to render proper range
hdr["cal_max"] = max((
abs(np.asanyarray(coeff.dataobj).min()),
np.asanyarray(coeff.dataobj).max(),
))
hdr["cal_min"] = - hdr["cal_max"]
return coeff.__class__(coeff.dataobj, newaff, hdr)
4 changes: 3 additions & 1 deletion sdcflows/workflows/apply/correction.py
Original file line number Diff line number Diff line change
Expand Up @@ -103,7 +103,9 @@ def init_unwarp_wf(omp_nthreads=1, debug=False, name="unwarp_wf"):

rotime = pe.Node(GetReadoutTime(), name="rotime")
rotime.interface._always_run = debug
resample = pe.Node(ApplyCoeffsField(num_threads=omp_nthreads), name="resample")
resample = pe.Node(ApplyCoeffsField(
num_threads=omp_nthreads if not debug else 1
), name="resample")
merge = pe.Node(MergeSeries(), name="merge")
average = pe.Node(RobustAverage(mc_method=None), name="average")

Expand Down
7 changes: 7 additions & 0 deletions sdcflows/workflows/apply/registration.py
Original file line number Diff line number Diff line change
Expand Up @@ -156,4 +156,11 @@ def init_coeff2epi_wf(
])
# fmt: on

if debug:
# fmt: off
workflow.connect([
(inputnode, map_coeff, [("target_ref", "fmap_target")]),
])
# fmt: on

return workflow

0 comments on commit fd53565

Please sign in to comment.