Skip to content

Commit

Permalink
add method to ValueView
Browse files Browse the repository at this point in the history
  • Loading branch information
lucacasonato committed Oct 28, 2024
1 parent 5b2734d commit 1eba610
Showing 1 changed file with 75 additions and 2 deletions.
77 changes: 75 additions & 2 deletions src/string.rs
Original file line number Diff line number Diff line change
Expand Up @@ -778,7 +778,8 @@ impl String {
}
}

/// Converts a [`crate::String`] to either an owned [`std::string::String`], or a borrowed [`str`], depending on whether it fits into the
/// Converts a [`crate::String`] to either an owned [`std::string::String`],
/// or a borrowed [`str`], depending on whether it fits into the
/// provided buffer.
pub fn to_rust_cow_lossy<'a, const N: usize>(
&self,
Expand All @@ -789,7 +790,9 @@ impl String {
let string = unsafe { Local::from_raw(self).unwrap_unchecked() };
let view = ValueView::new(scope, string);
match view.data() {
ValueViewData::OneByte(bytes) => latin1_to_cow_str(bytes, buffer),
ValueViewData::OneByte(bytes) => {
latin1_to_cow_str_always_copy(bytes, buffer)
}
ValueViewData::TwoByte(code_points) => {
wtf16_to_cow_str(code_points, buffer)
}
Expand Down Expand Up @@ -861,6 +864,44 @@ fn wtf16_to_string(code_points: &[u16]) -> std::string::String {

#[inline(always)]
fn latin1_to_cow_str<'a, const N: usize>(
bytes: &'a [u8],
buffer: &'a mut [MaybeUninit<u8>; N],
) -> Cow<'a, str> {
if bytes.is_ascii() {
// SAFETY: The string is ASCII, so it's valid UTF-8.
Cow::Borrowed(unsafe { std::str::from_utf8_unchecked(bytes) })
} else if bytes.len() * 2 < N {
// SAFETY: The string is Latin1 - we need to convert to UTF-8. But it
// is short enough to fit into the buffer, because the buffer is at
// least twice as large as the string and any non-ASCII one-byte
// character will be encoded as exactly two bytes in UTF-8.
let written = unsafe {
latin1_to_utf8(
bytes.len(),
bytes.as_ptr(),
buffer.as_mut_ptr() as *mut u8,
)
};
debug_assert!(written <= buffer.len());

// SAFETY: The buffer is filled with valid UTF-8 data.
let str = unsafe {
std::str::from_utf8_unchecked(std::slice::from_raw_parts(
buffer.as_ptr() as *const u8,
written,
))
};
Cow::Borrowed(str)
} else {
// TODO: this could likely be optimized for large strings by using SIMD to
// calculate the length of the resulting string and then allocating once,
// and then converting the string using SIMD.
Cow::Owned(std::string::String::from_utf8_lossy(bytes).into_owned())
}
}

#[inline(always)]
fn latin1_to_cow_str_always_copy<'a, const N: usize>(
bytes: &[u8],
buffer: &'a mut [MaybeUninit<u8>; N],
) -> Cow<'a, str> {
Expand Down Expand Up @@ -1145,3 +1186,35 @@ impl<'s> Drop for ValueView<'s> {
unsafe { v8__String__ValueView__DESTRUCT(self) }
}
}

impl ValueView<'_> {
/// Creates a copy of a [`ValueView`] in a [`std::string::String`].
/// Convenience function not present in the original V8 API.
pub fn to_rust_string_lossy(&self) -> std::string::String {
match self.data() {
ValueViewData::OneByte(bytes) => latin1_to_string(bytes),
ValueViewData::TwoByte(code_points) => wtf16_to_string(code_points),
}
}

/// Converts a [`ValueView`] to either an owned [`std::string::String`],
/// or a borrowed [`str`].
///
/// If the [`ValueView`] is an ASCII one-byte string, a reference to the
/// string is returned and no copies are performed. If the string is not
/// ASCII, but fits into the provided buffer, it is copied into the buffer
/// and a reference to the buffer is returned. If the string does not fit
/// into the buffer, it is copied into a newly allocated
/// [`std::string::String`] and returned.
pub fn to_rust_cow_lossy<'a, const N: usize>(
&'a self,
buffer: &'a mut [MaybeUninit<u8>; N],
) -> Cow<'a, str> {
match self.data() {
ValueViewData::OneByte(bytes) => latin1_to_cow_str(bytes, buffer),
ValueViewData::TwoByte(code_points) => {
wtf16_to_cow_str(code_points, buffer)
}
}
}
}

0 comments on commit 1eba610

Please sign in to comment.