fix assertion failed: validate_hidpi_factor(dpi_factor) (#607) (#618)

* fix assertion failed: validate_hidpi_factor(dpi_factor) (#607)

* added changelog entry
This commit is contained in:
Dennis Möhlmann 2018-08-02 19:03:15 +02:00 committed by Francesca Frangipane
parent 1a119bdfe9
commit e8e9fa2418
3 changed files with 13 additions and 25 deletions

View file

@ -8,6 +8,7 @@
- On iOS, the `UIApplication` is not started until `Window::new` is called. - On iOS, the `UIApplication` is not started until `Window::new` is called.
- Fixed thread unsafety with cursor hiding on macOS. - Fixed thread unsafety with cursor hiding on macOS.
- On iOS, fixed the size of the `JmpBuf` type used for `setjmp`/`longjmp` calls. Previously this was a buffer overflow on most architectures. - On iOS, fixed the size of the `JmpBuf` type used for `setjmp`/`longjmp` calls. Previously this was a buffer overflow on most architectures.
- On Windows, use cached window DPI instead of repeatedly querying the system. This fixes sporadic crashes on Windows 7.
# Version 0.16.2 (2018-07-07) # Version 0.16.2 (2018-07-07)

View file

@ -8,7 +8,6 @@ use winapi::shared::minwindef::{BOOL, UINT, FALSE};
use winapi::shared::windef::{ use winapi::shared::windef::{
DPI_AWARENESS_CONTEXT, DPI_AWARENESS_CONTEXT,
DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE, DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE,
HDC,
HMONITOR, HMONITOR,
HWND, HWND,
}; };
@ -145,7 +144,11 @@ pub fn dpi_to_scale_factor(dpi: u32) -> f64 {
dpi as f64 / BASE_DPI as f64 dpi as f64 / BASE_DPI as f64
} }
pub unsafe fn get_window_dpi(hwnd: HWND, hdc: HDC) -> u32 { pub unsafe fn get_hwnd_dpi(hwnd: HWND) -> u32 {
let hdc = winuser::GetDC(hwnd);
if hdc.is_null() {
panic!("[winit] `GetDC` returned null!");
}
if let Some(GetDpiForWindow) = *GET_DPI_FOR_WINDOW { if let Some(GetDpiForWindow) = *GET_DPI_FOR_WINDOW {
// We are on Windows 10 Anniversary Update (1607) or later. // We are on Windows 10 Anniversary Update (1607) or later.
match GetDpiForWindow(hwnd) { match GetDpiForWindow(hwnd) {
@ -181,16 +184,6 @@ pub unsafe fn get_window_dpi(hwnd: HWND, hdc: HDC) -> u32 {
} }
} }
// Use this when you have both the HWND and HDC on hand (i.e. window methods)
pub fn get_window_scale_factor(hwnd: HWND, hdc: HDC) -> f64 {
dpi_to_scale_factor(unsafe { get_window_dpi(hwnd, hdc) })
}
// Use this when you only have the HWND (i.e. event handling)
pub fn get_hwnd_scale_factor(hwnd: HWND) -> f64 { pub fn get_hwnd_scale_factor(hwnd: HWND) -> f64 {
let hdc = unsafe { winuser::GetDC(hwnd) }; dpi_to_scale_factor(unsafe { get_hwnd_dpi(hwnd) })
if hdc.is_null() {
panic!("[winit] `GetDC` returned null!");
}
unsafe { get_window_scale_factor(hwnd, hdc) }
} }

View file

@ -9,7 +9,7 @@ use std::sync::mpsc::channel;
use winapi::ctypes::c_int; use winapi::ctypes::c_int;
use winapi::shared::minwindef::{BOOL, DWORD, FALSE, LPARAM, TRUE, UINT, WORD, WPARAM}; use winapi::shared::minwindef::{BOOL, DWORD, FALSE, LPARAM, TRUE, UINT, WORD, WPARAM};
use winapi::shared::windef::{HDC, HWND, LPPOINT, POINT, RECT}; use winapi::shared::windef::{HWND, LPPOINT, POINT, RECT};
use winapi::um::{combaseapi, dwmapi, libloaderapi, winuser}; use winapi::um::{combaseapi, dwmapi, libloaderapi, winuser};
use winapi::um::objbase::COINIT_MULTITHREADED; use winapi::um::objbase::COINIT_MULTITHREADED;
use winapi::um::shobjidl_core::{CLSID_TaskbarList, ITaskbarList2}; use winapi::um::shobjidl_core::{CLSID_TaskbarList, ITaskbarList2};
@ -26,7 +26,7 @@ use {
WindowAttributes, WindowAttributes,
}; };
use platform::platform::{Cursor, PlatformSpecificWindowBuilderAttributes, WindowId}; use platform::platform::{Cursor, PlatformSpecificWindowBuilderAttributes, WindowId};
use platform::platform::dpi::{dpi_to_scale_factor, get_window_dpi, get_window_scale_factor}; use platform::platform::dpi::{dpi_to_scale_factor, get_hwnd_dpi};
use platform::platform::events_loop::{self, EventsLoop, DESTROY_MSG_ID, INITIAL_DPI_MSG_ID}; use platform::platform::events_loop::{self, EventsLoop, DESTROY_MSG_ID, INITIAL_DPI_MSG_ID};
use platform::platform::events_loop::WindowState; use platform::platform::events_loop::WindowState;
use platform::platform::icon::{self, IconType, WinIcon}; use platform::platform::icon::{self, IconType, WinIcon};
@ -417,7 +417,7 @@ impl Window {
#[inline] #[inline]
pub fn get_hidpi_factor(&self) -> f64 { pub fn get_hidpi_factor(&self) -> f64 {
get_window_scale_factor(self.window.0, self.window.1) self.window_state.lock().unwrap().dpi_factor
} }
fn set_cursor_position_physical(&self, x: i32, y: i32) -> Result<(), String> { fn set_cursor_position_physical(&self, x: i32, y: i32) -> Result<(), String> {
@ -773,7 +773,7 @@ impl Drop for Window {
/// A simple non-owning wrapper around a window. /// A simple non-owning wrapper around a window.
#[doc(hidden)] #[doc(hidden)]
#[derive(Clone)] #[derive(Clone)]
pub struct WindowWrapper(HWND, HDC); pub struct WindowWrapper(HWND);
// Send and Sync are not implemented for HWND and HDC, we have to wrap it and implement them manually. // Send and Sync are not implemented for HWND and HDC, we have to wrap it and implement them manually.
// For more info see: // For more info see:
@ -954,13 +954,7 @@ unsafe fn init(
format!("{}", io::Error::last_os_error())))); format!("{}", io::Error::last_os_error()))));
} }
let hdc = winuser::GetDC(handle); WindowWrapper(handle)
if hdc.is_null() {
return Err(CreationError::OsError(format!("GetDC function failed: {}",
format!("{}", io::Error::last_os_error()))));
}
WindowWrapper(handle, hdc)
}; };
// Set up raw input // Set up raw input
@ -974,7 +968,7 @@ unsafe fn init(
} }
} }
let dpi = get_window_dpi(real_window.0, real_window.1); let dpi = get_hwnd_dpi(real_window.0);
let dpi_factor = dpi_to_scale_factor(dpi); let dpi_factor = dpi_to_scale_factor(dpi);
if dpi_factor != guessed_dpi_factor { if dpi_factor != guessed_dpi_factor {
let (width, height): (u32, u32) = dimensions.into(); let (width, height): (u32, u32) = dimensions.into();