use std::fmt::{self, Debug, Display, Write as _};
use std::marker::PhantomData;
use std::ptr::NonNull;
use std::slice;
use std::str;
pub const MAX_NAME_LENGTH: usize = 65_536;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub struct CStrError;
#[derive(Copy, Clone)]
pub(crate) struct CStr<'a> {
ptr: NonNull<u8>,
marker: PhantomData<&'a [u8]>,
}
unsafe impl Send for CStr<'static> {}
unsafe impl Sync for CStr<'static> {}
impl<'a> CStr<'a> {
#[cfg(test)]
pub fn from_bytes_with_nul(bytes: &'static [u8]) -> Self {
assert_eq!(bytes.last(), Some(&b'\0'));
let ptr = NonNull::from(bytes).cast();
unsafe { Self::from_ptr(ptr) }
}
pub unsafe fn from_ptr(ptr: NonNull<i8>) -> Self {
CStr {
ptr: ptr.cast(),
marker: PhantomData,
}
}
pub fn len(self) -> Result<usize, CStrError> {
let start = self.ptr.as_ptr();
let mut end = start;
let mut len = 0usize;
unsafe {
while *end != 0 {
if len >= MAX_NAME_LENGTH {
return Err(CStrError);
}
end = end.add(1);
len += 1;
}
Ok(len)
}
}
pub fn to_bytes(self) -> Result<&'a [u8], CStrError> {
let len = self.len()?;
unsafe { Ok(slice::from_raw_parts(self.ptr.as_ptr(), len)) }
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::ptr::NonNull;
use std::thread;
#[test]
fn send_sync_static() {
static BYTES: &[u8] = b"static\0";
let cstr = CStr::from_bytes_with_nul(BYTES);
thread::spawn(move || {
assert_eq!(cstr.to_bytes().unwrap(), b"static");
})
.join()
.unwrap();
}
#[test]
fn len_ok() {
static BYTES: &[u8] = b"abc\0";
let cstr = CStr::from_bytes_with_nul(BYTES);
assert_eq!(cstr.len().unwrap(), 3);
}
#[test]
fn len_too_long() {
let mut bytes = vec![b'a'; MAX_NAME_LENGTH + 1];
bytes.push(0);
let ptr = NonNull::new(bytes.as_mut_ptr() as *mut i8).unwrap();
let cstr = unsafe { CStr::from_ptr(ptr) };
assert!(cstr.len().is_err());
}
}
impl Display for CStr<'_> {
fn fmt(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
let ptr = self.ptr.as_ptr();
let len = self.len().map_err(|_| fmt::Error)?;
let bytes = unsafe { slice::from_raw_parts(ptr, len) };
display_lossy(bytes, formatter)
}
}
impl Debug for CStr<'_> {
fn fmt(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
let ptr = self.ptr.as_ptr();
let len = self.len().map_err(|_| fmt::Error)?;
let bytes = unsafe { slice::from_raw_parts(ptr, len) };
debug_lossy(bytes, formatter)
}
}
fn display_lossy(mut bytes: &[u8], formatter: &mut fmt::Formatter) -> fmt::Result {
loop {
match str::from_utf8(bytes) {
Ok(valid) => return formatter.write_str(valid),
Err(utf8_error) => {
let valid_up_to = utf8_error.valid_up_to();
match str::from_utf8(&bytes[..valid_up_to]) {
Ok(valid) => formatter.write_str(valid)?,
Err(_) => return Err(fmt::Error),
}
formatter.write_char(char::REPLACEMENT_CHARACTER)?;
if let Some(error_len) = utf8_error.error_len() {
bytes = &bytes[valid_up_to + error_len..];
} else {
return Ok(());
}
}
}
}
}
pub(crate) fn debug_lossy(mut bytes: &[u8], formatter: &mut fmt::Formatter) -> fmt::Result {
const EMPTY: &str = "";
formatter.write_char('"')?;
while !bytes.is_empty() {
let from_utf8_result = str::from_utf8(bytes);
let valid = match from_utf8_result {
Ok(valid) => valid,
Err(utf8_error) => {
let valid_up_to = utf8_error.valid_up_to();
str::from_utf8(&bytes[..valid_up_to]).unwrap_or(EMPTY)
}
};
let mut written = 0;
for (i, ch) in valid.char_indices() {
let esc = ch.escape_debug();
if esc.len() != 1 && ch != '\'' {
formatter.write_str(&valid[written..i])?;
for ch in esc {
formatter.write_char(ch)?;
}
written = i + ch.len_utf8();
}
}
formatter.write_str(&valid[written..])?;
match from_utf8_result {
Ok(_valid) => break,
Err(utf8_error) => {
let end_of_broken = if let Some(error_len) = utf8_error.error_len() {
valid.len() + error_len
} else {
bytes.len()
};
for b in &bytes[valid.len()..end_of_broken] {
write!(formatter, "\\x{:02x}", b)?;
}
bytes = &bytes[end_of_broken..];
}
}
}
formatter.write_char('"')
}