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

Where are "passes" looped? #10

Open
brunoriemenschneider opened this issue Aug 20, 2020 · 3 comments
Open

Where are "passes" looped? #10

brunoriemenschneider opened this issue Aug 20, 2020 · 3 comments

Comments

@brunoriemenschneider
Copy link

brunoriemenschneider commented Aug 20, 2020

Hi,

I have converted some GE Scanarchives with several "passes", as GE calls them. In Orchestra, all passes can be recovered from the Scanarchive. However, after conversion to ismrmrd, I do not find the passes except for the first - the corresponding slices of further passes contain 0's. Tried in Matlab and Python.
While there are the variables contrasts and repetitions with values > 1 in encoding.encodingLimits, there is no data for contrast and repetition > 1 (i.e., only 0's). There are no other encodingLimits that could be looped through.

System version is DV26.0_R05_2008.a
Orchestra used with ge_to_ismrmrd: 1.7

@roopchansinghv
Copy link
Contributor

The current GE converter implementation has no explicit handling of pass loops.

The current ScanArchive handling routine reads how many acquisition packets are in the ScanArchive, and iterates over all packets reported back.

What you could possibly try is to figure out the number of passes in your acquisition. From what I could tell, it would be similar to how the converter currently determines the slice that acquisition belongs to. That code is:

sliceID = GERecon::Acquisition::GetPacketValue(packetContents.sliceNumH, packetContents.sliceNumL);

You would probably need to do something similar for identifying your pass, i.e. something like:

passID = GERecon::Acquisition::GetPacketValue(packetContents.passSourceL, packetContents.passDestL);

You can the assign the passID to segment acquisition encoding counter for ISMRMRD, something along the lines of:

idx.segment = passID;

and see if that allows all data in all passes to be encoded.

In the documentation I could see, anything else referring to passes refer to the older raw data format (P-Files), or to specific applications that uses passes. The above was the only generic, non-application specific reference to "passes" I could find.

Let me know if this works. If it does, I can patch in the above code.

@brunoriemenschneider
Copy link
Author

brunoriemenschneider commented Aug 21, 2020

According to our techs who acquired the data these were acquired with FLAIR product sequences (underlying sequence "FSE-XL").

Suggested changes lead to

[...]/ismrmrd/ge_to_ismrmrd/src/GenericConverter.cpp: In member function ‘virtual std::vectorISMRMRD::Acquisition PfileToIsmrmrd::GenericConverter::getAcquisitions(GERecon::ScanArchivePointer&, unsigned int)’:
[...]/ismrmrd/ge_to_ismrmrd/src/GenericConverter.cpp:218:3: error: ‘passID’ was not declared in this scope
passID = GERecon::Acquisition::GetPacketValue(packetContents.passSourceL, packetContents.passDestL);
^
[...]/ismrmrd/ge_to_ismrmrd/src/GenericConverter.cpp:218:64: error: ‘const struct GERecon::Acquisition::ProgrammableControlPacket’ has no member named ‘passSourceL’
passID = GERecon::Acquisition::GetPacketValue(packetContents.passSourceL, packetContents.passDestL);
^
[...]/ismrmrd/ge_to_ismrmrd/src/GenericConverter.cpp:218:92: error: ‘const struct GERecon::Acquisition::ProgrammableControlPacket’ has no member named ‘passDestL’
passID = GERecon::Acquisition::GetPacketValue(packetContents.passSourceL, packetContents.passDestL);

@roopchansinghv
Copy link
Contributor

roopchansinghv commented Aug 24, 2020

The first error is trivial - it just requires the 'passID' variable to be defined.

However, the subsequent errors are a bit more troubling, as it seems to indicate, that for the application (you mentioned FSE-XL), that passSourceL and passDestL were not defined.

Could you try this to see if this code (put at approximately the same place as the above) reports the correct number of passes that should be in your ScanArchive:

std::cout << "Number of passes is: " << processingControl->Value("NumPasses") << std::endl ;

If it does, then what you can try is to create an instance of the sliceTable describing your acquisition. So below the line:

GERecon::Control::ProcessingControlPointer processingControl = controlSource->CreateOrchestraProcessingControl();

add this line:

auto sliceTable = GERecon::Legacy::LxSliceTable(*lxData);

Then, look through Orchestra's documentation to see how you can access the elements with the LxSliceTable object to tell you which slice / pass / slice-in-pass you are dealing with.

I did a bit of digging at the end of last week and this morning, but this needs more time than I am able to spend on this right now. I am trying to get geometry information from GE raw data into the ISMRMRD header in a reasonable way, so people can make DICOMs as output ...

If you come up with a path to access the slice table data, I am happy to review a pull request to add to the existing functionality of the converter.

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

No branches or pull requests

2 participants