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

MLEstimateComponentBasedNormalisation refactor #1499

Open
wants to merge 36 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
3035db8
Refactor ML_estimate_component_based_normalisation creating sub-funct…
robbietuk Aug 29, 2024
bc58f39
Create MLEstimateComponentBasedNormalisation class and start to strip…
robbietuk Aug 29, 2024
0dca08e
Refactor to use class varables
robbietuk Aug 29, 2024
9fed40f
Add do_save_to_file guards
robbietuk Aug 29, 2024
2579f81
Documentation and minor refactors
robbietuk Aug 29, 2024
5d89071
Rename refactor
robbietuk Aug 29, 2024
86f6a82
Addition usage of shared_ptrs
robbietuk Aug 29, 2024
880e530
[ci skip] parameter rename, remove redundant logic, documentation
robbietuk Aug 29, 2024
dd711b1
Remove model_data member and pass by reference into the constructor
robbietuk Aug 30, 2024
2bcdc80
Readd stir/steam.h to includes
robbietuk Aug 30, 2024
0fa8455
[ci skip] Update copywrite
robbietuk Aug 30, 2024
90ea81b
[ci skip] Fix a bug with do_KL info print
robbietuk Aug 30, 2024
f0c788b
Add allocate override using a MLEstimateComponentBasedNormalisation
robbietuk Aug 30, 2024
e5e2cc7
[ci skip] cleanup comments
robbietuk Aug 30, 2024
8adb620
Rename files MLEstimateComponentBasedNormalisation
robbietuk Sep 3, 2024
9ff77fe
Remove allocate with norm estimator
robbietuk Sep 3, 2024
5c1a0f7
Add construct_bin_normfactors_from_components to MLEstimateComponentB…
robbietuk Sep 3, 2024
a114d14
Change efficiencies, norm_block_data, norm_geo_data values from sptr
robbietuk Sep 3, 2024
b0e05c9
Documentation update.
robbietuk Sep 3, 2024
1d6ca9b
Fix variable names
robbietuk Sep 4, 2024
0aae32e
Add guard to ensure measured and model data use the same pdi
robbietuk Sep 4, 2024
ca1140c
Add test for MLEstimateComponentBasedNormalisation
robbietuk Sep 4, 2024
2935b99
Correct comparison to objects, not sptr
robbietuk Sep 4, 2024
3c11502
[ci skip] Fix min/max tolerance check
robbietuk Sep 4, 2024
7c18bc7
Change scanner to Discovery
robbietuk Sep 12, 2024
42340f2
Add additional test on efficiency values
robbietuk Sep 12, 2024
f424ca8
Ensure normalization_projdata max and min values are about 2.0
robbietuk Sep 12, 2024
a5f4b91
Make test use check_if_equal
robbietuk Sep 13, 2024
1eefeff
Split test_normalization_calculation_with_efficiencies from the run_t…
robbietuk Sep 13, 2024
dcee251
Improve usage of data_is_processed()
robbietuk Sep 13, 2024
63f8560
Add getters and setters
robbietuk Sep 13, 2024
3d7e9ff
[ci skip] Add const to args
robbietuk Sep 13, 2024
55e5f1a
Merge remote-tracking branch 'UCL/master' into ML_estimate_component_…
robbietuk Sep 13, 2024
55897ee
[ci skip] Update release 6.3 notes
robbietuk Sep 13, 2024
cbe81ec
[ci skip] Fix error message
robbietuk Sep 16, 2024
fd194fc
Merge branch 'master' into ML_estimate_component_based_normalisation_…
robbietuk Dec 22, 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
12 changes: 6 additions & 6 deletions src/buildblock/ML_norm.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -987,6 +987,10 @@ get_fan_info(int& num_rings, int& num_detectors_per_ring, int& max_ring_diff, in
{
error("Can only process data without axial compression (i.e. span=1)\n");
}
if (proj_data_info_ptr->is_tof_data())
{
error("make_fan_data: Incompatible with TOF data. Abort.");
}
num_rings = proj_data_info.get_scanner_ptr()->get_num_rings();
num_detectors_per_ring = proj_data_info.get_scanner_ptr()->get_num_detectors_per_ring();
const int half_fan_size = min(proj_data_info.get_max_tangential_pos_num(), -proj_data_info.get_min_tangential_pos_num());
Expand Down Expand Up @@ -1141,23 +1145,19 @@ make_fan_data_remove_gaps(FanProjData& fan_data, const ProjData& proj_data)
int num_detectors_per_ring;
int fan_size;
int max_delta;

if (proj_data.get_proj_data_info_sptr()->is_tof_data())
error("make_fan_data: Incompatible with TOF data. Abort.");

const ProjDataInfo& proj_data_info = *proj_data.get_proj_data_info_sptr();
get_fan_info(num_rings, num_detectors_per_ring, max_delta, fan_size, proj_data_info);

if (proj_data.get_proj_data_info_sptr()->get_scanner_ptr()->get_scanner_geometry() == "Cylindrical")
{
auto proj_data_info_ptr = dynamic_cast<const ProjDataInfoCylindricalNoArcCorr* const>(&proj_data_info);
const auto proj_data_info_ptr = dynamic_cast<const ProjDataInfoCylindricalNoArcCorr* const>(&proj_data_info);
robbietuk marked this conversation as resolved.
Show resolved Hide resolved

make_fan_data_remove_gaps_help(
fan_data, num_rings, num_detectors_per_ring, max_delta, fan_size, *proj_data_info_ptr, proj_data);
}
else
{
auto proj_data_info_ptr = dynamic_cast<const ProjDataInfoBlocksOnCylindricalNoArcCorr* const>(&proj_data_info);
const auto proj_data_info_ptr = dynamic_cast<const ProjDataInfoBlocksOnCylindricalNoArcCorr* const>(&proj_data_info);

make_fan_data_remove_gaps_help(
fan_data, num_rings, num_detectors_per_ring, max_delta, fan_size, *proj_data_info_ptr, proj_data);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
#ifndef __stir_recon_buildblock_BinNormalisationPETFromComponents_H__
#define __stir_recon_buildblock_BinNormalisationPETFromComponents_H__

#include "ML_estimate_component_based_normalisation.h"
#include "stir/recon_buildblock/BinNormalisation.h"
#include "stir/ML_norm.h"
#include "stir/ProjData.h"
Expand Down Expand Up @@ -126,6 +127,8 @@ class BinNormalisationPETFromComponents : public BinNormalisation
void
allocate(shared_ptr<const ProjDataInfo>, bool do_eff, bool do_geo, bool do_block = false, bool do_symmetry_per_block = false);

void allocate(shared_ptr<const ProjDataInfo> pdi_sptr, MLEstimateComponentBasedNormalisation &normalization_estimator);

DetectorEfficiencies& crystal_efficiencies()
{
return efficiencies;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
/*
Copyright (C) 2022, University College London
Copyright (C) 2024, Robert Twyman Skelly
This file is part of STIR.

SPDX-License-Identifier: Apache-2.0
Expand All @@ -11,28 +12,177 @@
\ingroup recon_buildblock
\brief Declaration of stir::ML_estimate_component_based_normalisation
\author Kris Thielemans
\author Robert Twyman Skelly
*/
#include "stir/common.h"
#include <string>
#include "stir/ProjData.h"
#include "stir/ML_norm.h"

START_NAMESPACE_STIR

class ProjData;

/*!
\brief Find normalisation factors using a maximum likelihood approach

\brief Find normalisation factors using a maximum likelihood approach
\ingroup recon_buildblock
\param out_filename_prefix The prefix for the output files
\param measured_data The measured projection data
\param model_data The model projection data
\param num_eff_iterations The number of (sub-)efficiency iterations to perform per iteration of the algorithm
\param num_iterations The number of algorithm iterations to perform
\param do_geo Whether to perform geo normalization calculations
\param do_block Whether to perform block normalization calculations
\param do_symmetry_per_block Whether to perform block normalization calculations with symmetry
\param do_KL Whether to perform Kullback-Leibler divergence calculations and display the KL value. This can be slow.
\param do_display Whether to display the progress of the algorithm.
\param do_save_to_file Whether to save the each iteration of the efficiencies, geo data and block data to file.
*/
void ML_estimate_component_based_normalisation(const std::string& out_filename_prefix,
const ProjData& measured_data,
const ProjData& model_data,
const ProjData& measured_projdata,
const ProjData& model_projdata,
int num_eff_iterations,
int num_iterations,
bool do_geo,
bool do_block,
bool do_symmetry_per_block,
bool do_KL,
bool do_display);
bool do_display,
bool do_save_to_file = true);

/*!
\brief Find normalisation factors using a maximum likelihood approach
\ingroup recon_buildblock
*/
robbietuk marked this conversation as resolved.
Show resolved Hide resolved
class MLEstimateComponentBasedNormalisation
{
public:
/*!
\brief Constructor
\param out_filename_prefix The prefix for the output files
\param measured_data The measured projection data
\param model_data The model projection data
\param num_eff_iterations_v The number of (sub-)efficiency iterations to perform per iteration of the algorithm
\param num_iterations_v The number of algorithm iterations to perform
\param do_geo_v Whether to perform geo normalization calculations
\param do_block_v Whether to perform block normalization calculations
\param do_symmetry_per_block_v Whether to perform block normalization calculations with symmetry
\param do_KL_v Whether to perform Kullback-Leibler divergence calculations and display the KL value. This can be slow.
\param do_display_v Whether to display the progress of the algorithm.
\param do_save_to_file_v Whether to save the each iteration of the efficiencies, geo data and block data to file.
*/
MLEstimateComponentBasedNormalisation(std::string out_filename_prefix,
const ProjData& measured_data_v,
const ProjData& model_data_v,
int num_eff_iterations_v,
int num_iterations_v,
bool do_geo_v,
bool do_block_v,
bool do_symmetry_per_block_v,
bool do_KL_v,
bool do_display_v,
bool do_save_to_file_v);

/*!
\brief Run the normalisation estimation algorithm using the parameters provided in the constructor.
*/
void process();

//! Check if the data has been processed
bool has_processed_data() const;

//! Get the efficiencies, nullptr if not calculated
std::shared_ptr<DetectorEfficiencies> get_efficiencies() const;
//! Get the geo data, nullptr if not calculated
std::shared_ptr<GeoData3D> get_geo_data() const;
//! Get the block data, nullptr if not calculated
std::shared_ptr<BlockData3D> get_block_data() const;

private:
/*!
\brief Performs an efficiency iteration to update the efficiancies from the data_fan_sums and model.
Additionally, handles the saving of the efficiencies iteration to file, KL calculation and display.
\param[in] iter_num The iteration number
\param[in] eff_iter_num The efficiency iteration number
*/
void efficiency_iteration(int iter_num, int eff_iter_num);

/*!
\brief Performs a geo normalization iteration to update the geo data from the measured_geo_data and model_data.
Additionally, handles the saving of the geo data iteration to file, KL calculation and display.
\param[in] iter_num The iteration number
*/
void geo_normalization_iteration(int iter_num);

/*!
\brief Performs a block normalization iteration to update the block data from the measured_block_data and model_data.
Additionally, handles the saving of the block data iteration to file, KL calculation and display.
* @param iter_num The iteration number
*/
void block_normalization_iteration(int iter_num);

/*!
\brief Write the efficiencies to a file (regardless of the value of do_save_to_file)
\param[in] iter_num The iteration number
\param[in] eff_iter_num The efficiency iteration number
*/
void write_efficiencies_to_file(int iter_num, int eff_iter_num) const;

/*!
\brief Write the efficiencies to a file (regardless of the value of do_save_to_file)
\param[in] iter_num The iteration number
*/
void write_geo_data_to_file(int iter_num) const;

/*!
\brief Write the efficiencies to a file (regardless of the value of do_save_to_file)
\param[in] iter_num The iteration number
*/
void write_block_data_to_file(int iter_num) const;

/*!
\brief Computes the threshold for the Kullback-Leibler divergence calculation. This is a purely heuristic value.
\return The threshold value
*/
float compute_threshold_for_KL();

// Constructor parameters
//! The prefix for the output files
std::string out_filename_prefix;
//! The number of (sub-)efficiency iterations to perform per iteration of the algorithm
int num_eff_iterations;
//! The number of algorithm iterations to perform
int num_iterations;
//! Whether to perform geo normalization calculations
bool do_geo;
//! Whether to perform block normalization calculations
bool do_block;
//! Whether to perform block normalization calculations with symmetry
bool do_symmetry_per_block;
robbietuk marked this conversation as resolved.
Show resolved Hide resolved
//! Whether to perform Kullback-Leibler divergence calculations and display the KL value. This can be slow.
bool do_KL;
//! Whether to display the progress of the algorithm.
bool do_display;
//! Whether to save the each iteration of the efficiencies, geo data and block data to file.
bool do_save_to_file;

// Calculated variables
std::shared_ptr<DetectorEfficiencies> efficiencies_ptr;
std::shared_ptr<BlockData3D> norm_block_data_ptr;
std::shared_ptr<GeoData3D> norm_geo_data_ptr;
//! The threshold for the Kullback-Leibler divergence calculation
float threshold_for_KL;
FanProjData model_fan_data;
FanProjData fan_data;
DetectorEfficiencies data_fan_sums;
BlockData3D measured_block_data;
GeoData3D measured_geo_data;

bool data_processed = false;

// do_KL specific varaibles
FanProjData measured_fan_data;
DetectorEfficiencies fan_sums;
GeoData3D geo_data;
BlockData3D block_data;
robbietuk marked this conversation as resolved.
Show resolved Hide resolved
};

END_NAMESPACE_STIR
14 changes: 14 additions & 0 deletions src/recon_buildblock/BinNormalisationPETFromComponents.cxx
Original file line number Diff line number Diff line change
Expand Up @@ -173,6 +173,20 @@ BinNormalisationPETFromComponents::allocate(

this->_already_allocated = true;
}
void
BinNormalisationPETFromComponents::allocate(shared_ptr<const ProjDataInfo> pdi_sptr, MLEstimateComponentBasedNormalisation& normalization_estimator)
{
if (!normalization_estimator.has_processed_data())
{
error("BinNormalisationPETFromComponents: internal error: allocate called on MLEstimateComponentBasedNormalisation "
"without it having processed the data");
}
this->proj_data_info_sptr = pdi_sptr;
this->efficiencies = *normalization_estimator.get_efficiencies();
this->geo_data = *normalization_estimator.get_geo_data();
this->block_data = *normalization_estimator.get_block_data();
this->_already_allocated = true;
}
Copy link
Collaborator Author

Choose a reason for hiding this comment

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

I believe this is the cleanest way to implement all these changes. Here is a use case from my cpp program.

    MLEstimateComponentBasedNormalisation ml_estimator(out_filename_tmp, *measured_projdata, *model_activity_projdata,
                                                       num_eff_iterations, num_iterations, do_geo, do_block,
                                                       do_symmetry_per_block, do_KL, do_display, do_save_to_file);
    ml_estimator.process();

    // Apply the normalization factors
    BinNormalisationPETFromComponents bin_normalisation;
    bin_normalisation.allocate(measured_projdata->get_proj_data_info_sptr(), ml_estimator);
    bin_normalisation.set_up(measured_projdata->get_exam_info_sptr(), measured_projdata->get_proj_data_info_sptr());

    // Compute the projdata
    ProjDataInMemory normalization_projdata(*measured_projdata);
    normalization_projdata.fill(1.F);
    bin_normalisation.undo(normalization_projdata);

Copy link
Collaborator Author

@robbietuk robbietuk Aug 30, 2024

Choose a reason for hiding this comment

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

I acknowledge #1498 (comment), however this does work and provides an alternative allocation of the data.

Copy link
Collaborator

Choose a reason for hiding this comment

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

I'd rather do it the other way around I'm afraid. Your way is very different from anything else in STIR.

I very much prefer to let MLEstimateComponentBasedNormalisation have a shared_ptr<BinNormalisationPETFromComponents> normalisation_sptr member, which we can get via get_output() (returning a shared_ptr<const BinNorm...>). The MLEstimateComponentBasedNormalisation::get_efficiencies() would then just forward to normalisation_sptr->get_efficiencies() (note that I wouldn't let those return shared_ptr then, but references). Any reason not to do it this way?


void
BinNormalisationPETFromComponents::create_proj_data()
Expand Down
Loading
Loading