ESPIRiT Reconstruction with L1-Wavelet Demo

This is a demo on how to generate ESPIRiT maps and use them to perform ESPIRiT reconstruction for parallel imaging. It is based on the paper Uecker et. al, MRM 2013 DOI 10.1002/mrm.24751. ESPIRiT is a method that finds the subspace of multi-coil data from a calibration region in k-space using a series of eigen-value decompositions in k-space and image space. Here we also use the "soft" sense idea (Uecker et. al, "ESPIRiT Reconstruction using Soft-SENSE", Proceedings of the ISMRM 2013, pp-127) by using the eigen values to weight the eigen-vectors.

Contents

Prepare DATA

Here we perform ESPIRiT calibration on data which has strong aliasing in the phase-encode direction. SENSE often fails with this type of data.

load brain_8ch
DATA = DATA/max(max(max(abs(ifft2c(DATA))))) + eps;

ksize = [6,6]; % ESPIRiT kernel-window-size
eigThresh_k = 0.02; % threshold of eigenvectors in k-space
eigThresh_im = 0.9; % threshold of eigenvectors in image space

% parameters for L1-reconstruction with splitting
nIterCG = 5;       % number of CG iterations for the PI part
nIterSplit = 15;    % number of splitting iterations for CS part
splitWeight = 0.4;  % reasonable value
lambda = 0.0025;    % L1-Wavelet threshold


[sx,sy,Nc] = size(DATA);

% create a sampling mask to simulate x2 undersampling with autocalibration
% lines
mask = mask_randm_x4;
mask = repmat(mask,[1,1,8]);
ncalib = getCalibSize(mask_randm_x4);



DATAc = DATA.*mask;
calib = crop(DATAc,[ncalib,Nc]);

Display coil images:

im = ifft2c(DATAc);


figure, imshow3(abs(im),[],[1,Nc]);
title('magnitude of physical coil images');
colormap((gray(256))); colorbar;

figure, imshow3(angle(im),[],[1,Nc]);
title('phase of physical coil images');
colormap('default'); colorbar;

Compute Eigen-Value Maps

Maps are computed in two steps.

% compute Calibration matrix, perform 1st SVD and convert singular vectors
% into k-space kernels

[k,S] = dat2Kernel(calib,ksize);

idx = max(find(S >= S(1)*eigThresh_k));

Display the singular vectors and values of the calibration matrix

kdisp = reshape(k,[ksize(1)*ksize(2)*Nc,ksize(1)*ksize(2)*Nc]);
figure, subplot(211), plot([1:ksize(1)*ksize(2)*Nc],S,'LineWidth',2);
hold on,
plot([1:ksize(1)*ksize(2)*Nc],S(1)*eigThresh_k,'r-','LineWidth',2);
plot([idx,idx],[0,S(1)],'g--','LineWidth',2)
legend('signular vector value','threshold')
title('Singular Vectors')
subplot(212), imagesc(abs(kdisp)), colormap(gray(256));
xlabel('Singular value #');
title('Singular vectors')

crop kernels and compute eigen-value decomposition in image space to get maps

[M,W] = kernelEig(k(:,:,:,1:idx),[sx,sy]);

show eigen-values and eigen-vectors. The last set of eigen-vectors corresponding to eigen-values 1 look like sensitivity maps

figure, imshow3(abs(W),[],[1,Nc]);
title('Eigen Values in Image space');
colormap((gray(256))); colorbar;

figure, imshow3(abs(M),[],[Nc,Nc]);
title('Magnitude of Eigen Vectors');
colormap(gray(256)); colorbar;

figure, imshow3(angle(M),[],[Nc,Nc]);
title('Magnitude of Eigen Vectors');
colormap(jet(256)); colorbar;
Warning: Image is too big to fit on screen; displaying at 67% 
Warning: Image is too big to fit on screen; displaying at 67% 

Compute Soft-SENSE ESPIRiT Maps

crop sensitivity maps according to eigenvalues==1. Note that we have to use 2 sets of maps. Here we weight the 2 maps with the eigen-values

maps = M(:,:,:,end-1:end);

% Weight the eigenvectors with soft-senses eigen-values
weights = W(:,:,end-1:end) ;
weights = (weights - eigThresh_im)./(1-eigThresh_im).* (W(:,:,end-1:end) > eigThresh_im);
weights = -cos(pi*weights)/2 + 1/2;

% create and ESPIRiT operator
ESP = ESPIRiT(maps,weights);

Reconstructions

ESPIRiT CG reconstruction with soft-sense and 1 sets of maps

XOP = Wavelet('Daubechies_TI',4,6);
FT = p2DFT(mask,[sx,sy,Nc]);

disp('Performing ESPIRiT reconstruction from 2 maps')
tic; [reskESPIRiT, resESPIRiT] = cgESPIRiT(DATAc,ESP, nIterCG*3, 0.01,DATAc*0); toc

disp('Performing L1-ESPIRiT reconstruction from 2 maps')
tic
[resL1ESPIRiT] = cgL1ESPIRiT(DATAc, resESPIRiT*0, FT, ESP, nIterCG,XOP,lambda,splitWeight,nIterSplit);
toc

% GRAPPA reconstruction
disp('Performing GRAPPA reconstruction ... slow in Matlab! ')
tic; reskGRAPPA = GRAPPA(DATAc,calib,[5,5],0.01);toc
resGRAPPA = ifft2c(reskGRAPPA);
Performing ESPIRiT reconstruction from 2 maps
Elapsed time is 3.460917 seconds.
Performing L1-ESPIRiT reconstruction from 2 maps
Iteration: 1, consistency: 18.537199
Iteration: 2, consistency: 6.907292
Iteration: 3, consistency: 4.212248
Iteration: 4, consistency: 3.600387
Iteration: 5, consistency: 3.386763
Iteration: 6, consistency: 3.279046
Iteration: 7, consistency: 3.215016
Iteration: 8, consistency: 3.173626
Iteration: 9, consistency: 3.145386
Iteration: 10, consistency: 3.125352
Iteration: 11, consistency: 3.110704
Iteration: 12, consistency: 3.099735
Iteration: 13, consistency: 3.091360
Iteration: 14, consistency: 3.084862
Iteration: 15, consistency: 3.079751
Elapsed time is 22.371217 seconds.
Performing GRAPPA reconstruction ... slow in Matlab! 
reconstructing coil 1
reconstructing coil 2
reconstructing coil 3
reconstructing coil 4
reconstructing coil 5
reconstructing coil 6
reconstructing coil 7
reconstructing coil 8
Elapsed time is 225.601446 seconds.

Note the typical center FOV aliasing in SENSE. Also, note that ESPIRiT has (very slightly) less error than GRAPPA

figure, imshow(cat(2,sos(resESPIRiT), sos(resL1ESPIRiT),sos(resGRAPPA)),[0,1]);
title('ESPIRiT reconstruction  vs L1-ESPIRiT  vs GRAPPA')
figure, imshow(cat(2,sos(ifft2c(reskESPIRiT-DATA)),sos(ifft2c(fft2c(ESP*resL1ESPIRiT)-DATA)),sos(ifft2c(reskGRAPPA-DATA))).^(1/2),[])
title('ESPIRiT reconstruction error vs L1-ESPIRiT vs GRAPPA')