From a79c79209bfdbd1700c734f9a9e22e0fb51efb93 Mon Sep 17 00:00:00 2001 From: "Jesse A. Tov" Date: Wed, 1 Dec 2021 17:23:15 -0600 Subject: [PATCH] Add tests for ClosureOnce. MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit We check that we can call a ClosureOnce once successfully, but it’s an error to call it twice. Normally if this error occurs, the program exits (code 2), because it isn’t safe to panic when calling a callback. However, in cfg(test) this error results in a panic so that we can test the error using #[should_panic]. --- libffi-rs/src/high/mod.rs | 38 ++++++++++++++++++++++++++++++++------ 1 file changed, 32 insertions(+), 6 deletions(-) diff --git a/libffi-rs/src/high/mod.rs b/libffi-rs/src/high/mod.rs index c7730020..d54bd94b 100644 --- a/libffi-rs/src/high/mod.rs +++ b/libffi-rs/src/high/mod.rs @@ -92,8 +92,7 @@ macro_rules! define_closure_mod { pub mod $module { use std::any::Any; use std::marker::PhantomData; - use std::{mem, process, ptr}; - use std::io::{self, Write}; + use std::{mem, ptr}; use super::*; use crate::{low, middle}; @@ -387,10 +386,19 @@ macro_rules! define_closure_mod { } }); } else { - // There is probably a better way to abort here. - let _ = - io::stderr().write(b"FnOnce closure already used"); - process::exit(2); + eprintln!( + "FnOnce closure already used (userdata {:?})", + userdata as *const Option); + + // When testing we need to panic so that we can + // catch it with #[should_panic]. + #[cfg(test)] + panic!("FnOnce closure already used"); + + // Otherwise, we exit to avoid panicking across + // an FFI boundary. + #[cfg(not(test))] + std::process::exit(2); } } } @@ -545,4 +553,22 @@ mod test { assert_eq!(6, counter.call(1)); assert_eq!(8, counter.call(2)); } + + #[test] + fn once_ok() { + let closure = ClosureOnce1::new(|x| x + 1); + let inc = closure.code_ptr(); + + assert_eq!(4, inc.call(3)); + } + + #[test] + #[should_panic] + fn once_error() { + let closure = ClosureOnce1::new(|x| x + 1); + let inc = closure.code_ptr(); + + assert_eq!(4, inc.call(3)); + assert_eq!(5, inc.call(4)); + } }