From 468dc02f330c16c7c2ca4deb8ea98e8d5985f53b Mon Sep 17 00:00:00 2001 From: Tony Duan Date: Sat, 5 Nov 2022 19:44:48 +0800 Subject: [PATCH] prevent nvim_buf_get_name frees buf->b_ffname nvim_buf_get_name C API doesn't copy memory but return a String with pointer to buf->b_ffname rust shouldn't free buf->b_ffname --- crates/nvim-api/src/ffi/buffer.rs | 18 ++++-------------- crates/nvim-types/src/lib.rs | 2 ++ crates/nvim-types/src/str.rs | 28 ++++++++++++++++++++++++++++ 3 files changed, 34 insertions(+), 14 deletions(-) create mode 100644 crates/nvim-types/src/str.rs diff --git a/crates/nvim-api/src/ffi/buffer.rs b/crates/nvim-api/src/ffi/buffer.rs index 46e91b63..a071803a 100644 --- a/crates/nvim-api/src/ffi/buffer.rs +++ b/crates/nvim-api/src/ffi/buffer.rs @@ -1,19 +1,10 @@ use nvim_types::{ - Array, - BufHandle, - Dictionary, - Error, - Integer, - LuaRef, - NonOwning, - Object, - String, + Array, BufHandle, Dictionary, Error, Integer, LuaRef, NonOwning, Object, + Str, String, }; use crate::opts::{ - KeyDict_get_commands, - KeyDict_keymap, - KeyDict_user_command, + KeyDict_get_commands, KeyDict_keymap, KeyDict_user_command, }; extern "C" { @@ -118,8 +109,7 @@ extern "C" { ) -> Array; // https://github.com/neovim/neovim/blob/master/src/nvim/api/buffer.c#L1086 - pub(crate) fn nvim_buf_get_name(buf: BufHandle, err: *mut Error) - -> String; + pub(crate) fn nvim_buf_get_name(buf: BufHandle, err: *mut Error) -> Str; // https://github.com/neovim/neovim/blob/master/src/nvim/api/buffer.c#L876 pub(crate) fn nvim_buf_get_offset( diff --git a/crates/nvim-types/src/lib.rs b/crates/nvim-types/src/lib.rs index dac129bc..625454a8 100644 --- a/crates/nvim-types/src/lib.rs +++ b/crates/nvim-types/src/lib.rs @@ -11,8 +11,10 @@ mod non_owning; mod object; #[cfg(feature = "serde")] pub mod serde; +mod str; mod string; +pub use crate::str::Str; pub use array::{Array, ArrayIterator}; pub use dictionary::{DictIterator, Dictionary, KeyValuePair}; pub use error::Error; diff --git a/crates/nvim-types/src/str.rs b/crates/nvim-types/src/str.rs new file mode 100644 index 00000000..b7c2ffb3 --- /dev/null +++ b/crates/nvim-types/src/str.rs @@ -0,0 +1,28 @@ +use std::{ + ffi::{c_char, OsStr}, + os::unix::prelude::OsStrExt, + path::PathBuf, + slice, +}; + +#[repr(C)] +pub struct Str { + pub(crate) data: *mut c_char, + pub(crate) size: usize, +} + +impl Str { + fn as_bytes(&self) -> &[u8] { + if self.data.is_null() { + &[] + } else { + unsafe { slice::from_raw_parts(self.data as *const u8, self.size) } + } + } +} + +impl From for PathBuf { + fn from(s: Str) -> Self { + OsStr::from_bytes(s.as_bytes()).to_owned().into() + } +}