summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--Cargo.toml7
-rw-r--r--src/lib.rs103
-rw-r--r--src/mock/mod.rs111
-rw-r--r--src/pluginfo.rs39
4 files changed, 142 insertions, 118 deletions
diff --git a/Cargo.toml b/Cargo.toml
index cde21f6..9f1325e 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -18,5 +18,8 @@ nightly_tests = ["dep:selfref"]
 [dependencies]
 libc = "0.2"
 impl_trait = "0.1"
-ltptr = "0.1.2"
-selfref = { version = "0.1.3", optional = true }
+ltptr = "0.1.5"
+selfref = { version = "0.4.0", optional = true }
+
+[patch.crates-io]
+ltptr = { path = "/home/soniex2/git/selfhosted/rust.ltptr/" }
diff --git a/src/lib.rs b/src/lib.rs
index 00d0845..0ef9354 100644
--- a/src/lib.rs
+++ b/src/lib.rs
@@ -15,8 +15,7 @@
 // along with this program.  If not, see <https://www.gnu.org/licenses/>.
 
 #![cfg_attr(feature="nightly_tests", feature(c_variadic))]
-#![cfg_attr(feature="nightly_tests", feature(generic_associated_types))]
-//#![deny(unsafe_op_in_unsafe_fn)]
+#![deny(unsafe_op_in_unsafe_fn)]
 
 //! Write hexchat plugins in Rust!
 //!
@@ -257,6 +256,9 @@ pub use libc::{c_char, c_int, c_void, time_t};
 #[doc(hidden)]
 pub use ltptr::MutLtPtr;
 
+use ltptr::ConstLtPtr;
+use ltptr::FromLtPtr;
+
 // ****** //
 // PUBLIC //
 // ****** //
@@ -2213,12 +2215,16 @@ fn check_pluginpref_var(var: impl Into<Vec<u8>>) -> Result<CString, PluginPrefEr
 // *********************** //
 
 #[doc(hidden)]
-pub unsafe fn hexchat_plugin_init<'ph, T>(ph: MutLtPtr<'ph, RawPh<'ph>>,
-                                     plugin_name: *mut *const c_char,
-                                     plugin_desc: *mut *const c_char,
-                                     plugin_version: *mut *const c_char,
-                                     arg: *const c_char) -> c_int
-                                     where T: Plugin<'ph> + Default + 'ph {
+pub unsafe fn hexchat_plugin_init<'ph, T>(
+    ph: MutLtPtr<'ph, RawPh<'ph>>,
+    plugin_name: *mut *const c_char,
+    plugin_desc: *mut *const c_char,
+    plugin_version: *mut *const c_char,
+    arg: ConstLtPtr<'_, c_char>,
+) -> c_int
+where
+    T: Plugin<'ph> + Default + 'ph,
+{
     if ph.as_raw().is_null() || plugin_name.is_null() || plugin_desc.is_null() || plugin_version.is_null() {
         // we can't really do anything here. just hope this doesn't panic.
         eprintln!("hexchat_plugin_init called with a null pointer that shouldn't be null - broken hexchat");
@@ -2266,7 +2272,7 @@ pub unsafe fn hexchat_plugin_init<'ph, T>(ph: MutLtPtr<'ph, RawPh<'ph>>,
     }
     // we got banned from an IRC network over the following line of code.
     hexchat_print_str(ph, "hexchat-unsafe-plugin  Copyright (C) 2018, 2021, 2022 Soni L.  GPL-3.0-or-later  This software is made with love by a queer trans person.", false);
-    let mut pluginfo = if let Some(pluginfo) = PluginInfo::new(plugin_name, plugin_desc, plugin_version) {
+    let mut pluginfo = if let Some(pluginfo) = unsafe { PluginInfo::new(plugin_name, plugin_desc, plugin_version) } {
         pluginfo
     } else {
         return 0;
@@ -2287,16 +2293,18 @@ pub unsafe fn hexchat_plugin_init<'ph, T>(ph: MutLtPtr<'ph, RawPh<'ph>>,
             });
             let contexts = Rc::clone(&pluginhandle.contexts);
             let mut plug = Box::pin(T::default());
-            if plug.as_mut().init(&mut pluginhandle, &filename, if !arg.is_null() { Some(CStr::from_ptr(arg).to_str().expect("arg not valid utf-8 - broken hexchat")) } else { None }) {
-                if !(pluginfo.name.is_null() || pluginfo.desc.is_null() || pluginfo.vers.is_null()) {
+            if plug.as_mut().init(&mut pluginhandle, &filename, if !arg.is_null() { Some(unsafe { CStr::from_lt_ptr(arg) }.to_str().expect("arg not valid utf-8 - broken hexchat")) } else { None }) {
+                if unsafe { pluginfo.is_registered() } {
                     Some(Box::new(PhUserdata { plug, pluginfo, contexts, _context_hook: context_hook }))
                 } else {
                     // TODO log: forgot to call register
                     None
                 }
             } else {
-                if !(pluginfo.name.is_null() || pluginfo.desc.is_null() || pluginfo.vers.is_null()) {
-                    pluginfo.drop_info()
+                if unsafe { pluginfo.is_registered() } {
+                    unsafe {
+                        pluginfo.drop_info()
+                    }
                 }
                 None
             }
@@ -2304,12 +2312,16 @@ pub unsafe fn hexchat_plugin_init<'ph, T>(ph: MutLtPtr<'ph, RawPh<'ph>>,
     };
     match r {
         Result::Ok(Option::Some(plug @ _)) => {
-            put_userdata(ph, plug);
+            unsafe {
+                put_userdata(ph, plug);
+            }
             1
         },
         r @ _ => {
             if let Err(e) = r {
-                log_panic(ph, e);
+                unsafe {
+                    log_panic(ph, e);
+                }
             }
             0
         },
@@ -2317,22 +2329,24 @@ pub unsafe fn hexchat_plugin_init<'ph, T>(ph: MutLtPtr<'ph, RawPh<'ph>>,
 }
 
 #[doc(hidden)]
-pub unsafe extern "C" fn hexchat_plugin_deinit<'ph, T>(ph: MutLtPtr<'ph, RawPh<'ph>>) -> c_int where T: Plugin<'ph> {
+pub unsafe extern "C" fn hexchat_plugin_deinit<'ph, T: Plugin<'ph>>(
+    ph: MutLtPtr<'ph, RawPh<'ph>>,
+) -> c_int {
     let mut safe_to_unload = 1;
     // plugin_handle should never be null, but just in case.
-    if !ph.as_raw().is_null() {
+    if !unsafe { ph.as_raw() }.is_null() {
         // userdata should also never be null.
-        if !(*ph.as_raw()).userdata.is_null() {
+        if !unsafe { (*ph.as_raw()).userdata }.is_null() {
             {
                 let mut info: Option<PluginInfo> = None;
                 {
                     let ausph = AssertUnwindSafe(ph);
                     let mut ausinfo = AssertUnwindSafe(&mut info);
                     safe_to_unload = match catch_unwind(move || {
-                        let mut userdata = pop_userdata(*ausph);
+                        let mut userdata = unsafe { pop_userdata(*ausph) };
                         let pluginfo = userdata.pluginfo;
                         if let Err(e) = catch_unwind(AssertUnwindSafe(|| {
-                            userdata.plug.as_mut().deinit(&mut {
+                            userdata.plug.as_mut().deinit(&mut unsafe {
                                 PluginHandle::new(
                                     *ausph,
                                     pluginfo,
@@ -2343,7 +2357,9 @@ pub unsafe extern "C" fn hexchat_plugin_deinit<'ph, T>(ph: MutLtPtr<'ph, RawPh<'
                             // panics in deinit may be retried.
                             // however, one may need to close hexchat if that
                             // happens.
-                            put_userdata(*ausph, userdata);
+                            unsafe {
+                                put_userdata(*ausph, userdata);
+                            }
                             std::panic::resume_unwind(e);
                         }
                         **ausinfo = Some(pluginfo);
@@ -2351,18 +2367,28 @@ pub unsafe extern "C" fn hexchat_plugin_deinit<'ph, T>(ph: MutLtPtr<'ph, RawPh<'
                     }) {
                         Ok(_) => 1,
                         Err(e) => {
-                            log_panic(ph, e);
+                            unsafe {
+                                log_panic(ph, e);
+                            }
                             0
                         }
                     };
                 }
                 if let Some(mut info) = info {
-                    info.drop_info();
+                    unsafe {
+                        info.drop_info();
+                    }
                     safe_to_unload = 1;
                 }
             }
         } else {
-            hexchat_print_str(ph, "plugin userdata was null, broken hexchat-unsafe-plugin?", false);
+            unsafe {
+                hexchat_print_str(
+                    ph,
+                    "plugin userdata was null, broken hexchat-unsafe-plugin?",
+                    false
+                );
+            }
         }
     } else {
         // we are once again hoping for the best here.
@@ -2377,18 +2403,29 @@ pub unsafe extern "C" fn hexchat_plugin_deinit<'ph, T>(ph: MutLtPtr<'ph, RawPh<'
 macro_rules! hexchat_plugin {
     ($l:lifetime, $t:ty) => {
         #[no_mangle]
-        pub unsafe extern "C" fn hexchat_plugin_init<$l>(plugin_handle: $crate::MutLtPtr<$l, $crate::RawPh<$l>>,
-                                              plugin_name: *mut *const $crate::c_char,
-                                              plugin_desc: *mut *const $crate::c_char,
-                                              plugin_version: *mut *const $crate::c_char,
-                                              arg: *const $crate::c_char) -> $crate::c_int {
-            $crate::hexchat_plugin_init::<$l, $t>(plugin_handle, plugin_name, plugin_desc, plugin_version, arg)
+        pub unsafe extern "C" fn hexchat_plugin_init<$l>(
+            plugin_handle: $crate::MutLtPtr<$l, $crate::RawPh<$l>>,
+            plugin_name: *mut *const $crate::c_char,
+            plugin_desc: *mut *const $crate::c_char,
+            plugin_version: *mut *const $crate::c_char,
+            arg: *const $crate::c_char,
+        ) -> $crate::c_int {
+            $crate::hexchat_plugin_init::<$l, $t>(
+                plugin_handle,
+                plugin_name,
+                plugin_desc,
+                plugin_version,
+                arg
+            )
         }
         #[no_mangle]
-        pub unsafe extern "C" fn hexchat_plugin_deinit<$l>(plugin_handle: $crate::MutLtPtr<$l, $crate::RawPh<$l>>) -> $crate::c_int {
+        pub unsafe extern "C" fn hexchat_plugin_deinit<$l>(
+            plugin_handle: $crate::MutLtPtr<$l, $crate::RawPh<$l>>,
+        ) -> $crate::c_int {
             $crate::hexchat_plugin_deinit::<$l, $t>(plugin_handle)
         }
-        // unlike what the documentation states, there's no need to define hexchat_plugin_get_info.
-        // so we don't. it'd be impossible to make it work well with rust anyway.
+        // unlike what the documentation states, there's no need to define
+        // hexchat_plugin_get_info. so we don't. it'd be impossible to make it
+        // work well with rust anyway.
     };
 }
diff --git a/src/mock/mod.rs b/src/mock/mod.rs
index 0e4ba03..b21c451 100644
--- a/src/mock/mod.rs
+++ b/src/mock/mod.rs
@@ -55,7 +55,7 @@ use ::libc::{c_char, c_int};
 
 use ::ltptr::*;
 
-use ::selfref::{Holder, NewWith, OperateIn, opaque};
+use ::selfref::{Holder, opaque};
 
 struct Context {
 }
@@ -401,19 +401,15 @@ unsafe extern "C" fn hexchat_event_attrs_free<'ph>(ph: MutLtPtr<'ph, Ph<'ph>>,
 impl<'env> PluginEnvironment<'env> {
     /// Creates a new plugin environment.
     pub fn new() -> Holder<'static, PluginEnvironmentK> {
-        struct NewEnv;
-        impl<'k> NewWith<'k, PluginEnvironmentK> for NewEnv {
-            fn new_with<'env>(self) -> PluginEnvironment<'env> {
-                PluginEnvironment {
-                    plugins: Default::default(),
-                    hooks: Default::default(),
-                    extra_hooks: Default::default(),
-                    depth: Default::default(),
-                    contexts: Default::default(),
-                }
+        Holder::<'_, PluginEnvironmentK>::new_with(|builder| builder.build(
+            PluginEnvironment {
+                plugins: Default::default(),
+                hooks: Default::default(),
+                extra_hooks: Default::default(),
+                depth: Default::default(),
+                contexts: Default::default(),
             }
-        }
-        Holder::new_with(NewEnv)
+        ))
     }
 
     /// Loads a plugin.
@@ -426,38 +422,23 @@ impl<'env> PluginEnvironment<'env> {
         // add the plugin
         self.plugins.borrow_mut().push(plug.clone());
 
-        struct MockPluginInit<'arg, T> {
-            arg: Option<&'arg CStr>,
-            _p: PhantomData<T>
-        }
-        impl<'k, 'env: 'k, 'arg, T> OperateIn<'k, MockPluginK<'env>>
-        for MockPluginInit<'arg, T>
-        where T: PluginFactory {
-            type Out = bool;
-            fn operate_in<'ph>(
-                self,
-                ph: Pin<&'ph UnsafeCell<MockPlugin<'ph, 'env>>>,
-            ) -> bool where 'k: 'ph {
-                let ptr = ph.get();
-                let arg = self.arg;
-                unsafe {
-                    // setup deinit function
-                    (*ptr).deinit = Some(
-                        crate::hexchat_plugin_deinit::<'ph, T::Plugin<'ph>>
-                    );
-                    crate::hexchat_plugin_init::<'ph, T::Plugin<'ph>>(
-                        MutLtPtr::from_raw(ptr as *mut _),
-                        ptr::addr_of_mut!((*ptr).plugin_name),
-                        ptr::addr_of_mut!((*ptr).plugin_desc),
-                        ptr::addr_of_mut!((*ptr).plugin_vers),
-                        arg.map(|arg| arg.as_ptr()).unwrap_or(ptr::null()),
-                    ) != 0
-                }
+        if plug.as_ref().operate_in(|ph| {
+            let ptr = ph.get();
+            unsafe {
+                // setup deinit function
+                (*ptr).deinit = Some(
+                    crate::hexchat_plugin_deinit::<'_, T::Plugin<'_>>
+                );
+                crate::hexchat_plugin_init::<T::Plugin<'_>>(
+                    MutLtPtr::from_raw(ptr as *mut _),
+                    ptr::addr_of_mut!((*ptr).plugin_name),
+                    ptr::addr_of_mut!((*ptr).plugin_desc),
+                    ptr::addr_of_mut!((*ptr).plugin_vers),
+                    ConstLtPtr::from_raw(
+                        arg.map(|arg| arg.as_ptr()).unwrap_or(ptr::null())
+                    ),
+                ) != 0
             }
-        }
-        if plug.as_ref().operate_in(MockPluginInit::<T> {
-            arg,
-            _p: PhantomData,
         }) {
             Ok(())
         } else {
@@ -482,17 +463,8 @@ impl<'ph, 'env> MockPlugin<'ph, 'env> {
         env: Pin<&'env PluginEnvironment<'env>>,
         filename: CString,
     ) -> Pin<Rc<Holder<'env, MockPluginK<'env>>>> {
-        struct MockPluginNewWith<'env> {
-            filename: CString,
-            env: Pin<&'env PluginEnvironment<'env>>
-        }
-        impl<'k, 'env> NewWith<'k, MockPluginK<'env>>
-        for MockPluginNewWith<'env>
-        where 'env: 'k {
-            fn new_with<'ph>(self) -> UnsafeCell<MockPlugin<'ph, 'env>>
-            where 'k: 'ph {
-                let filename = self.filename;
-                let env = self.env;
+        let plug = Rc::pin(Holder::<'_, MockPluginK<'env>>::new_with(
+            |builder| builder.build({
                 UnsafeCell::new(MockPlugin {
                     methods: Ph {
                         hexchat_hook_command,
@@ -547,28 +519,15 @@ impl<'ph, 'env> MockPlugin<'ph, 'env> {
                     deinit: None,
                     context: None,
                 })
+            })
+        ));
+        plug.as_ref().operate_in(|ph| {
+            let ptr = ph.get();
+            unsafe {
+                // initialize plugin_name from filename(!)
+                (*ptr).plugin_name = (*ptr).filename.as_ptr();
             }
-        }
-        let plug = Rc::pin(Holder::new_with(MockPluginNewWith {
-            filename,
-            env,
-        }));
-        struct MockPluginPreInit;
-        impl<'k, 'env: 'k> OperateIn<'k, MockPluginK<'env>>
-        for MockPluginPreInit {
-            type Out = ();
-            fn operate_in<'ph>(
-                self,
-                ph: Pin<&'ph UnsafeCell<MockPlugin<'ph, 'env>>>,
-            ) where 'k: 'ph {
-                let ptr = ph.get();
-                unsafe {
-                    // initialize plugin_name from filename(!)
-                    (*ptr).plugin_name = (*ptr).filename.as_ptr();
-                }
-            }
-        }
-        plug.as_ref().operate_in(MockPluginPreInit);
+        });
         plug
     }
 }
diff --git a/src/pluginfo.rs b/src/pluginfo.rs
index 195aa26..507751a 100644
--- a/src/pluginfo.rs
+++ b/src/pluginfo.rs
@@ -1,5 +1,5 @@
 // This file is part of Hexchat Plugin API Bindings for Rust
-// Copyright (C) 2018, 2021 Soni L.
+// Copyright (C) 2018, 2021, 2022 Soni L.
 //
 // This program is free software: you can redistribute it and/or modify
 // it under the terms of the GNU General Public License as
@@ -36,13 +36,19 @@ impl PluginInfo {
     ///
     /// # Safety
     ///
-    /// This function is unsafe, as it can't guarantee the validity of its arguments. It does more
-    /// checking than `new_unchecked` however.
+    /// This function is unsafe, as it can't guarantee the validity of its
+    /// arguments. It does more checking than `new_unchecked` however.
+    ///
+    /// See `new_unchecked` for the full requirements.
     ///
     /// # Panics
     ///
     /// This function explicitly doesn't panic. Call unwrap() on the result instead.
-    pub unsafe fn new(name: *mut *const libc::c_char, desc: *mut *const libc::c_char, vers: *mut *const libc::c_char) -> Option<PluginInfo> {
+    pub unsafe fn new(
+        name: *mut *const libc::c_char,
+        desc: *mut *const libc::c_char,
+        vers: *mut *const libc::c_char,
+    ) -> Option<PluginInfo> {
         if name.is_null() || desc.is_null() || vers.is_null() || name == desc || desc == vers || name == vers {
             None
         } else {
@@ -54,9 +60,14 @@ impl PluginInfo {
     ///
     /// # Safety
     ///
-    /// This function is unsafe, as it doesn't check the validity of the arguments. You're expected
-    /// to only pass in non-aliased non-null pointers. Use new if unsure.
-    pub unsafe fn new_unchecked(name: *mut *const libc::c_char, desc: *mut *const libc::c_char, vers: *mut *const libc::c_char) -> PluginInfo {
+    /// This function is unsafe, as it doesn't check the validity of the
+    /// arguments. You're expected to only pass in non-aliased non-null valid
+    /// pointers. Use `new` if unsure.
+    pub unsafe fn new_unchecked(
+        name: *mut *const libc::c_char,
+        desc: *mut *const libc::c_char,
+        vers: *mut *const libc::c_char,
+    ) -> PluginInfo {
         PluginInfo {
             name, desc, vers
         }
@@ -84,5 +95,19 @@ impl PluginInfo {
             *self.vers = cstr(EMPTY_CSTRING_DATA);
         }
     }
+
+    /// Returns whether this `PluginInfo` has been registered.
+    ///
+    /// # Safety
+    ///
+    /// We cannot currently guarantee the validity of the pointers contained in
+    /// here.
+    pub unsafe fn is_registered(&self) -> bool {
+        ![
+            self.name,
+            self.desc,
+            self.vers,
+        ].into_iter().any(|meta| unsafe { *meta }.is_null())
+    }
 }