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

FIX: maintain reference to related displays to avoid gc #1137

Merged
merged 2 commits into from
Jan 8, 2025

Conversation

ZLLentz
Copy link
Member

@ZLLentz ZLLentz commented Dec 6, 2024

We noticed a niche issue when it comes to using the PyDMRelatedDisplayButton widget outside of a pydm application.

In these cases, the display that gets loaded is orphaned, and will eventually get garbage collected because there are no python references to the display object. This isn't a problem inside the pydm applications because in these cases we either start a new process, which retains a reference to the screen, or we replace our main widget with the new widget, so the window retains a reference.

Users have been reporting that their displays would "crash with no error message" and I tracked it down to this.

In this PR, I create a list and stash these related displays in the list. This means they have one python reference, so they aren't closed automatically.

# Parent the display to avoid garbage collection
display.setParent(self)
display.setWindowFlags(Qt.Window)
display.show()

Choose a reason for hiding this comment

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

display will have already been shown in

loaded_display.show()

Does showing it twice do anything?

Copy link
Member Author

Choose a reason for hiding this comment

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

It doesn't appear without the second show, somehow

@KaushikMalapati
Copy link

This seems to have a side-effect of causing displays opened by a non pydm-app to close whenever their parent display is closed, which did not happen before. You can try this on one of the lucid screens using your branch vs whatever version of pydm dev_conda uses. Maybe a parent was intentionally not set for this reason?

@ZLLentz
Copy link
Member Author

ZLLentz commented Dec 7, 2024

This seems to have a side-effect of causing displays opened by a non pydm-app to close whenever their parent display is closed, which did not happen before. You can try this on one of the lucid screens using your branch vs whatever version of pydm dev_conda uses. Maybe a parent was intentionally not set for this reason?

This is a good point. Maybe this should be implemented differently.

@KaushikMalapati
Copy link

KaushikMalapati commented Dec 7, 2024

This seems to have a side-effect of causing displays opened by a non pydm-app to close whenever their parent display is closed, which did not happen before. You can try this on one of the lucid screens using your branch vs whatever version of pydm dev_conda uses. Maybe a parent was intentionally not set for this reason?

This is a good point. Maybe this should be implemented differently.

I think this is intentional pyqt behavior. widgets without a parent are considered top-level and can exist independently (garbage collection notwithstanding) - widgets with a parent only exist as long as their parent does no matter what window they're in.

One alternative is to just add a list member to this class and add any non-pydm opened displays to the list without setting their parent. This stops garbage collection and keeps the new display top-level.

@ZLLentz
Copy link
Member Author

ZLLentz commented Dec 11, 2024

add a list member to this class and add any non-pydm opened displays to the list without setting their parent

I'll experiment a bit. My intuition is that stashing these in a list could be problematic in a memory-management sense for displays that are left open for a long time, if the user clicks the button a lot and closes the displays.

@ZLLentz
Copy link
Member Author

ZLLentz commented Dec 12, 2024

It turns out to be pretty simple to prune the list over time to avoid the reference list growing forever.

@ZLLentz ZLLentz changed the title FIX: parent dialog related displays to avoid gc FIX: maintain reference to related displays to avoid gc Dec 12, 2024
@ZLLentz
Copy link
Member Author

ZLLentz commented Jan 7, 2025

I'm pretty happy with where this PR ended up last year.
Any comment from the core pydm maintainers?

I think the only thing left to possibly add would be an extra unit test that runs through this code path and tries to execute the garbage collector, confirming that the subscreen stays open. This might be at least a little bit tricky to write.

Copy link
Collaborator

@jbellister-slac jbellister-slac left a comment

Choose a reason for hiding this comment

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

Looks good to me and works as expected. A test would be nice, but I'm good with that just being a potential follow up later

@jbellister-slac jbellister-slac merged commit 52f581e into slaclab:master Jan 8, 2025
19 checks passed
@ZLLentz ZLLentz deleted the fix_reldisp_nonpydm branch January 8, 2025 17:38
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

Successfully merging this pull request may close these issues.

3 participants