Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[ENH] XForm headers #202

Merged
merged 22 commits into from
Oct 10, 2017
Merged

Conversation

oesteban
Copy link
Member

@oesteban oesteban commented Oct 7, 2017

This PR adds utilities to copy xform matrices after registrations or using a new interface. Necessary for nipreps/fmriprep#710

@oesteban
Copy link
Member Author

oesteban commented Oct 8, 2017

Sorry Chris, I've asked your review but we need nipreps/fmriprep#710 to be passing before merging this.

@effigies
Copy link
Member

effigies commented Oct 8, 2017

Assume you mean #743? And looks like it's passing against the latest commit on this branch.

@oesteban
Copy link
Member Author

oesteban commented Oct 8, 2017

You're right, I meant nipreps/fmriprep#743. And yes, I've taken a look and it reports look good, registration steps don't seem to be broken.

@effigies
Copy link
Member

effigies commented Oct 8, 2017

Did it resolve the issue with a dataset exhibiting the problems in nipreps/fmriprep#710?

@oesteban
Copy link
Member Author

oesteban commented Oct 8, 2017

Sorry, I missed to say the most relevant bit of info. Yes, this #202 and nipreps/fmriprep#743 together do fix nipreps/fmriprep#710.

def _copyxform(ref_image, out_image, message=None):
# Read in reference and output
orig = nb.load(ref_image)
resampled = nb.load(out_image)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wonder if it would be a good idea to do some basic check, such as the two images having the same spatial dimensions. I think that's a reasonable precondition to copying an affine matrix, right?

sform, sform_code = orig.header.get_sform(coded=True)
header = resampled.header.copy()
header.set_qform(qform, int(qform_code))
header.set_sform(sform, int(sform_code))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The strong assumption being made here is that we're resampling so that, not only are the images in the same RAS space, but in the same IJK (voxel) space, as well. Is this just how ANTs always does things (especially the tools we're "fixing"), or do they ever do an affine update, so that the images are aligned in RAS, but not necessarily in IJK?

This may be the same issue as checking whether they have the same dimensions, so at least by adding that check, we'll reduce the possibility that we are copying an affine to an un-resampled image.

Also, perhaps we should make xform_code an optional parameter. Most of the time with registration, we'll want code 2 (#define NIFTI_XFORM_ALIGNED_ANAT 2 /! Coordinates aligned to another file's, or to anatomical "truth".), while the anatomical image may still have a qform/sform code of 1.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If I understand, the place we should make sure we set the appropriate xform code 2 is at the point we generate the reference for the bold_preproc spaces. All others should inherit the original xforms. Correct?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actually that code should be 4 when the target is MNI

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would say 90% of the time you're using Registration or ApplyTransform, you're resampling from one image to another, so it should be 2, unless (as you say) it should be 4. There might be cases where you're really not moving, and I think the only example I can think of is the SyN-SDC transform where we go out of our way to bring the reference into the moving image's space to start with.

Also, I don't think you need the CopyXForm interface at all (at least for fmriprep right now). See nipreps/fmriprep#743 (comment).

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Given the precedence (sform > qform > base affine), perhaps the logic we want is:

xform = orig.affine  # nibabel will pick the best affine
_, qform_code = orig.header.get_qform(coded=True)
_, sform_code = orig.header.get_sform(coded=True)

xform_code = sform_code if sform_code > 0 else qform_code
if xform_code == 1:
    xform_code = 2
# Keep 0, 2, 3, 4 unchanged
header.set_qform(xform, xform_code)
header.set_sform(xform, xform_code)

That way, as long as there's a legitimate affine matrix, both the qform and sform will be updated (though qform will drop any shear). Then software that handles sforms badly will have the best hints we can give.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have added your code as follows: first, I'd move GenerateSamplingReference from fmriprep to niworkflows:

https://github.com/poldracklab/niworkflows/blob/510881813e331c7febe3b1ca43113a999cf3ac70/niworkflows/interfaces/utils.py#L117-L143

and this is how it looks with some modifications:

https://github.com/poldracklab/niworkflows/blob/510881813e331c7febe3b1ca43113a999cf3ac70/niworkflows/interfaces/utils.py#L168-L206

As I said, I think registrations should not change (or fix) the xform matrices. We need to make sure that they are correct when a) conforming inputs and b) generating sampling references. Any other process that happens in a given space should just forward the xforms. That's why I'd be reluctant to replace the copy with a copy&fix xforms.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay.

header.set_qform(qform, int(qform_code))
header.set_sform(sform, int(sform_code))
for quatern in ['b', 'c', 'd']:
header['quatern_' + quatern] = orig.header['quatern_' + quatern]
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm pretty sure this is done in set_qform.

newpath=runtime.cwd)
# Copy and replace header
shutil.copy(self.inputs.in_file, out_name)
_copyxform(self.inputs.hdr_file, out_name, message='CopyXForm')
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe message='CopyXForm; niworkflows v%s' % __version__?


class CopyXForm(SimpleInterface):
"""
Copy a header from the `hdr_file` to `out_file` with data drawn from
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We're copying the qform/sform matrices from hdr_file, not the whole header.

resampled = nb.load(out_image)
orig = nb.load(ref_image)

if orig.affine != resampled.affine:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should assume the affines might not match; that's why we're copying. I'd say:

if orig.shape[:3] != resampled.shape[:3]:

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The problem on nipreps/fmriprep#710 was exactly that the affines matched, but at certain steps, both ANTs and FSL would change the x-forms. But the affines still matched.

This is also to answer the previous comment: we need to make sure that xforms are not changed by accident. The rationale behind this and the accompanying PR in fmriprep is that:

  1. When moving within the same space (SDC, head motion correction, SBRef-EPI), we keep the original xforms (even if they are wrong in the original files).

  2. When moving to different space (anatomical or MNI), we are generating an image taken as reference space. So it is this point when we need to make sure that the xform code is the appropriate one. Since we pass on this reference image to the registration interface, the xform matrices should, indeed, be copied over the resulting images. Same for ApplyTransforms.

Bottomline: no interface should change xforms unless we are certain we want to do that. Only GenerateSamplingReference should make these changes.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the affines match, what are we correcting? Is it that the sforms matched but the qforms didn't?

sform, sform_code = orig.header.get_sform(coded=True)
header = resampled.header.copy()
header.set_qform(qform, int(qform_code))
header.set_sform(sform, int(sform_code))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would say 90% of the time you're using Registration or ApplyTransform, you're resampling from one image to another, so it should be 2, unless (as you say) it should be 4. There might be cases where you're really not moving, and I think the only example I can think of is the SyN-SDC transform where we go out of our way to bring the reference into the moving image's space to start with.

Also, I don't think you need the CopyXForm interface at all (at least for fmriprep right now). See nipreps/fmriprep#743 (comment).

if out_file is not None and out_file:
_copyxform(self.inputs.moving_image[0],
os.path.abspath(out_file),
message=self.__class__.__name__)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we add the niworkflows version to the messages in this file, as well?

resampled = nb.load(out_image)
orig = nb.load(ref_image)

if np.any(orig.affine != resampled.affine):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we do want to check affines, I'd prefer this check:

if not np.allclose(orig.affine, resampled.affine):

Copy link
Member

@effigies effigies left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay. I think I'm happy with this for now. I think perhaps we want to revisit at some point (possibly related to nipreps/fmriprep#475) all of the places we might want to fix an affine, but this is a clear improvement.

if out_file is not None and out_file:
_copyxform(self.inputs.fixed_image[0],
os.path.abspath(out_file),
message=self.__class__.__name__)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, one more place to update message.

import nilearn.image as nli

from .. import __version__
from ..nipype import logging
from niworkflows.nipype.utils.filemanip import fname_presuffix
from niworkflows.nipype.utils.misc import normalize_mc_params
from niworkflows.nipype.interfaces.base import (
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While we're at it, can we make these ..nipype imports?

resampled.header.set_qform(xform, xform_code)
resampled.header.set_sform(xform, xform_code)
resampled.header['descrip'] = 'reference image generated by %s.' % (
message or '(unknown software)')
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So all of this... it's just updating the qform_code to match the sform_code, and setting descrip field. Because resampled is being generated by nilearn/nibabel, qform already == sform (modulo shearing):

https://github.com/nipy/nibabel/blob/8c1d0bc4bd9cd757f9b9658547f0e2af55e8ca6c/nibabel/nifti1.py#L1789-L1795

It's not wrong, just I'm not sure what we're buying here.

Copy link
Member Author

@oesteban oesteban Oct 9, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see. You are right, this is not buying much. I'd prefer to keep it though, to have your code handy when we need it. Still, the interface now can manually set a different code, which will be useful when we actually start using it.

sform, sform_code = orig.header.get_sform(coded=True)
header = resampled.header.copy()
header.set_qform(qform, int(qform_code))
header.set_sform(sform, int(sform_code))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay.

@oesteban
Copy link
Member Author

oesteban commented Oct 9, 2017

<spooky voice>tests are passing...</spooky voice>

@effigies effigies merged commit c550524 into nipreps:master Oct 10, 2017
@oesteban oesteban deleted the enh/HeaderSafeAntsApplyTransform branch October 10, 2017 20:47
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants