From 2b39e744777eb1fd7ee2ce7e35c44dcccfc7d193 Mon Sep 17 00:00:00 2001 From: Divy Srivastava Date: Wed, 6 Oct 2021 05:43:56 +0530 Subject: [PATCH] fix(ext/ffi): formatting dlopen errors on Windows (#12301) --- Cargo.lock | 1 + ext/ffi/Cargo.toml | 3 ++ ext/ffi/lib.rs | 102 ++++++++++++++++++++++++++++++++++++++++++++- 3 files changed, 104 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index a9237ec2a4..10bd939df2 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -779,6 +779,7 @@ dependencies = [ "dlopen", "serde", "tokio", + "winapi 0.3.9", ] [[package]] diff --git a/ext/ffi/Cargo.toml b/ext/ffi/Cargo.toml index 8799cd2e0d..73c4c8c26f 100644 --- a/ext/ffi/Cargo.toml +++ b/ext/ffi/Cargo.toml @@ -19,3 +19,6 @@ dlopen = "0.1.8" libffi = { version = "=0.0.7", package = "deno-libffi" } serde = { version = "1.0.129", features = ["derive"] } tokio = { version = "1.10.1", features = ["full"] } + +[target.'cfg(windows)'.dependencies] +winapi = { version = "0.3.9", features = ["errhandlingapi", "minwindef", "ntdef", "winbase", "winnt"] } diff --git a/ext/ffi/lib.rs b/ext/ffi/lib.rs index ebf5572a71..b5505fb0cf 100644 --- a/ext/ffi/lib.rs +++ b/ext/ffi/lib.rs @@ -1,5 +1,6 @@ // Copyright 2021 the Deno authors. All rights reserved. MIT license. +use deno_core::error::anyhow; use deno_core::error::bad_resource_id; use deno_core::error::AnyError; use deno_core::include_js_files; @@ -279,6 +280,82 @@ struct FfiLoadArgs { symbols: HashMap, } +// `path` is only used on Windows. +#[allow(unused_variables)] +pub(crate) fn format_error(e: dlopen::Error, path: String) -> String { + match e { + #[cfg(target_os = "windows")] + // This calls FormatMessageW with library path + // as replacement for the insert sequences. + // Unlike libstd which passes the FORMAT_MESSAGE_IGNORE_INSERTS + // flag without any arguments. + // + // https://github.com/denoland/deno/issues/11632 + dlopen::Error::OpeningLibraryError(e) => { + use std::ffi::OsStr; + use std::os::windows::ffi::OsStrExt; + use winapi::shared::minwindef::DWORD; + use winapi::shared::ntdef::WCHAR; + use winapi::shared::winerror::ERROR_INSUFFICIENT_BUFFER; + use winapi::um::errhandlingapi::GetLastError; + use winapi::um::winbase::FormatMessageW; + use winapi::um::winbase::FORMAT_MESSAGE_ARGUMENT_ARRAY; + use winapi::um::winbase::FORMAT_MESSAGE_FROM_SYSTEM; + use winapi::um::winnt::LANG_SYSTEM_DEFAULT; + use winapi::um::winnt::MAKELANGID; + use winapi::um::winnt::SUBLANG_SYS_DEFAULT; + + let err_num = match e.raw_os_error() { + Some(err_num) => err_num, + // This should never hit unless dlopen changes its error type. + None => return e.to_string(), + }; + + // Language ID (0x0800) + let lang_id = + MAKELANGID(LANG_SYSTEM_DEFAULT, SUBLANG_SYS_DEFAULT) as DWORD; + + let mut buf = vec![0 as WCHAR; 500]; + + let path = OsStr::new(&path) + .encode_wide() + .chain(Some(0).into_iter()) + .collect::>(); + + let arguments = [path.as_ptr()]; + + loop { + unsafe { + let length = FormatMessageW( + FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ARGUMENT_ARRAY, + std::ptr::null_mut(), + err_num as DWORD, + lang_id as DWORD, + buf.as_mut_ptr(), + buf.len() as DWORD, + arguments.as_ptr() as _, + ); + + if length == 0 { + let err_num = GetLastError(); + if err_num == ERROR_INSUFFICIENT_BUFFER { + buf.resize(buf.len() * 2, 0); + continue; + } + + // Something went wrong, just return the original error. + return e.to_string(); + } + + let msg = String::from_utf16_lossy(&buf[..length as usize]); + return msg; + } + } + } + _ => e.to_string(), + } +} + fn op_ffi_load( state: &mut deno_core::OpState, args: FfiLoadArgs, @@ -287,11 +364,14 @@ fn op_ffi_load( where FP: FfiPermissions + 'static, { + let path = args.path; + check_unstable(state, "Deno.dlopen"); let permissions = state.borrow_mut::(); - permissions.check(&args.path)?; + permissions.check(&path)?; + + let lib = Library::open(&path).map_err(|e| anyhow!(format_error(e, path)))?; - let lib = Library::open(args.path)?; let mut resource = DynamicLibraryResource { lib, symbols: HashMap::new(), @@ -422,3 +502,21 @@ async fn op_ffi_call_nonblocking( .await .unwrap() } + +#[cfg(test)] +mod tests { + #[cfg(target_os = "windows")] + #[test] + fn test_format_error() { + use super::format_error; + + // BAD_EXE_FORMAT + let err = dlopen::Error::OpeningLibraryError( + std::io::Error::from_raw_os_error(0x000000C1), + ); + assert_eq!( + format_error(err, "foo.dll".to_string()), + "foo.dll is not a valid Win32 application.\r\n".to_string(), + ); + } +}