macOS: Fix monitors connected via certain Thunderbolt hubs (#4207)
Instead of panicking, raise a warning and return `None` or similar. Co-authored-by: RJ <rj@metabrew.com>
This commit is contained in:
parent
1800fa1670
commit
aa83726c72
3 changed files with 29 additions and 9 deletions
|
|
@ -256,3 +256,4 @@ changelog entry.
|
||||||
- On Wayland, apply fractional scaling to custom cursors.
|
- On Wayland, apply fractional scaling to custom cursors.
|
||||||
- On macOS, fixed `VideoMode::refresh_rate_millihertz` for fractional refresh rates.
|
- On macOS, fixed `VideoMode::refresh_rate_millihertz` for fractional refresh rates.
|
||||||
- On macOS, store monitor handle to avoid panics after going in/out of sleep.
|
- On macOS, store monitor handle to avoid panics after going in/out of sleep.
|
||||||
|
- On macOS, allow certain invalid monitor handles and return `None` instead of panicking.
|
||||||
|
|
|
||||||
|
|
@ -141,10 +141,18 @@ impl MonitorHandle {
|
||||||
let refresh_rate_millihertz = self.refresh_rate_millihertz();
|
let refresh_rate_millihertz = self.refresh_rate_millihertz();
|
||||||
let monitor = self.clone();
|
let monitor = self.clone();
|
||||||
|
|
||||||
let array = unsafe { CGDisplayCopyAllDisplayModes(self.display_id(), None) }
|
let array = unsafe { CGDisplayCopyAllDisplayModes(self.display_id(), None) };
|
||||||
.expect("failed to get list of display modes");
|
let modes = if let Some(array) = array {
|
||||||
// SAFETY: `CGDisplayCopyAllDisplayModes` is documented to return an array of display modes.
|
// SAFETY: `CGDisplayCopyAllDisplayModes` is documented to return an array of
|
||||||
let modes = unsafe { CFRetained::cast_unchecked::<CFArray<CGDisplayMode>>(array) };
|
// display modes.
|
||||||
|
unsafe { CFRetained::cast_unchecked::<CFArray<CGDisplayMode>>(array) }
|
||||||
|
} else {
|
||||||
|
// Occasionally, certain CalDigit Thunderbolt Hubs report a spurious monitor during
|
||||||
|
// sleep/wake/cycling monitors. It tends to have null or 1 video mode only.
|
||||||
|
// See <https://github.com/bevyengine/bevy/issues/17827>.
|
||||||
|
warn!(monitor = ?self, "failed to get a list of display modes");
|
||||||
|
CFArray::empty()
|
||||||
|
};
|
||||||
|
|
||||||
modes.into_iter().map(move |mode| {
|
modes.into_iter().map(move |mode| {
|
||||||
let cg_refresh_rate_hertz = unsafe { CGDisplayMode::refresh_rate(Some(&mode)) };
|
let cg_refresh_rate_hertz = unsafe { CGDisplayMode::refresh_rate(Some(&mode)) };
|
||||||
|
|
@ -165,9 +173,14 @@ impl MonitorHandle {
|
||||||
let uuid = self.uuid();
|
let uuid = self.uuid();
|
||||||
NSScreen::screens(mtm).into_iter().find(|screen| {
|
NSScreen::screens(mtm).into_iter().find(|screen| {
|
||||||
let other_native_id = get_display_id(screen);
|
let other_native_id = get_display_id(screen);
|
||||||
// Display ID just fetched from live NSScreen, should be fine to unwrap.
|
if let Some(other) = MonitorHandle::new(other_native_id) {
|
||||||
let other = MonitorHandle::new(other_native_id).expect("invalid display ID");
|
uuid == other.uuid()
|
||||||
uuid == other.uuid()
|
} else {
|
||||||
|
// Display ID was just fetched from live NSScreen, but can still result in `None`
|
||||||
|
// with certain Thunderbolt docked monitors.
|
||||||
|
warn!(other_native_id, "comparing against screen with invalid display ID");
|
||||||
|
false
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1756,8 +1756,14 @@ impl WindowDelegate {
|
||||||
// Allow directly accessing the current monitor internally without unwrapping.
|
// Allow directly accessing the current monitor internally without unwrapping.
|
||||||
pub(crate) fn current_monitor_inner(&self) -> Option<MonitorHandle> {
|
pub(crate) fn current_monitor_inner(&self) -> Option<MonitorHandle> {
|
||||||
let display_id = get_display_id(&*self.window().screen()?);
|
let display_id = get_display_id(&*self.window().screen()?);
|
||||||
// Display ID just fetched from live NSScreen, should be fine to unwrap.
|
if let Some(monitor) = MonitorHandle::new(display_id) {
|
||||||
Some(MonitorHandle::new(display_id).expect("invalid display ID"))
|
Some(monitor)
|
||||||
|
} else {
|
||||||
|
// NOTE: Display ID was just fetched from live NSScreen, but can still result in `None`
|
||||||
|
// with certain Thunderbolt docked monitors.
|
||||||
|
warn!(display_id, "got screen with invalid display ID");
|
||||||
|
None
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue