blob: a1818f3315752d60964ffeed9dafcce6e5d434ef [file] [log] [blame] [edit]
use super::*;
use crate::Enum;
pub trait UpbTypeConversions<Tag>: Proxied {
fn upb_type() -> upb::CType;
fn to_message_value(val: View<'_, Self>) -> upb_MessageValue;
/// # Safety
/// - `raw_arena` must point to a valid upb arena.
unsafe fn into_message_value_fuse_if_required(
raw_arena: RawArena,
val: Self,
) -> upb_MessageValue;
/// # Safety
/// - `msg_val` must be the correct variant for `Self`.
/// - `msg_val` pointers must point to memory valid for `'msg` lifetime.
/// - If `Self` is a closed enum, then `msg_val.int32_val` must be a valid enum entry.
unsafe fn from_message_value<'msg>(msg_val: upb_MessageValue) -> View<'msg, Self>;
/// # Safety
/// - `raw` must be the correct variant for `Self`.
/// - `raw` pointers must point to memory valid for `'msg` lifetime.
#[allow(unused_variables)]
unsafe fn from_message_mut<'msg>(raw: RawMessage, arena: &'msg Arena) -> Mut<'msg, Self>
where
Self: Message,
{
panic!("mut_from_message_value is only implemented for messages.")
}
/// # Safety
/// - `src` must be a valid array of `Self`.
/// - `dest` must be a valid mutable array of `Self`.
/// - `arena` must point to an arena that will outlive `dest`.
unsafe fn copy_repeated(src: RawArray, dest: RawArray, arena: RawArena);
}
impl<T> UpbTypeConversions<MessageTag> for T
where
Self: Message,
for<'a> View<'a, Self>: MessageViewInterop<'a>,
for<'a> Mut<'a, Self>: From<MessageMutInner<'a, Self>>,
{
fn upb_type() -> CType {
CType::Message
}
fn to_message_value(val: View<'_, Self>) -> upb_MessageValue {
upb_MessageValue { msg_val: Some(val.get_ptr(Private).raw()) }
}
unsafe fn into_message_value_fuse_if_required(
raw_parent_arena: RawArena,
mut val: Self,
) -> upb_MessageValue {
// SAFETY: The arena memory is not freed due to `ManuallyDrop`.
let parent_arena =
std::mem::ManuallyDrop::new(unsafe { Arena::from_raw(raw_parent_arena) });
parent_arena.fuse(val.get_arena(Private));
upb_MessageValue { msg_val: Some(val.get_ptr(Private).raw()) }
}
unsafe fn from_message_value<'msg>(msg: upb_MessageValue) -> View<'msg, Self> {
unsafe {
let raw = msg.msg_val.expect("expected present message value in map");
View::<Self>::__unstable_wrap_raw_message_unchecked_lifetime(
raw.as_ptr() as *const std::ffi::c_void
)
}
}
unsafe fn from_message_mut<'msg>(msg: RawMessage, arena: &'msg Arena) -> Mut<'msg, Self> {
unsafe { MessageMutInner::<'msg, Self>::wrap_raw(msg, arena).into() }
}
unsafe fn copy_repeated(src: RawArray, dest: RawArray, arena: RawArena) {
// SAFETY:
// - `src` is a valid `const upb_Array*`.
// - `dest` is a valid `upb_Array*`.
// - Elements of `src` and `dest` have minitable `Self::mini_table()`.
unsafe {
let size = upb_Array_Size(src);
if !upb_Array_Resize(dest, size, arena) {
panic!("upb_Array_Resize failed (alloc should be infallible)");
}
for i in 0..size {
let src_msg =
upb_Array_Get(src, i).msg_val.expect("upb_Array* element should not be NULL");
// Avoid the use of `upb_Array_DeepClone` as it creates an
// entirely new `upb_Array*` at a new memory address.
let cloned_msg = upb_Message_DeepClone(src_msg, Self::mini_table(), arena)
.expect("upb_Message_DeepClone failed (alloc should be infallible)");
upb_Array_Set(dest, i, upb_MessageValue { msg_val: Some(cloned_msg) });
}
}
}
}
impl<T> UpbTypeConversions<EnumTag> for T
where
Self: Enum,
{
fn upb_type() -> CType {
CType::Enum
}
fn to_message_value(val: View<'_, Self>) -> upb_MessageValue {
upb_MessageValue { int32_val: val.into() }
}
unsafe fn into_message_value_fuse_if_required(
_raw_parent_arena: RawArena,
val: Self,
) -> upb_MessageValue {
upb_MessageValue { int32_val: val.into() }
}
unsafe fn from_message_value<'msg>(val: upb_MessageValue) -> View<'msg, Self> {
// SAFETY: The caller guarantees that `val` is the correct variant.
let result = Self::try_from(unsafe { val.int32_val });
std::debug_assert!(result.is_ok());
// SAFETY:
// - The caller guarantees that `val.int32_val` is valid for this enum.
unsafe { result.unwrap_unchecked() }
}
unsafe fn copy_repeated(src: RawArray, dest: RawArray, arena: RawArena) {
// SAFETY:
// - Enum arrays have the same representation as i32 arrays.
// - The caller guarantees that src and dest are enum arrays and that `arena` will outlive
// `dest`.
unsafe {
<i32 as UpbTypeConversions<PrimitiveTag>>::copy_repeated(src, dest, arena);
}
}
}
macro_rules! impl_upb_type_conversions_for_scalars {
($($t:ty, $ufield:ident, $upb_tag:expr, $zero_val:literal;)*) => {
$(
impl UpbTypeConversions<PrimitiveTag> for $t {
#[inline(always)]
fn upb_type() -> upb::CType {
$upb_tag
}
#[inline(always)]
fn to_message_value(val: View<'_, $t>) -> upb_MessageValue {
upb_MessageValue { $ufield: val }
}
#[inline(always)]
unsafe fn into_message_value_fuse_if_required(_: RawArena, val: $t) -> upb_MessageValue {
<Self as UpbTypeConversions<PrimitiveTag>>::to_message_value(val)
}
#[inline(always)]
unsafe fn from_message_value<'msg>(msg: upb_MessageValue) -> View<'msg, $t> {
unsafe { msg.$ufield }
}
#[inline(always)]
unsafe fn copy_repeated(src: RawArray, dest: RawArray, arena: RawArena) {
// SAFETY:
// - `upb_Array_Resize` is unsafe but assumed to be always sound to call.
// - `copy_nonoverlapping` is unsafe but here we guarantee that both pointers
// are valid, the pointers are `#[repr(u8)]`, and the size is correct.
unsafe {
let len = upb_Array_Size(src);
if (!upb_Array_Resize(dest, len, arena)) {
panic!("upb_Array_Resize failed (alloc should be infallible)");
}
ptr::copy_nonoverlapping(
upb_Array_DataPtr(src).cast::<u8>(),
upb_Array_MutableDataPtr(dest).cast::<u8>(),
size_of::<$t>() * len);
}
}
}
)*
};
}
impl_upb_type_conversions_for_scalars!(
f32, float_val, upb::CType::Float, 0f32;
f64, double_val, upb::CType::Double, 0f64;
i32, int32_val, upb::CType::Int32, 0i32;
u32, uint32_val, upb::CType::UInt32, 0u32;
i64, int64_val, upb::CType::Int64, 0i64;
u64, uint64_val, upb::CType::UInt64, 0u64;
bool, bool_val, upb::CType::Bool, false;
);
impl UpbTypeConversions<PrimitiveTag> for ProtoBytes {
fn upb_type() -> upb::CType {
upb::CType::Bytes
}
fn to_message_value(val: View<'_, ProtoBytes>) -> upb_MessageValue {
upb_MessageValue { str_val: val.into() }
}
unsafe fn into_message_value_fuse_if_required(
raw_parent_arena: RawArena,
val: ProtoBytes,
) -> upb_MessageValue {
// SAFETY: The arena memory is not freed due to `ManuallyDrop`.
let parent_arena = ManuallyDrop::new(unsafe { Arena::from_raw(raw_parent_arena) });
let (view, arena) = val.inner.into_raw_parts();
parent_arena.fuse(&arena);
upb_MessageValue { str_val: view }
}
unsafe fn from_message_value<'msg>(msg: upb_MessageValue) -> View<'msg, ProtoBytes> {
let _ = unsafe { msg.str_val.as_ref() };
// We actually need view of ProtoBytes, the original code had:
// unsafe { msg.str_val.as_ref() }
// Wait, ProtoBytes view is `&[u8]`. What was the actual code?
// Ah, it actually was: `unsafe { msg.str_val.as_ref() }`
// But the return type is `View<'msg, ProtoBytes>`. `View<ProtoBytes>` is `&[u8]`.
// Let's just return what `msg.str_val.as_ref()` returns, which is `&[u8]`.
unsafe { msg.str_val.as_ref() }
}
unsafe fn copy_repeated(src: RawArray, dest: RawArray, arena: RawArena) {
unsafe {
copy_repeated_bytes(src, dest, arena);
}
}
}
impl UpbTypeConversions<PrimitiveTag> for ProtoString {
fn upb_type() -> upb::CType {
upb::CType::String
}
fn to_message_value(val: View<'_, ProtoString>) -> upb_MessageValue {
upb_MessageValue { str_val: val.as_bytes().into() }
}
unsafe fn into_message_value_fuse_if_required(
raw_arena: RawArena,
val: ProtoString,
) -> upb_MessageValue {
// SAFETY: `raw_arena` is valid as promised by the caller
unsafe {
<ProtoBytes as UpbTypeConversions<PrimitiveTag>>::into_message_value_fuse_if_required(
raw_arena,
val.into(),
)
}
}
unsafe fn from_message_value<'msg>(msg: upb_MessageValue) -> View<'msg, ProtoString> {
ProtoStr::from_utf8_unchecked(unsafe { msg.str_val.as_ref() })
}
unsafe fn copy_repeated(src: RawArray, dest: RawArray, arena: RawArena) {
unsafe {
copy_repeated_bytes(src, dest, arena);
}
}
}
/// # Safety
/// - `src` must be a valid array of string or bytes.
/// - `dest` must be a valid mutable array of the same type as `src`.
/// - `arena` must point to an arena that will outlive `dest`.
unsafe fn copy_repeated_bytes(src: RawArray, dest: RawArray, arena: RawArena) {
// SAFETY:
// - `upb_Array_Resize` is unsafe but assumed to be always sound to call.
// - `upb_Array` ensures its elements are never uninitialized memory.
// - The `DataPtr` and `MutableDataPtr` functions return pointers to spans
// of memory that are valid for at least `len` elements of PtrAndLen.
// - `copy_nonoverlapping` is unsafe but here we guarantee that both pointers
// are valid, the pointers are `#[repr(u8)]`, and the size is correct.
// - The bytes held within a valid array are valid.
unsafe {
let len = upb_Array_Size(src);
let arena = ManuallyDrop::new(Arena::from_raw(arena));
if !upb_Array_Resize(dest, len, arena.raw()) {
panic!("upb_Array_Resize failed (alloc should be infallible)");
}
let src_ptrs: &[PtrAndLen] = slice::from_raw_parts(upb_Array_DataPtr(src).cast(), len);
let dest_ptrs: &mut [PtrAndLen] =
slice::from_raw_parts_mut(upb_Array_MutableDataPtr(dest).cast(), len);
for (src_ptr, dest_ptr) in src_ptrs.iter().zip(dest_ptrs) {
*dest_ptr = arena.copy_slice_in(src_ptr.as_ref()).unwrap().into();
}
}
}