// Protocol Buffers - Google's data interchange format // Copyright 2023 Google LLC. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file or at // https://developers.google.com/open-source/licenses/bsd use std::fmt::Debug; use crate::__internal::Private; use crate::__runtime::InnerPrimitiveMut; use crate::vtable::{PrimitiveWithRawVTable, ProxiedWithRawVTable, RawVTableOptionalMutatorData}; use crate::{Mut, MutProxy, Proxied, ProxiedWithPresence, SettableValue, View, ViewProxy}; /// A mutator for a primitive (numeric or enum) value of `T`. /// /// This type is `protobuf::Mut<'msg, T>`. pub struct PrimitiveMut<'msg, T> { inner: InnerPrimitiveMut<'msg, T>, } impl<'msg, T> Debug for PrimitiveMut<'msg, T> where T: PrimitiveWithRawVTable, { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { f.debug_struct("PrimitiveMut").field("inner", &self.inner).finish() } } impl<'msg, T> PrimitiveMut<'msg, T> { /// # Safety /// `inner` must be valid and non-aliased for `T` for `'msg` #[doc(hidden)] pub unsafe fn from_inner(_private: Private, inner: InnerPrimitiveMut<'msg, T>) -> Self { Self { inner } } } // SAFETY: all `T` that can perform mutations don't mutate through a shared // reference. unsafe impl<'msg, T> Sync for PrimitiveMut<'msg, T> {} impl<'msg, T> PrimitiveMut<'msg, T> where T: PrimitiveWithRawVTable, { /// Gets the current value of the field. pub fn get(&self) -> View<'_, T> { T::make_view(Private, self.inner) } /// Sets a new value for the field. pub fn set(&mut self, val: impl SettableValue) { val.set_on(Private, self.as_mut()) } #[doc(hidden)] pub fn set_primitive(&mut self, _private: Private, value: T) { // SAFETY: the raw mutator is valid for `'msg` as enforced by `Mut` unsafe { self.inner.set(value) } } } impl<'msg, T> ViewProxy<'msg> for PrimitiveMut<'msg, T> where T: PrimitiveWithRawVTable, { type Proxied = T; fn as_view(&self) -> View<'_, Self::Proxied> { self.get() } fn into_view<'shorter>(self) -> View<'shorter, Self::Proxied> { self.get() } } impl<'msg, T> MutProxy<'msg> for PrimitiveMut<'msg, T> where T: PrimitiveWithRawVTable, { fn as_mut(&mut self) -> Mut<'_, Self::Proxied> { PrimitiveMut { inner: self.inner } } fn into_mut<'shorter>(self) -> Mut<'shorter, Self::Proxied> where 'msg: 'shorter, { self } } macro_rules! impl_singular_primitives { ($($t:ty),*) => { $( impl Proxied for $t { type View<'msg> = $t; type Mut<'msg> = PrimitiveMut<'msg, $t>; } impl<'msg> ViewProxy<'msg> for $t { type Proxied = $t; fn as_view(&self) -> View<'_, Self::Proxied> { *self } fn into_view<'shorter>(self) -> View<'shorter, Self::Proxied> { self } } impl SettableValue<$t> for $t { fn set_on<'msg>(self, private: Private, mut mutator: Mut<'msg, $t>) where $t: 'msg { mutator.set_primitive(private, self) } fn set_on_absent( self, _private: Private, absent_mutator: <$t as ProxiedWithPresence>::PresentMutData<'_>, ) -> <$t as ProxiedWithPresence>::AbsentMutData<'_> { absent_mutator.set(Private, self) } } impl ProxiedWithPresence for $t { type PresentMutData<'msg> = RawVTableOptionalMutatorData<'msg, $t>; type AbsentMutData<'msg> = RawVTableOptionalMutatorData<'msg, $t>; fn clear_present_field( present_mutator: Self::PresentMutData<'_>, ) -> Self::AbsentMutData<'_> { present_mutator.clear(Private) } fn set_absent_to_default( absent_mutator: Self::AbsentMutData<'_>, ) -> Self::PresentMutData<'_> { absent_mutator.set_absent_to_default(Private) } } impl PrimitiveWithRawVTable for $t {} // ProxiedInRepeated is implemented in {cpp,upb}.rs )* } } impl_singular_primitives!(bool, f32, f64, i32, i64, u32, u64);