-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathrun_retroMoCoDemo_NUFFT.m
executable file
·153 lines (122 loc) · 5.88 KB
/
run_retroMoCoDemo_NUFFT.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
% run_retroMoCoDemo_NUFFT.m
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% Demo of the NUFFT operation to peform retrospective motion-correction on
% real example 3D datasets.
%
%%% The data for this demo can be downloaded from:
% 1mm data: https://bit.ly/retroData_2(32 MB)
% 600 um data: https://bit.ly/retroData_3 (86 MB)
% 1mm data (only very small motion): https://bit.ly/retroData_1 (32 MB)
%
% Note that these datasets are stripped-down versions of 'real' MP2RAGE
% scans to save on disk space and memory requirements for this demo.
% They show very poor contrast between GM and WM because they only
% include the 'INV2' part of the two acquisitions which constitute the
% MP2RAGE scan. The 32-channels of the RF array coil have also been
% combined into a single 'virtual channel' using an SVD. This gives good
% signal many places in the brain - but is also the reason for the hole in
% the middle!
%
% If you are interested in processing full example datasets
% please contact me (gallichand@cardiff.ac.uk) - the 1mm dataset is 4.3 GB
% and the 600um dataset is 9.3 GB.
%
% Change these lines to match your paths:
exampleData = load('../exampleData/example_retroMocoData.mat'); % 1mm resolution example data
% exampleData = load('../exampleData/example_retroMocoData_600.mat'); % 600 um resolution example data
% exampleData = load('../exampleData/example_retroMocoData_1mm_smallMotion.mat'); %1mm resolution example data with very small subject motion
run('addRetroMoCoBoxToPath.m')
% The NUFFT uses the Michigan Image Reconstruction Toolbox (MIRT)
% (http://web.eecs.umich.edu/~fessler/code/index.html)
% This is now included directly in retroMoCoBox/mirt_nufft
%% Create the image without MoCo
% raw data was acquired with 3/4 partial Fourier in both PE directions, so
% extend the k-space before the Fourier transform for the no-Moco image
image_noMoco = exampleData.newData;
hxyz = size(image_noMoco); Hxyz = exampleData.Hxyz;
image_noMoco(Hxyz(1),Hxyz(2),Hxyz(3)) = 0; % extend to new size
image_noMoco = circshift(image_noMoco,double([Hxyz(1)-hxyz(1) Hxyz(2)-hxyz(2) 0]));
image_noMoco = ifft3s(image_noMoco);
image_noMoco = image_noMoco / percentile(abs(image_noMoco),95);
%% Do the MoCo
rawData = exampleData.newData; % the original raw data
% In this example it is the principal component of the SVD coil combination
% so that a single virtual coil channel can be used to have coverage over
% most of the brain. The combination was performed *after* the GRAPPA
% reconstruction of the original undersampled data. It is also only the
% data for the second inversion time for MP2RAGE, and because the first
% inversion time is missing then the final image does not have as good a
% contrast as the combined 'UNI' image would have. However, there is enough
% contrast to see that the MoCo is working!
fitMats = exampleData.fitMats_mm_toApply; % the motion parameters
% These are the affine matrices describing the estimated motion. In this
% case they were derived from SPMs 'realign' function (SPM was used as it
% appears to be good for maintining accuracy for sub-pixel alignments of
% the same image). It is very important to remap the output from SPM into
% the coordinate system of the data to be corrected - with the centre of
% the coordinate system at the centre of the FOV so that the rotations will
% be applied properly.
% As this data was also acquired with GRAPPA acceleration, the 'trick' has
% been used of interpolating the motion parameters from neighbouring
% acquired k-space lines to create 'virtual' motion parameters for the
% unacquired lines. This is perhaps not as stupid as it might seem at
% first, as it leads to a kind of interpolation of the new k-space
% positions. To see how this was implemented, look inside reconstructSiemensMP2RAGEwithFatNavs.m
alignDim = exampleData.alignDim; % the dimension of the raw data that corresponds to chronological time
% This data was acquired with a simple looping hierarchy. The code can also
% cope with more complicated reordering schemes - although this has not
% been tested exhaustively.
alignIndices = exampleData.alignIndices;
% allow for the fact that time could be running in a positive or negative
% direction in the k-space matrix - and also encode a 2D reordering if
% necessary.
hostVoxDim_mm = exampleData.hostVoxDim_mm;
% This is necessary both to scale the translations appropriately - as well
% as dealing with anisotropic FOVs (again not tested exhaustively
% though...)
Hxyz = exampleData.Hxyz; % The full matrix size of the reconstructed image
kspaceCentre_xyz = exampleData.kspaceCentre_xyz;
% The centre coordinates for the acquired k-space (needed to know where to rotate around in k-space)
%%% And actually do the MoCo:
tic
image_withMoco = applyRetroMC_nufft(rawData,fitMats,alignDim,alignIndices,11,hostVoxDim_mm,Hxyz,kspaceCentre_xyz);
toc
image_withMoco = image_withMoco / percentile(abs(image_withMoco),95);
%% Load both images in a 3D viewer
%
% Use the + and - keys on the keyboard to go back and forth between
% corrected and uncorrected volumes
if hostVoxDim_mm(1)==1
clims = [0 2];
else
clims = [0 1.5];
end
SliceBrowser2(cat(4,abs(image_noMoco),abs(image_withMoco)),clims,{'No MoCo','With MoCo'})
%% View the result
ov1 = orthoview(image_noMoco,'drawIms',0);
ov2 = orthoview(image_withMoco,'drawIms',0);
figure
set(gcf,'Position',[ 76 143 1048 757])
clims = [0 1.8];
subplot1(2,2)
subplot1(1)
imab(ov1.oneIm,clims)
title('No MoCo')
subplot1(3)
imab(ov2.oneIm,clims)
title('With MoCo')
if hostVoxDim_mm(1)==1
yrange = 122:212; zrange = 138:241;
else
yrange = 61:160 ; zrange = 163:271;
end
subplot1(2)
imab(ov1.im3(yrange,zrange))
subplot1(4)
imab(ov2.im3(yrange,zrange))
fontScale(1.4)
colormap(gray)
%% And view the motion parameters that were used
fitpars_trans = squeeze(fitMats(1:3,4,:));
fitpars_rot = rotmat2euler(fitMats(1:3,1:3,:));
plotFitPars(cat(1,fitpars_trans,fitpars_rot));