run_parser_test_suite/cstr/
mod.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
use std::fmt::{self, Display, Write as _};
use std::slice;
use std::str;
use core::ffi::c_char;

pub(crate) struct CStr {
    pub(crate) ptr: *const u8,
}

impl CStr {
    pub(crate) unsafe fn from_ptr(ptr: *const c_char) -> Self {
        CStr { ptr: ptr.cast() }
    }
}

impl Display for CStr {
    fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
        let len = unsafe { strlen(self.ptr) };
        let mut bytes = unsafe { slice::from_raw_parts(self.ptr, len) };
        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();
                    let valid = unsafe {
                        str::from_utf8_unchecked(&bytes[..valid_up_to])
                    };
                    formatter.write_str(valid)?;
                    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(());
                    }
                }
            }
        }
    }
}

unsafe fn strlen(s: *const u8) -> usize {
    let mut end = s;
    while *end != 0 {
        end = end.add(1);
    }
    end.offset_from(s) as usize
}