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

making scatter work with blocks #1006

Merged
merged 24 commits into from
Jun 27, 2022
Merged

Conversation

danieldeidda
Copy link
Collaborator

No description provided.

Copy link
Collaborator

@KrisThielemans KrisThielemans left a comment

Choose a reason for hiding this comment

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

some comments as discussed. Also probably drop changes to inverse_SSRB as discussed (although we probably want to do that nearer to the end of this PR)

@danieldeidda
Copy link
Collaborator Author

I am a bit confused by the following:

error("ScatterSimulation: output projection data incompatible with what was used for set_up()");

Are we meant to provide a template which already describes the downsample scanner?

@KrisThielemans
Copy link
Collaborator

process_data() requires set_up() first. This has logic to create the template via downsampling (if you didn't give one yet I guess).

@danieldeidda
Copy link
Collaborator Author

process_data() requires set_up() first. This has logic to create the template via downsampling (if you didn't give one yet I guess).

OK so it should work even if we don't give a template in input? In my mind the template was the original scanner that then is downsampled

@KrisThielemans
Copy link
Collaborator

for cylindrical, the template is optional (as far as I recall), i.e. it will downsample then according to (optional) parameters.

It'd be nice to end up there in that situation for blocksoncylindrical as well, which I think you nearly had it.

Before @NikEfth created the downsampling functionality, we had to create the template by hand. Now it should be optional.

@danieldeidda
Copy link
Collaborator Author

danieldeidda commented Mar 23, 2022

for cylindrical, the template is optional (as far as I recall), i.e. it will downsample then according to (optional) parameters.

It'd be nice to end up there in that situation for blocksoncylindrical as well, which I think you nearly had it.

Before @NikEfth created the downsampling functionality, we had to create the template by hand. Now it should be optional.

It seems that if I don't give a template name then post_processing() which calls set_output_proj_data(const std::string& filename) throws the error:

set_output_proj_data(const std::string& filename)
{

    if(is_null_ptr(this->proj_data_info_cyl_noarc_cor_sptr))
    {
        error("ScatterSimulation: Template ProjData has not been set. Aborting.");
    }

I do have the downsample ready but it stops before it can even do the downsampling

@KrisThielemans
Copy link
Collaborator

Possibly that's a bug.then. I believe we do run ScatterSimulation without template in ScatterEstimation. Might need some debugging. sorry

@danieldeidda
Copy link
Collaborator Author

danieldeidda commented Mar 24, 2022

Possibly that's a bug.then. I believe we do run ScatterSimulation without template in ScatterEstimation. Might need some debugging. sorry

OK after some debugging I think for the ScattereEstimation it makes sense to allow no template because the info about the scanner are provided in that class and post_processing is not called, therefore we don't encounter that error. However, when we run the simulate_scatter utility on its own we need a template otherwise it does not have any clue about scanner geometry. I think the easiest solution would be to re-set output_proj_data after the downsampling in this way everything runs and the simulated scatter looks something reasonably alike to a cylinder

@KrisThielemans
Copy link
Collaborator

for the ScattereEstimation it makes sense to allow no template because the info about the scanner are provided in that class and post_processing is not called, therefore we don't encounter that error. However, when we run the simulate_scatter utility on its own we need a template otherwise it does not have any clue about scanner geometry.

yes, makes sense.

re-set output_proj_data after the downsampling in this way everything runs and the simulated scatter looks something reasonably alike to a cylinder

I'm 100% lost. reset where? ScatterEstimation ,but that works already? ScatterSimulation, but you just said it cannot work.

@danieldeidda
Copy link
Collaborator Author

I'm 100% lost. reset where? ScatterEstimation ,but that works already? ScatterSimulation, but you just said it cannot work.

In ScatterSimulation, we give a template which is the original scanner (out_projdata is set equal to that in post_processing()), then we downsample this scanner and change out_projdata info with the downsampled one.

@KrisThielemans
Copy link
Collaborator

ok. I think it does make sense to (optionally?) let ScatterSimulation do the downsampling, but then I also think it has to do the upsampling, i.e. output has to be the same size as the template. This would be good, but obviously out of scope for this PR.

- add condition on Geometry for contructor of ProjDataCylindrical as it gets called by ProjDataGeneric
- revert inversSSRB
- downsample_scatter is by default true and  downsampled_num_dets=-1 to downsample the view downsampled_num_dets needs to be set >0
@danieldeidda
Copy link
Collaborator Author

This seems to run OK. However, if we look at the downsample scatter and the upsample one in the figure below, we can see that the downsample seem to have some empty part which I am not sure is correct

image

@KrisThielemans
Copy link
Collaborator

who knows how Amide displays output of extract_segments. "image" dimensions might not make sense anyway. A better test is to compute the scatter with full sampling (in m) and compare that to the upsampled sino. Clearly, you will have to use something with asymmetry in z

@danieldeidda
Copy link
Collaborator Author

who knows how Amide displays output of extract_segments. "image" dimensions might not make sense anyway. A better test is to compute the scatter with full sampling (in m) and compare that to the upsampled sino. Clearly, you will have to use something with asymmetry in z

I am running simulate_scatter with original scanner. It has been running for 25h with openMP on. does this seem normal?

@KrisThielemans
Copy link
Collaborator

it will take a long time yes, certainly if you have it in full 3D. That's the whole reason we do the downsampling... You should be able to monitor progress in the log files. It'll give an estimate of run-time, which initially is too pessimistic as it ignores the benefit of the caching of "scatter-point <-> detector" lines, but it would still give you some idea.

@danieldeidda
Copy link
Collaborator Author

it will take a long time yes, certainly if you have it in full 3D. That's the whole reason we do the downsampling... You should be able to monitor progress in the log files. It'll give an estimate of run-time, which initially is too pessimistic as it ignores the benefit of the caching of "scatter-point <-> detector" lines, but it would still give you some idea.

I think this look weird, is it possible that the z_shift needs to be applied again?

image

@danieldeidda
Copy link
Collaborator Author

danieldeidda commented Apr 7, 2022

it will take a long time yes, certainly if you have it in full 3D. That's the whole reason we do the downsampling... You should be able to monitor progress in the log files. It'll give an estimate of run-time, which initially is too pessimistic as it ignores the benefit of the caching of "scatter-point <-> detector" lines, but it would still give you some idea.

I think this look weird, is it possible that the z_shift needs to be applied again?

while the following is the same but using Cylindrical geometry
image

@KrisThielemans
Copy link
Collaborator

I guess it'd be good to see coronal slice of the activity/attenuation images.

Might need a hunt through the ScatterSimulation code how it gets the coordinates, and in particular if there's something checks on cylindrical or not.

@danieldeidda
Copy link
Collaborator Author

I guess it'd be good to see coronal slice of the activity/attenuation images.

Might need a hunt through the ScatterSimulation code how it gets the coordinates, and in particular if there's something checks on cylindrical or not.

I had put this check for Cylindrical only but pheraps needs to be there and in fact detector_coord_A.z() (and B) is different than 0

if(this->proj_data_info_cyl_noarc_cor_sptr->get_scanner_sptr()->get_scanner_geometry()=="Cylindrical"){

assert(detector_coord_A.z() == 0);

assert(detector_coord_B.z() == 0);

before this line find_cartesian_coordinates_of_detection(detector_coord_A, detector_coord_B, Bin(0, 0, 0, 0)); is called which adds a z_shift further away from zero

@danieldeidda
Copy link
Collaborator Author

danieldeidda commented Apr 8, 2022

Might need a hunt through the ScatterSimulation code how it gets the coordinates, and in particular if there's something checks on cylindrical or not.

So I had a look at the code and there is a shift in the following:

this->shift_detector_coordinates_to_origin =
CartesianCoordinate3D<float>(this->proj_data_info_cyl_noarc_cor_sptr->get_m(Bin(0, 0, 0, 0)), 0, 0);

when I change the sign of this shift I get something that looks more like the equivalent in the cylindrica geometry
image

@KrisThielemans
Copy link
Collaborator

unfortunately that makes little sense to me. You did a lot of hard work to make sure that find_cartesian* and get_m give the same results for cylindrical and block. That then implies that the shift has to be the same in both cases. So I'm not sure what's going on.

Possibly you could enable the debug code above, removing lines

if(this->proj_data_info_cyl_noarc_cor_sptr->get_scanner_sptr()->get_scanner_geometry()=="Cylindrical"){
and
if(this->proj_data_info_cyl_noarc_cor_sptr->get_scanner_sptr()->get_scanner_geometry()=="Cylindrical")
as those were appropriate before we shifted everything back.

@danieldeidda
Copy link
Collaborator Author

unfortunately that makes little sense to me. You did a lot of hard work to make sure that find_cartesian* and get_m give the same results for cylindrical and block. That then implies that the shift has to be the same in both cases. So I'm not sure what's going on.
Possibly you could enable the debug code above, removing lines

if(this->proj_data_info_cyl_noarc_cor_sptr->get_scanner_sptr()->get_scanner_geometry()=="Cylindrical"){

and

if(this->proj_data_info_cyl_noarc_cor_sptr->get_scanner_sptr()->get_scanner_geometry()=="Cylindrical")

as those were appropriate before we shifted everything back.
L394 I have already removed and works fine but the 385 will fail. the z position appears to be -axial_length

or pheraps we could have a check like in L385 and if is not zero calculate the difference and force it zero? I know it sounds ugly even while I am writing :)

@KrisThielemans
Copy link
Collaborator

yup. ugly 😁. My interpretation is that this check has to be satisfied, as otherwise there is something wrong with the coordinate systems. No?

By the way, I feel there's no need to quote whole messages if the reply is "in sequence".

…te the shift. All instances have inverted sign;

- fixed axial upsampling
- change scanner_lenght in ScatterSimulation to include gaps
@danieldeidda
Copy link
Collaborator Author

This now seems to be working. Although I am a bit confused with the sampling in particular the line below:

return get_scanner_ptr()->get_ring_spacing()/2; // TODO currently restricted to span=1 get_num_axial_poss_per_ring_inc(segment_num);

to me this should not be divided by 2

@KrisThielemans
Copy link
Collaborator

./run_scatter_tests.sh is causing time-outs (run time of more than 5 hours)

@KrisThielemans
Copy link
Collaborator

The "2d" sino output is not SSRB'd.

danieldeidda and others added 3 commits June 14, 2022 13:38
Co-authored-by: Kris Thielemans <KrisThielemans@users.noreply.github.com>
Co-authored-by: Kris Thielemans <KrisThielemans@users.noreply.github.com>
@KrisThielemans
Copy link
Collaborator

looks like it's still taking too much time. Something else I guess. Did you run the test locally?

@danieldeidda
Copy link
Collaborator Author

not yet. Still waiting for my computer

@KrisThielemans
Copy link
Collaborator

My suggestions cause the tests to fail with

ERROR: ProjDataInfoCylindrical::get_ring_pair_for_segment_axial_pos_num does not work for this type of sampled data

which is thrown in get_ring_pair_for_segment_axial_pos_num

if (!sampling_corresponds_to_physical_rings)
error("ProjDataInfoCylindrical::get_ring_pair_for_segment_axial_pos_num does not work for this type of sampled data");

So now I'm confused. This code needs segment_axial_pos_to_ring1_plus_ring2 which previously we did initialise, but potentially wrongly, as it used m_offset etc, which does not necessarily make sense for the block scanner... There seems to be something deeper that needs fixing.

@danieldeidda
Copy link
Collaborator Author

I am kind of lost here :D

@KrisThielemans
Copy link
Collaborator

We will fix the initialisation of the ring diff arrays in a future PR.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants