Skip to content

Commit 6e70c02

Browse files
authored
Merge pull request #186 from nipy/fix/optimize-afni-toras
FIX: Inefficient iterative reloading of reference and moving images
2 parents 1674e86 + 01cfcbc commit 6e70c02

6 files changed

+33
-6
lines changed

nitransforms/io/afni.py

+21-5
Original file line numberDiff line numberDiff line change
@@ -130,9 +130,17 @@ class AFNILinearTransformArray(BaseLinearTransformList):
130130

131131
def to_ras(self, moving=None, reference=None):
132132
"""Return a nitransforms' internal RAS matrix."""
133-
return np.stack(
134-
[xfm.to_ras(moving=moving, reference=reference) for xfm in self.xforms]
135-
)
133+
134+
pre_rotation = post_rotation = np.eye(4)
135+
if reference is not None and _is_oblique(ref_aff := _ensure_image(reference).affine):
136+
pre_rotation = _cardinal_rotation(ref_aff, True)
137+
if moving is not None and _is_oblique(mov_aff := _ensure_image(moving).affine):
138+
post_rotation = _cardinal_rotation(mov_aff, False)
139+
140+
return np.stack([
141+
post_rotation @ (xfm.to_ras() @ pre_rotation)
142+
for xfm in self.xforms
143+
])
136144

137145
def to_string(self):
138146
"""Convert to a string directly writeable to file."""
@@ -144,14 +152,22 @@ def to_string(self):
144152
if line.strip()
145153
]
146154
strings += lines
147-
return "\n".join(strings)
155+
return "\n".join(strings + [""])
148156

149157
@classmethod
150158
def from_ras(cls, ras, moving=None, reference=None):
151159
"""Create an ITK affine from a nitransform's RAS+ matrix."""
152160
_self = cls()
161+
162+
pre_rotation = post_rotation = np.eye(4)
163+
164+
if reference is not None and _is_oblique(ref_aff := _ensure_image(reference).affine):
165+
pre_rotation = _cardinal_rotation(ref_aff, False)
166+
if moving is not None and _is_oblique(mov_aff := _ensure_image(moving).affine):
167+
post_rotation = _cardinal_rotation(mov_aff, True)
168+
153169
_self.xforms = [
154-
cls._inner_type.from_ras(ras[i, ...], moving=moving, reference=reference)
170+
cls._inner_type.from_ras(post_rotation @ ras[i, ...] @ pre_rotation)
155171
for i in range(ras.shape[0])
156172
]
157173
return _self
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
affine-RAS.afni-array
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
affine-RAS.afni-array
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# 3dvolreg matrices (DICOM-to-DICOM, row-by-row):
2+
0.999999 -0.000999999 -0.001 -4 0.00140494 0.621609 0.783327 -2 -0.000161717 -0.783327 0.62161 -1
3+
0.999999 -0.000999999 -0.001 -4 0.00140494 0.621609 0.783327 -2 -0.000161717 -0.783327 0.62161 -1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
affine-RAS.afni-array

nitransforms/tests/test_io.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -180,7 +180,7 @@ def test_LT_conversions(data_path, fname):
180180
"oblique",
181181
],
182182
)
183-
@pytest.mark.parametrize("sw", ["afni", "fsl", "fs", "itk"])
183+
@pytest.mark.parametrize("sw", ["afni", "fsl", "fs", "itk", "afni-array"])
184184
def test_Linear_common(tmpdir, data_path, sw, image_orientation, get_testdata):
185185
tmpdir.chdir()
186186

@@ -190,6 +190,8 @@ def test_Linear_common(tmpdir, data_path, sw, image_orientation, get_testdata):
190190
ext = ""
191191
if sw == "afni":
192192
factory = afni.AFNILinearTransform
193+
elif sw == "afni-array":
194+
factory = afni.AFNILinearTransformArray
193195
elif sw == "fsl":
194196
factory = fsl.FSLLinearTransform
195197
elif sw == "itk":
@@ -222,6 +224,9 @@ def test_Linear_common(tmpdir, data_path, sw, image_orientation, get_testdata):
222224

223225
# Test from_ras
224226
RAS = from_matvec(euler2mat(x=0.9, y=0.001, z=0.001), [4.0, 2.0, -1.0])
227+
if sw == "afni-array":
228+
RAS = np.array([RAS, RAS])
229+
225230
xfm = factory.from_ras(RAS, reference=reference, moving=moving)
226231
assert np.allclose(xfm.to_ras(reference=reference, moving=moving), RAS)
227232

0 commit comments

Comments
 (0)