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
49
50
51
52
53
54
55
56
57
58
59
use std::{marker::PhantomData, str::Utf8Error};

use crate::raw::TableGenStringRef;

#[derive(Debug, Clone, Copy)]
pub struct StringRef<'a> {
    raw: TableGenStringRef,
    _reference: PhantomData<&'a TableGenStringRef>,
}

impl<'a> StringRef<'a> {
    pub unsafe fn to_raw(self) -> TableGenStringRef {
        self.raw
    }

    pub unsafe fn from_raw(raw: TableGenStringRef) -> Self {
        Self {
            raw,
            _reference: PhantomData,
        }
    }

    pub unsafe fn from_option_raw(raw: TableGenStringRef) -> Option<Self> {
        if !raw.data.is_null() {
            Some(Self::from_raw(raw))
        } else {
            None
        }
    }

    pub fn as_str(&self) -> Result<&str, Utf8Error> {
        <&str as TryFrom<Self>>::try_from(*self)
    }
}

impl<'a> From<&'a str> for StringRef<'a> {
    fn from(value: &'a str) -> Self {
        unsafe {
            StringRef::from_raw(TableGenStringRef {
                data: value.as_ptr() as *const i8,
                len: value.len(),
            })
        }
    }
}

impl<'a> TryFrom<StringRef<'a>> for &'a str {
    type Error = Utf8Error;

    fn try_from(value: StringRef<'a>) -> Result<Self, Self::Error> {
        std::str::from_utf8(value.into())
    }
}

impl<'a> From<StringRef<'a>> for &'a [u8] {
    fn from(value: StringRef<'a>) -> Self {
        unsafe { std::slice::from_raw_parts(value.raw.data as *const u8, value.raw.len) }
    }
}