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

Modifications for psmr2024 #1430

Open
wants to merge 83 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 6 commits
Commits
Show all changes
83 commits
Select commit Hold shift + click to select a range
9e246fd
Adding support for singleTOF scanner.
NikEfth Mar 26, 2024
4634d30
Fast reconstruction for LAFOV scanners
NikEfth May 15, 2024
4ba55e6
Update src/buildblock/ProjDataInfo.cxx
NikEfth May 16, 2024
f5d663d
Update src/include/stir/ProjDataInfo.inl
NikEfth May 16, 2024
438be2f
Update src/recon_buildblock/ProjMatrixByBin.cxx
NikEfth May 16, 2024
0fb593d
Minor fixes and comments.
NikEfth May 16, 2024
7bf7e52
Minor fixes and executable to cache listmode files without reconstruc…
NikEfth May 16, 2024
6aa04b0
Update CMakeLists.txt
NikEfth May 16, 2024
5fa949c
Update src/IO/InterfileHeader.cxx
NikEfth May 16, 2024
e67faec
Minor fixes
NikEfth May 16, 2024
e7a2656
Merge branch 'master' into modifications_for_psmr2024
NikEfth May 16, 2024
9179482
Split GeneralisedPriorTests into several classes
KrisThielemans Apr 26, 2024
f7c15e6
fix abs() bug for RDP and slightly cleaner code
KrisThielemans Apr 26, 2024
a9c3ddb
change RDP Hessian behaviour at 0,0 when epsilon==0
KrisThielemans Apr 27, 2024
005acb2
extra tests for the Relative Difference Prior
KrisThielemans Apr 26, 2024
b7f727d
make epsilon in numerical test for prior gradient a bit smaller
KrisThielemans Apr 27, 2024
d8b3826
fix and document RDP large limit test for numerical error
KrisThielemans Apr 27, 2024
f17a7b0
update RDP doc [ci skip]
KrisThielemans Apr 27, 2024
619cc86
increase numerical precision of priors
KrisThielemans Apr 27, 2024
32dc591
[GHA] remove explicit setting of xcode version
KrisThielemans Apr 28, 2024
fa21161
update Byte Order compile-time checks
KrisThielemans Apr 30, 2024
ead279e
fix white-space
KrisThielemans May 9, 2024
205218d
List-mode objective function: always uses OpenMP for gradient
KrisThielemans May 9, 2024
eb31e8d
use more local variables in LM_distributable_computation
KrisThielemans May 10, 2024
45a102c
prepare LM_distributable function for Hessian
KrisThielemans May 12, 2024
7286488
minor clean-up and clarifications
NikEfth May 16, 2024
e1438a8
LM_listmode_function: implement Hessian and move functions
KrisThielemans May 12, 2024
60445db
fix error when OpenMP is not present
KrisThielemans May 12, 2024
7487039
Add compute_gradient* (as opposed to only subset)
KrisThielemans May 13, 2024
055b17e
Move common test to ObjectiveFunctionTests
KrisThielemans May 13, 2024
e11fbf9
add numerical test for accumulate_Hessian_times_input
KrisThielemans May 13, 2024
2315597
fix sign of LM Hessian
KrisThielemans May 13, 2024
9022195
LM obj-fun: add value() and tests
KrisThielemans May 13, 2024
f2f29bd
LM objfun tests slightly better diagnostics
KrisThielemans May 13, 2024
735e590
[GHA] upload ctest artefact if failure
KrisThielemans May 13, 2024
f3daa55
add "increment" test for gradient (enabled for LM data)
KrisThielemans May 14, 2024
0be2403
[GHA] disabled LM objfunc test on MacOS
KrisThielemans May 14, 2024
db1d7b7
fix return value FastErf::get_maximum_sample_value
KrisThielemans May 14, 2024
5864938
[CMake] force C++ version for more recent ROOT
KrisThielemans May 14, 2024
c813638
update to 6.1 and minor doc improvements
KrisThielemans May 14, 2024
a7bbe64
minor changes preparing for 6.1
KrisThielemans May 15, 2024
58143b1
Changing all occurrences of abs() to std::abs() (#1425)
markus-jehl May 15, 2024
b7003ad
add TOF loop in computation of Hessian for projdata
KrisThielemans May 15, 2024
1f6f8c5
fix set_up of norm in TOF ProjData gradient
KrisThielemans May 16, 2024
f4c1fb4
small rationalising of warnings
KrisThielemans May 16, 2024
44af378
let git blame ignore some maintenance commits
KrisThielemans May 16, 2024
23be801
updates to release notes
KrisThielemans May 16, 2024
d6231b0
update CITATION.cff according to git-fame
KrisThielemans May 16, 2024
1180e45
small change to credits
KrisThielemans May 16, 2024
4eac4ed
update history for release 6.1
KrisThielemans May 16, 2024
4439d5f
change comment to avoid confusing doxygen [ci skip]
KrisThielemans May 16, 2024
055f1c1
Merge remote-tracking branch 'refs/remotes/origin/modifications_for_p…
NikEfth May 16, 2024
0d6516b
New utility to convert ProjData to FanProjData3D.
NikEfth May 17, 2024
5c1af3f
Added option to load the model fanprojdata from the disk.
NikEfth May 17, 2024
a743a32
Update src/buildblock/Scanner.cxx
NikEfth May 17, 2024
c05dc3e
Fixes for test_time_of_flight.
NikEfth May 17, 2024
da59b78
minor fix
NikEfth May 17, 2024
fc36235
Fix to pass recon tests
NikEfth May 17, 2024
21597d0
find ML normalization from listmode data.
NikEfth May 17, 2024
9162578
Parallelization changes and labels
NikEfth May 17, 2024
069231c
Important fix.
NikEfth May 17, 2024
74fe00b
OpenMP parallelise Array::sum/find_max/find_min
KrisThielemans May 17, 2024
b297080
parallelise some functions in ML_norm
KrisThielemans May 17, 2024
75b9d66
ML_norm estimation: avoid doing some block calculations if !do_block
KrisThielemans May 17, 2024
3b56f1a
some ML_norm parallelisations
KrisThielemans May 17, 2024
c54c8a5
set KL_threshold to avoid problems
KrisThielemans May 17, 2024
6c411d8
More parallelization.
NikEfth May 18, 2024
a455248
small fix
NikEfth May 18, 2024
23cd5c4
fixes minor
NikEfth May 18, 2024
9860aa3
parallellized apply_geo_norm
NikEfth May 18, 2024
e1149e2
Fix single TOF position
NikEfth May 18, 2024
14f3cea
Print TOF
NikEfth May 18, 2024
2100f42
extra documentation for ML_estimate_component_based_normalisation
KrisThielemans May 18, 2024
a5d816a
[SWIG] expose ML_estimate etc.
KrisThielemans May 18, 2024
a22c45c
remove hard-wired 50 threads
KrisThielemans May 18, 2024
4aa5466
Some restructuring, needed.
NikEfth May 18, 2024
b1333e7
Merge remote-tracking branch 'refs/remotes/origin/modifications_for_p…
NikEfth May 18, 2024
53c9d05
fix writing of single-TOF bin data
KrisThielemans May 19, 2024
0c18fb6
another fix on tof data
KrisThielemans May 19, 2024
9520af8
Merge remote-tracking branch 'origin/master' into modifications_for_p…
KrisThielemans Jun 1, 2024
7eea917
fix Interfile writing of TOF data with 1 TOF bin
KrisThielemans Jun 1, 2024
d26b68a
replace std::cout calls with info()
KrisThielemans Jun 1, 2024
34a35de
Merge remote-tracking branch 'origin/master' into modifications_for_p…
KrisThielemans Jun 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ include(src/cmake/SetC++Version.cmake)

# minimum C++ version required (you can still ask for a more recent one
# by setting CMAKE_CXX_STANDARD)
UseCXX(14)
UseCXX(17)
NikEfth marked this conversation as resolved.
Show resolved Hide resolved

# set default build-type to Release
if(NOT CMAKE_BUILD_TYPE)
Expand Down
7 changes: 4 additions & 3 deletions src/IO/InterfileHeader.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -612,19 +612,19 @@ InterfilePDFSHeader::InterfilePDFSHeader()
reference_energy = -1.f;
add_key("Reference energy (in keV)", &reference_energy);

max_num_timing_poss = -1;
max_num_timing_poss = 1;
KrisThielemans marked this conversation as resolved.
Show resolved Hide resolved
add_key("Maximum number of (unmashed) TOF time bins", &max_num_timing_poss);
#if STIR_VERSION < 070000
add_alias_key("Maximum number of (unmashed) TOF time bins", "Number of TOF time bins");
#endif
timing_poss_sequence.clear();
add_key("TOF bin order", &timing_poss_sequence);
size_of_timing_pos = -1.f;
size_of_timing_pos = 0.0f;
add_key("Size of unmashed TOF time bins (ps)", &size_of_timing_pos);
#if STIR_VERSION < 070000
add_alias_key("Size of unmashed TOF time bins (ps)", "Size of timing bin (ps)");
#endif
timing_resolution = -1.f;
timing_resolution = 0.0f;
add_key("TOF timing resolution (ps)", &timing_resolution);
#if STIR_VERSION < 070000
add_alias_key("TOF timing resolution (ps)", "timing resolution (ps)");
Expand Down Expand Up @@ -1430,6 +1430,7 @@ InterfilePDFSHeader::post_processing()
data_info_sptr.reset(new ProjDataInfoGenericNoArcCorr(
scanner_sptr_from_file, sorted_num_rings_per_segment, sorted_min_ring_diff, sorted_max_ring_diff, num_views, num_bins));
}
std::cout << data_info_sptr->get_num_tof_poss() << " " << num_timing_poss << std::endl;
NikEfth marked this conversation as resolved.
Show resolved Hide resolved
if (data_info_sptr->get_num_tof_poss() != num_timing_poss)
error(boost::format("Interfile header parsing with TOF: inconsistency between number of TOF bins in data (%d), "
"TOF mashing factor (%d) and max number of TOF bins in scanner info (%d)")
Expand Down
34 changes: 33 additions & 1 deletion src/buildblock/ProjDataInfo.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -177,9 +177,41 @@ ProjDataInfo::set_tof_mash_factor(const int new_num)
{
tof_mash_factor = new_num;
if (tof_mash_factor > scanner_ptr->get_max_num_timing_poss())
{
error("ProjDataInfo::set_tof_mash_factor: TOF mashing factor (" + std::to_string(tof_mash_factor)
+ +") must be smaller than or equal to the scanner's number of max timing bins ("
+ std::to_string(scanner_ptr->get_max_num_timing_poss()) + ").");
}
else if (tof_mash_factor == num_tof_bins)
NikEfth marked this conversation as resolved.
Show resolved Hide resolved
// This is a special case that we just want boundaries for the coincidence window.
{

Bin bin;
bin.timing_pos_num() = min_tof_pos_num;
//Get lowest low
float cur_low = get_k(bin) - get_sampling_in_k(bin) / 2.f;
//Get highest high
bin.timing_pos_num() = max_tof_pos_num;
float cur_high = get_k(bin) + get_sampling_in_k(bin) / 2.f;


min_tof_pos_num = 0;
max_tof_pos_num = 0;
num_tof_bins = 1;
// Upper and lower boundaries of the timing poss;
tof_bin_boundaries_mm.recycle();
tof_bin_boundaries_mm.grow(min_tof_pos_num, max_tof_pos_num);
tof_bin_boundaries_ps.recycle();
tof_bin_boundaries_ps.grow(min_tof_pos_num, max_tof_pos_num);

tof_bin_boundaries_mm[0].low_lim = cur_low;
tof_bin_boundaries_mm[0].high_lim = cur_high;

tof_bin_boundaries_ps[0].low_lim = static_cast<float>(mm_to_tof_delta_time(tof_bin_boundaries_mm[0].low_lim));
tof_bin_boundaries_ps[0].high_lim = static_cast<float>(mm_to_tof_delta_time(tof_bin_boundaries_mm[0].high_lim));

return;
}

#if 0
// KT: code disabled as buggy but currently not needed
Expand Down Expand Up @@ -288,7 +320,7 @@ ProjDataInfo::ProjDataInfo(const shared_ptr<Scanner>& scanner_ptr_v,
max_tof_pos_num = 0;
tof_increament_in_mm = 0.f;
tof_mash_factor = 0;
num_tof_bins = 1;
num_tof_bins = 0;
NikEfth marked this conversation as resolved.
Show resolved Hide resolved
}

// TOF version.
Expand Down
58 changes: 58 additions & 0 deletions src/buildblock/Scanner.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -1231,6 +1231,64 @@ Scanner::Scanner(Type scanner_type)
);
break;


case PSMR_3rings_scanner:
set_params(PSMR_3rings_scanner,
string_list("PSMR_3rings_scanner"),
3 * 14 * 6,// + 8,
344,
344,
24 * 7 * 5,
405.0 - 18/2,
7.0F,
2.85F,
2.08626F,
static_cast<float>(-0.11),
14,// int num_axial_blocks_per_bucket_v,
5, // int num_transaxial_blocks_per_bucket_v,
6, // int num_axial_crystals_per_block_v,
7,// int num_transaxial_crystals_per_block_v,
252, // int num_axial_crystals_per_singles_unit_v,
35, // int num_transaxial_crystals_per_singles_unit_v,
1,
0.0F, 511.F,
#ifdef STIR_TOF
1,
2011.5F*2.F, // Size of coincidence window
0.F
#endif
);
break;

case PSMR_3rings_scanner_TOF:
set_params(PSMR_3rings_scanner_TOF,
string_list("PSMR_3rings_scanner_TOF"),
3 * 14 * 6,// + 8,
344, 344,
24 * 7 * 5,
405.0 - 18/2,
7.0F,
2.85F,
2.08626F,
static_cast<float>(-0.11),
14,// int num_axial_blocks_per_bucket_v,
5, // int num_transaxial_blocks_per_bucket_v,
6, // int num_axial_crystals_per_block_v,
7,// int num_transaxial_crystals_per_block_v,
252, // int num_axial_crystals_per_singles_unit_v,
35, // int num_transaxial_crystals_per_singles_unit_v,
NikEfth marked this conversation as resolved.
Show resolved Hide resolved
1,
0.0F, 511.F,
#ifdef STIR_TOF
// (short int)(2*3.51*1000 -1),
// (float)(1.f),
(short int)(13), // 13 TOF bins
(float)(2011.5F*2.F/13), //
(float)(500.F)
#endif
);
break;

case User_defined_scanner: // zlong, 08-04-2004, Userdefined support

set_params(User_defined_scanner,
Expand Down
4 changes: 3 additions & 1 deletion src/include/stir/ProjDataInfo.h
Original file line number Diff line number Diff line change
Expand Up @@ -150,7 +150,9 @@ class ProjDataInfo

//! Similar to create_shared_clone() but returns a non-tof version of ProjDataInfo setting tof mashing factor = 0
inline shared_ptr<ProjDataInfo> create_non_tof_clone() const;

//! Similar to create_non_tof_clone() but returns a single TOF version of the ProjDataInfo setting the TOF mashing factor sush that,
//! a single TOF bin remains. This is usefull to apply the coincidence window boundaries without affecting the probabilities.
inline shared_ptr<ProjDataInfo> create_single_tof_clone() const;
Copy link
Collaborator

Choose a reason for hiding this comment

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

I suppose we need this for backwards compatibility? i.e. need to keep create_non_tof_clone unmodified? If so, I'd be tempted to add a bool to create_non_tof_clone() instead.

Copy link
Collaborator

Choose a reason for hiding this comment

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

Note: will be obsolete once we merge another PR on SSRB for TOF

//! Destructor
virtual ~ProjDataInfo() {}

Expand Down
13 changes: 13 additions & 0 deletions src/include/stir/ProjDataInfo.inl
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,19 @@ ProjDataInfo::create_non_tof_clone() const
return sptr;
}

shared_ptr<ProjDataInfo>
ProjDataInfo::create_single_tof_clone() const
{
shared_ptr<ProjDataInfo> sptr(this->clone());
int a = sptr->get_num_tof_poss();
if (a > 0)
sptr->set_tof_mash_factor(a);
// - TODO: remove this later
// int b = sptr->get_num_tof_poss();
// std::cout <<"Old number of TOF poss " << a << " new number of TOF poss " << b << std::endl;
return sptr;
}

int
ProjDataInfo::get_num_segments() const
{
Expand Down
8 changes: 8 additions & 0 deletions src/include/stir/Scanner.h
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,12 @@ class Succeeded;
while others (such as CTI scanners) give only singles for a
collection of blocks. A \c singles_unit is then a set of crystals
for which we can get singles rates.
\li \c If the user set number for TOF positions to 1 and a bin size equal to the
coincidence window then the reconstruction will essential be nonTOF but the
projector will restrict the size of the LOR to the size of the coincidence window.
\li \c The scanner will be classified as TOF enabled when the numer of TOF bins and
TOF bin size are >1 and >0, respectively. If the energy resolution is not set that will be fine
as long as the final TOF possitions is 1. Then we just restict the size of the LOR.
NikEfth marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
\li \c If the user set number for TOF positions to 1 and a bin size equal to the
coincidence window then the reconstruction will essential be nonTOF but the
projector will restrict the size of the LOR to the size of the coincidence window.
\li \c The scanner will be classified as TOF enabled when the numer of TOF bins and
TOF bin size are >1 and >0, respectively. If the energy resolution is not set that will be fine
as long as the final TOF possitions is 1. Then we just restict the size of the LOR.
\li \c The scanner will be classified as TOF enabled when the number of TOF bins and
TOF bin size are >1 and >0, respectively.
\li \c If the user sets the number for TOF positions to 1, the TOF bin size should correspond to the size of the coincidence window. If the TOF timing resolution is zero, then the reconstruction will essentially be nonTOF but the
projector will restrict the length of the LOR according to the TOF bin width.


A further complication is that some scanners (including many Siemens scanners)
insert virtual crystals in the sinogram data (corresponding to gaps between
Expand Down Expand Up @@ -168,6 +174,8 @@ class Scanner
HZLR,
RATPET,
PANDA,
PSMR_3rings_scanner,
PSMR_3rings_scanner_TOF,
HYPERimage,
nanoPET,
HRRT,
Expand Down
2 changes: 1 addition & 1 deletion src/include/stir/Scanner.inl
Original file line number Diff line number Diff line change
Expand Up @@ -262,7 +262,7 @@ Scanner::get_timing_resolution() const
bool
Scanner::is_tof_ready() const
{
return (max_num_of_timing_poss > 0 && size_timing_pos > 0.0f && timing_resolution > 0.0f);
return (max_num_of_timing_poss > 0 && size_timing_pos > 0.0f ); //&& timing_resolution > 0.0f);
}

std::string
Expand Down
15 changes: 12 additions & 3 deletions src/include/stir/recon_buildblock/ProjMatrixByBin.inl
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ ProjMatrixByBin::get_proj_matrix_elems_for_one_bin(ProjMatrixElemsForOneBin& pro
#ifndef NDEBUG
probabilities.check_state();
#endif
if (proj_data_info_sptr->is_tof_data() && this->tof_enabled)
if (proj_data_info_sptr->is_tof_data())
{ // Apply TOF kernel to basic bin
apply_tof_kernel(probabilities);
}
Expand Down Expand Up @@ -97,7 +97,7 @@ ProjMatrixByBin::get_proj_matrix_elems_for_one_bin(ProjMatrixElemsForOneBin& pro
#ifndef NDEBUG
probabilities.check_state();
#endif
if (proj_data_info_sptr->is_tof_data() && this->tof_enabled)
if (proj_data_info_sptr->is_tof_data())
{ // Apply TOF kernel to basic bin
apply_tof_kernel(probabilities);
}
Expand Down Expand Up @@ -149,7 +149,16 @@ ProjMatrixByBin::get_tof_value(const float d1, const float d2) const
if ((d1_n >= 4.f && d2_n >= 4.f) || (d1_n <= -4.f && d2_n <= -4.f))
return 0.F;
else
return 0.5f * (erf_interpolation(d2_n) - erf_interpolation(d1_n));
{
if (this->tof_enabled)
{
return 0.5f * (erf_interpolation(d2_n) - erf_interpolation(d1_n));
}
else
{
return 1.0f;
}
}
}

END_NAMESPACE_STIR
Original file line number Diff line number Diff line change
Expand Up @@ -340,7 +340,7 @@ DataSymmetriesForBins_PET_CartesianGrid::DataSymmetriesForBins_PET_CartesianGrid
}

// RT Disabling some symmetries due to tof data
if (proj_data_info_ptr->is_tof_data())
if (proj_data_info_ptr->is_tof_data() && proj_data_info_ptr->get_num_tof_poss() > 1)
{
if (this->do_symmetry_90degrees_min_phi || this->do_symmetry_180degrees_min_phi)
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -280,14 +280,14 @@ PoissonLogLikelihoodWithLinearModelForMeanAndListModeDataWithProjMatrixByBin<Tar
> 1)) // TODO this check needs to cover the case if we reconstruct only TOF bin 0
{
// sets non-tof backprojector for sensitivity calculation (clone of the back_projector + set projdatainfo to non-tof)
this->sens_proj_data_info_sptr = this->proj_data_info_sptr->create_non_tof_clone();
this->sens_proj_data_info_sptr = this->proj_data_info_sptr->create_single_tof_clone();
// TODO disable caching of the matrix
this->sens_backprojector_sptr.reset(projector_pair_sptr->get_back_projector_sptr()->clone());
this->sens_backprojector_sptr->set_up(this->sens_proj_data_info_sptr, target_sptr);
}
else
{
// just use the normal backprojector
// just use a signle TOF backprojector
this->sens_proj_data_info_sptr = this->proj_data_info_sptr;
this->sens_backprojector_sptr = projector_pair_sptr->get_back_projector_sptr();
}
Expand Down Expand Up @@ -622,60 +622,79 @@ PoissonLogLikelihoodWithLinearModelForMeanAndListModeDataWithProjMatrixByBin<Tar
{
// TODO replace with call to distributable function

const int min_segment_num = this->proj_data_info_sptr->get_min_segment_num();
const int max_segment_num = this->proj_data_info_sptr->get_max_segment_num();

info(boost::format("Calculating sensitivity for subset %1%") % subset_num);

int min_timing_pos_num = use_tofsens ? this->proj_data_info_sptr->get_min_tof_pos_num() : 0;
int max_timing_pos_num = use_tofsens ? this->proj_data_info_sptr->get_max_tof_pos_num() : 0;
if (min_timing_pos_num < 0 || max_timing_pos_num > 0)
if (min_timing_pos_num < 0 || max_timing_pos_num > 1)
Copy link
Collaborator

Choose a reason for hiding this comment

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

Suggested change
if (min_timing_pos_num < 0 || max_timing_pos_num > 1)
if (min_timing_pos_num < 0 || max_timing_pos_num > 0)

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

No, this is correct. Up to one TOF bin should not be considered TOF reconstruction.

Copy link
Collaborator

Choose a reason for hiding this comment

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

still not with you on this one (i.e. with the 1). Why not say num_timing_poss > 1

error("TOF code for sensitivity needs work");

this->sens_backprojector_sptr->start_accumulating_in_new_target();

// warning: has to be same as subset scheme used as in distributable_computation
int runs = 1;

//TODO: For long scanners we should run better split this. - Let's discuss more.
// if((this->proj_data_info_sptr->get_max_segment_num() -
// this->proj_data_info_sptr->get_min_segment_num()) > 100)
// {
// runs = ceil(this->proj_data_info_sptr->get_max_segment_num() -
// this->proj_data_info_sptr->get_min_segment_num())/ 100;
// }

info(boost::format("The number of runs needed for the sensitivity image is %1%: ") % runs);

for (int run = 0; run < runs; ++run)
{
const int min_segment_num = this->proj_data_info_sptr->get_min_segment_num();
const int max_segment_num = this->proj_data_info_sptr->get_max_segment_num();
info(boost::format("Current run %1% of %2%: ") % run % runs);
// warning: has to be same as subset scheme used as in distributable_computation
#ifdef STIR_OPENMP
# if _OPENMP < 201107
# pragma omp parallel for schedule(dynamic)
# else
# pragma omp parallel for collapse(2) schedule(dynamic)
# endif
#endif
for (int segment_num = min_segment_num; segment_num <= max_segment_num; ++segment_num)
{
for (int view = this->sens_proj_data_info_sptr->get_min_view_num() + subset_num;
view <= this->sens_proj_data_info_sptr->get_max_view_num();
view += this->num_subsets)
for (int segment_num = min_segment_num; segment_num <= max_segment_num; ++segment_num)
{
const ViewSegmentNumbers view_segment_num(view, segment_num);
for (int view = this->sens_proj_data_info_sptr->get_min_view_num() + subset_num;
view <= this->sens_proj_data_info_sptr->get_max_view_num();
view += this->num_subsets)
{
const ViewSegmentNumbers view_segment_num(view, segment_num);

if (!this->sens_backprojector_sptr->get_symmetries_used()->is_basic(view_segment_num))
continue;
// for (int timing_pos_num = min_timing_pos_num; timing_pos_num <= max_timing_pos_num; ++timing_pos_num)
{
shared_ptr<DataSymmetriesForViewSegmentNumbers> symmetries_used(
this->sens_backprojector_sptr->get_symmetries_used()->clone());
if (!this->sens_backprojector_sptr->get_symmetries_used()->is_basic(view_segment_num))
continue;

info(boost::format("Current seg %1%, view: %1% ") % segment_num, view);
// for (int timing_pos_num = min_timing_pos_num; timing_pos_num <= max_timing_pos_num; ++timing_pos_num)
{
shared_ptr<DataSymmetriesForViewSegmentNumbers> symmetries_used(
this->sens_backprojector_sptr->get_symmetries_used()->clone());

RelatedViewgrams<float> viewgrams = this->sens_proj_data_info_sptr->get_empty_related_viewgrams(
view_segment_num, symmetries_used, false); //, timing_pos_num);
RelatedViewgrams<float> viewgrams = this->sens_proj_data_info_sptr->get_empty_related_viewgrams(
view_segment_num, symmetries_used, false); //, timing_pos_num);

viewgrams.fill(1.F);
// find efficiencies
{
this->normalisation_sptr->undo(viewgrams);
}
// backproject
{
const int min_ax_pos_num = viewgrams.get_min_axial_pos_num();
const int max_ax_pos_num = viewgrams.get_max_axial_pos_num();
viewgrams.fill(1.F);
// find efficiencies
{
this->normalisation_sptr->undo(viewgrams);
}
// backproject
{
const int min_ax_pos_num = viewgrams.get_min_axial_pos_num();
const int max_ax_pos_num = viewgrams.get_max_axial_pos_num();

this->sens_backprojector_sptr->back_project(viewgrams, min_ax_pos_num, max_ax_pos_num);
this->sens_backprojector_sptr->back_project(viewgrams, min_ax_pos_num, max_ax_pos_num);
}
}
}
}
}
this->sens_backprojector_sptr->get_output(sensitivity);
// Next run
}
this->sens_backprojector_sptr->get_output(sensitivity);

}

template <typename TargetT>
Expand Down
Loading
Loading