Issues tracking two tools at once

Hello everyone.

I am working on developing a small tracking script to store specific tracking data, and in my specific case I need to track two tools at the same time. One of them is a pointer, and the other can be anything. Tracking works fine for both tools separately, and when they are both in view it also works fine, however, the first time the pointer tool enters in view it ruins the tracking for the other tool.

For some reason, the tracking from that moment onwards seems to be linked with the pointer tool. Anytime it is not in view, the other tool does not update its transform information anymore. The moment the pointer comes back into view, then everything works fine. I’m not sure if this behaviour is being caused by how the data is transmitted to slicer, or by the data handling from the module. Any tips or information on this would be really appreciated.

I am using Slicer 5.8 for this, however I get the same behaviour in 5.6.2

If there is anyway I can somehow get a trace of events that happen internally within the OpenIGTLinkIF module whenever handling tracking data, that would probably help to figure out the issue, is that possible without building from source?

A quick clarification, in the first part of the post I said “…the first time the pointer tool enters in view it ruins the tracking for the other tool.”, however I meant “the first time the pointer tool enters in view while the secondary is in view aswell it ruins the tracking for the other tool.”

After taking a closer look and printing out the contents of the query response (IGTLTrackingDataBundleNode) and the elements inside it (LinearTransformNode), there are no updates happening to the transform matrices of the secondary tool once the issue starts. If the secondary tool is in view on its own after the issue, the only thing that updates inside the LinearTransformNode is the MTime, which keeps going up as usual.

At first I thought this might be an issue with the device I am getting the tracking data from, however when trying to communicate to the device through a data recording script (the Example for Tracker Client Program in C++ provided by OpenIGTLink), I can see inside the TDATA that all of the transform data updates just fine.

This makes me think that there’s some issue with how the Slicer module is currently handling the data it receives, or maybe the checks it uses to update nodes. I haven’t been able to find anything in the C++ files that seems like it could cause this, but I am not as experienced in C++. Would really appreciate any help.

I have figured out the issue after building everything from source and doing some debugging.

To give a summary of everything once again, I was trying to track two tools, Tool1 and Tool2 using information obtained from OpenIGTLink. Separately they are both tracked fine, however the moment they both come into view at the same time, the tracking for Tool2 (when Tool1 is not in view) is broken. It does not update its transform data anymore (even though the MTime variable does). After some debugging, I figured out that the positioning of the data inside the igtlioTrackingDataConverter::ContentData object alongside how the transforms are updated to be the cause of this issue.

ContentData has a map which carries the information for each tracked tool.

struct ContentData
{
  std::map<int, ContentEntry> trackingDataElements;
};

However when multiple tools apear, the position of those tools in that map changes. They are seemingly sorted alphabetically, so if both tools appear at once, then the map updates to show Tool1 first then Tool2. However, if Tool1 goes out of view and Tool2 remains, the map still has a version of Tool2 at its old position. So if we loop through the map, we would get the new and updated Tool2 with the correct transform, then the old Tool2 with the old and wrong transform.

Due to how iterating and updating is currently handled in void vtkMRMLIGTLConnectorNode::ProcessIncomingDeviceModifiedEvent at the moment (coming from SlicerOpenIGTLink/OpenIGTLinkIF/MRML) , the correct version is handled first, the tool position is updated correctly, and then the old version is handled after and simply reverts the position back. This keeps happening until Tool1 comes back into view with Tool2, where now Tool2 updates normally again (as the maps only version of Tool2 is now correct).

At the moment I have remedied this by adding the following code:

for (int b = 0; b < content.trackingDataElements.size()-1; b++) {
    if (content.trackingDataElements[b].deviceName.compare(content.trackingDataElements[b + 1].deviceName) == 0)
    {
        content.trackingDataElements.erase(b + 1);
    }
}   

This is placed between
igtlioTrackingDataConverter::ContentData content = tdataDevice->GetContent(); and
for (auto iter = content.trackingDataElements.begin(); iter != content.trackingDataElements.end(); iter++) inside the else if case handling “TDATA” in the void vtkMRMLIGTLConnectorNode::ProcessIncomingDeviceModifiedEvent function.

I think the easiest way to solve this would be to have the map keys be something other than integers, since they have no guarantee that the same tool is not updated twice.

I would love to get some suggestions on this as I plan to make a pull request to hopefully fix this issue in a later version of Slicer.

1 Like

Thanks for the detective work, this is very useful. TDATA messages are not used very often, probably that’s why this error has not been noticed so far.

Why does trackingDataElements contain multiple entries for the same tool?

  • A. the TDATA message contained multiple entries for the same tool
  • B. entries remain there from previous TDATA messages

If A) then the sender of the message should be fixed (and maybe Slicer/OpenIGTLink/OpenIGTLinkIO could be made more robust).

If B) then probably we should clear out the trackingDataElements before starting to process a TDATA message.

I will clarify whether it is A) or B) tomorrow after taking another look at the data extracted through the “Example for Tracker Client Program” script.

I took a look at the int igtlioTrackingDataConverter::fromIGTL() function inside OpenIGTLinkIO/Converter/igtlioTrackingDataConverter.cxx and the data coming from igtl::TrackingDataMessage::Pointer transMsg (which is copying the information straight from the igtl::MessageBase source object). There are no duplicates inside, so I think the issue is somewhere within how trackingDataElements is updated.

I will have more time next week to investigate further and perhaps introduce a better solution than just deleting duplicate names from trackingDataElements.