Add RepeatedMut::clear, free for owned Repeated
PiperOrigin-RevId: 591046913
diff --git a/rust/cpp.rs b/rust/cpp.rs
index 80f6e8e..2f43225 100644
--- a/rust/cpp.rs
+++ b/rust/cpp.rs
@@ -215,20 +215,23 @@
macro_rules! impl_repeated_primitives {
(@impl $($t:ty => [
$new_thunk:ident,
+ $free_thunk:ident,
$add_thunk:ident,
$size_thunk:ident,
$get_thunk:ident,
$set_thunk:ident,
+ $clear_thunk:ident,
$copy_from_thunk:ident $(,)?
]),* $(,)?) => {
$(
- // TODO: Add clear, free
extern "C" {
fn $new_thunk() -> RawRepeatedField;
+ fn $free_thunk(f: RawRepeatedField);
fn $add_thunk(f: RawRepeatedField, v: $t);
fn $size_thunk(f: RawRepeatedField) -> usize;
fn $get_thunk(f: RawRepeatedField, i: usize) -> $t;
fn $set_thunk(f: RawRepeatedField, i: usize, v: $t);
+ fn $clear_thunk(f: RawRepeatedField);
fn $copy_from_thunk(src: RawRepeatedField, dst: RawRepeatedField);
}
@@ -239,12 +242,19 @@
Repeated::from_inner(InnerRepeatedMut::new(Private, $new_thunk()))
}
}
+ #[allow(dead_code)]
+ unsafe fn repeated_free(_: Private, f: &mut Repeated<$t>) {
+ unsafe { $free_thunk(f.as_mut().as_raw(Private)) }
+ }
fn repeated_len(f: View<Repeated<$t>>) -> usize {
unsafe { $size_thunk(f.as_raw(Private)) }
}
fn repeated_push(mut f: Mut<Repeated<$t>>, v: View<$t>) {
unsafe { $add_thunk(f.as_raw(Private), v) }
}
+ fn repeated_clear(mut f: Mut<Repeated<$t>>) {
+ unsafe { $clear_thunk(f.as_raw(Private)) }
+ }
unsafe fn repeated_get_unchecked(f: View<Repeated<$t>>, i: usize) -> View<$t> {
unsafe { $get_thunk(f.as_raw(Private), i) }
}
@@ -262,10 +272,12 @@
impl_repeated_primitives!(@impl $(
$t => [
[< __pb_rust_RepeatedField_ $t _new >],
+ [< __pb_rust_RepeatedField_ $t _free >],
[< __pb_rust_RepeatedField_ $t _add >],
[< __pb_rust_RepeatedField_ $t _size >],
[< __pb_rust_RepeatedField_ $t _get >],
[< __pb_rust_RepeatedField_ $t _set >],
+ [< __pb_rust_RepeatedField_ $t _clear >],
[< __pb_rust_RepeatedField_ $t _copy_from >],
],
)*);
diff --git a/rust/cpp_kernel/cpp_api.cc b/rust/cpp_kernel/cpp_api.cc
index 05aec1f..0536b93 100644
--- a/rust/cpp_kernel/cpp_api.cc
+++ b/rust/cpp_kernel/cpp_api.cc
@@ -12,6 +12,10 @@
google::protobuf::RepeatedField<ty>* __pb_rust_RepeatedField_##rust_ty##_new() { \
return new google::protobuf::RepeatedField<ty>(); \
} \
+ void __pb_rust_RepeatedField_##rust_ty##_free( \
+ google::protobuf::RepeatedField<ty>* r) { \
+ delete r; \
+ } \
void __pb_rust_RepeatedField_##rust_ty##_add(google::protobuf::RepeatedField<ty>* r, \
ty val) { \
r->Add(val); \
@@ -31,6 +35,10 @@
void __pb_rust_RepeatedField_##rust_ty##_copy_from( \
google::protobuf::RepeatedField<ty> const& src, google::protobuf::RepeatedField<ty>& dst) { \
dst.CopyFrom(src); \
+ } \
+ void __pb_rust_RepeatedField_##rust_ty##_clear( \
+ google::protobuf::RepeatedField<ty>* r) { \
+ r->Clear(); \
}
expose_repeated_field_methods(int32_t, i32);
diff --git a/rust/repeated.rs b/rust/repeated.rs
index a5b4522..22d6fb9 100644
--- a/rust/repeated.rs
+++ b/rust/repeated.rs
@@ -110,8 +110,6 @@
where
T: ProxiedInRepeated + ?Sized + 'msg,
{
- // TODO: Add clear, free
-
/// # Safety
/// - `inner` must be valid to read and write from for `'msg`
/// - There must be no aliasing references or mutations on the same
@@ -158,6 +156,11 @@
pub fn copy_from(&mut self, src: RepeatedView<'_, T>) {
T::repeated_copy_from(src, self.as_mut())
}
+
+ /// Clears the repeated field.
+ pub fn clear(&mut self) {
+ T::repeated_clear(self.as_mut())
+ }
}
/// Types that can appear in a `Repeated<T>`.
@@ -171,19 +174,30 @@
/// - It must be sound to call `*_unchecked*(x)` with an `index` less than
/// `repeated_len(x)`.
pub unsafe trait ProxiedInRepeated: Proxied {
- // TODO: Add clear, free
/// Constructs a new owned `Repeated` field.
#[doc(hidden)]
fn repeated_new(_private: Private) -> Repeated<Self> {
unimplemented!("not required")
}
+ /// Frees the repeated field in-place, for use in `Drop`.
+ ///
+ /// # Safety
+ /// - After `repeated_free`, no other methods on the input are safe to call.
+ #[doc(hidden)]
+ unsafe fn repeated_free(_private: Private, _repeated: &mut Repeated<Self>) {
+ unimplemented!("not required")
+ }
+
/// Gets the length of the repeated field.
fn repeated_len(repeated: View<Repeated<Self>>) -> usize;
/// Appends a new element to the end of the repeated field.
fn repeated_push(repeated: Mut<Repeated<Self>>, val: View<Self>);
+ /// Clears the repeated field of elements.
+ fn repeated_clear(repeated: Mut<Repeated<Self>>);
+
/// # Safety
/// `index` must be less than `Self::repeated_len(repeated)`
unsafe fn repeated_get_unchecked(repeated: View<Repeated<Self>>, index: usize) -> View<Self>;
@@ -235,8 +249,8 @@
_phantom: PhantomData<T>,
}
-#[allow(dead_code)]
impl<T: ?Sized + ProxiedInRepeated> Repeated<T> {
+ #[allow(dead_code)]
pub(crate) fn new() -> Self {
T::repeated_new(Private)
}
@@ -245,6 +259,7 @@
Self { inner, _phantom: PhantomData }
}
+ #[allow(dead_code)]
pub(crate) fn inner(&mut self) -> InnerRepeatedMut<'static> {
self.inner
}
@@ -254,6 +269,13 @@
}
}
+impl<T: ?Sized + ProxiedInRepeated> Drop for Repeated<T> {
+ fn drop(&mut self) {
+ // SAFETY: only called once
+ unsafe { T::repeated_free(Private, self) }
+ }
+}
+
// SAFETY: `Repeated` does not allow for shared mutability.
unsafe impl<T: ProxiedInRepeated> Sync for Repeated<T> {}
@@ -388,6 +410,11 @@
r.iter().collect::<Vec<$t>>(), elements_are![$(eq($vals)),*]);
r.set(0, <$t as Default>::default());
assert_that!(r.get(0).expect("elem 0"), eq(<$t as Default>::default()));
+
+ r.clear();
+ assert!(r.is_empty(), "is_empty after clear");
+ assert!(r.iter().next().is_none(), "iter empty after clear");
+ assert!(r.into_iter().next().is_none(), "mut iter empty after clear");
})*
}
}
diff --git a/rust/upb.rs b/rust/upb.rs
index 406572a..cef685c 100644
--- a/rust/upb.rs
+++ b/rust/upb.rs
@@ -370,7 +370,6 @@
macro_rules! impl_repeated_primitives {
($(($t:ty, $ufield:ident, $upb_tag:expr)),* $(,)?) => {
$(
- // TODO: Add clear, free
unsafe impl ProxiedInRepeated for $t {
#[allow(dead_code)]
fn repeated_new(_: Private) -> Repeated<$t> {
@@ -385,6 +384,16 @@
})
}
}
+ #[allow(dead_code)]
+ unsafe fn repeated_free(_: Private, f: &mut Repeated<$t>) {
+ // Freeing the array itself is handled by `Arena::Drop`
+ // SAFETY:
+ // - `f.raw_arena()` is a live `upb_Arena*` as
+ // - This function is only called once for `f`
+ unsafe {
+ upb_Arena_Free(f.inner().arena);
+ }
+ }
fn repeated_len(f: View<Repeated<$t>>) -> usize {
unsafe { upb_Array_Size(f.as_raw(Private)) }
}
@@ -396,6 +405,9 @@
f.raw_arena(Private))
}
}
+ fn repeated_clear(mut f: Mut<Repeated<$t>>) {
+ unsafe { upb_Array_Resize(f.as_raw(Private), 0, f.raw_arena(Private)); }
+ }
unsafe fn repeated_get_unchecked(f: View<Repeated<$t>>, i: usize) -> View<$t> {
unsafe { upb_Array_Get(f.as_raw(Private), i).$ufield }
}