This CL migrates messages, enums, and primitive types all onto the same blanket implementation of the `ProxiedInMapValue` trait. This gets us to the point where messages and enums no longer need to generate any significant amount of extra code just in case they might be used as a map value. There are a few big pieces to this: - I generalized the message-specific FFI endpoints in `rust/cpp_kernel/map.cc` to be able to additionally handle enums and primitive types as values. This mostly consisted of replacing `MessageLite*` parameters with a new `MapValue` tagged union. - On the Rust side, I added a new blanket implementation of `ProxiedInMapValue` in rust/cpp.rs. It relies on its value type to implement a new `CppMapTypeConversions` trait so that it can convert to and from the `MapValue` tagged union used for FFI. - In the Rust generated code, I deleted the generated `ProxiedInMapValue` implementations for messages and enums and replaced them with implementations of the `CppMapTypeConversions` trait. PiperOrigin-RevId: 687355817pull/18828/head
parent
2ff033011f
commit
cbb3edd86d
9 changed files with 617 additions and 662 deletions
@ -1,122 +0,0 @@ |
||||
#ifndef GOOGLE_PROTOBUF_RUST_CPP_KERNEL_MAP_H__ |
||||
#define GOOGLE_PROTOBUF_RUST_CPP_KERNEL_MAP_H__ |
||||
|
||||
#include <memory> |
||||
#include <type_traits> |
||||
|
||||
#include "google/protobuf/map.h" |
||||
#include "google/protobuf/message_lite.h" |
||||
#include "rust/cpp_kernel/strings.h" |
||||
|
||||
namespace google { |
||||
namespace protobuf { |
||||
namespace rust { |
||||
|
||||
// String and bytes values are passed across the FFI boundary as owned raw
|
||||
// pointers when we do map insertions. Unlike other types, they have to be
|
||||
// explicitly deleted. This MakeCleanup() helper does nothing by default, but
|
||||
// for std::string pointers it returns a std::unique_ptr to take ownership of
|
||||
// the raw pointer.
|
||||
template <typename T> |
||||
auto MakeCleanup(T value) { |
||||
if constexpr (std::is_same<T, std::string*>::value) { |
||||
return std::unique_ptr<std::string>(value); |
||||
} else { |
||||
return 0; |
||||
} |
||||
} |
||||
|
||||
} // namespace rust
|
||||
} // namespace protobuf
|
||||
} // namespace google
|
||||
|
||||
// Defines concrete thunks to access typed map methods from Rust.
|
||||
#define __PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \ |
||||
key_ty, rust_key_ty, ffi_key_ty, to_cpp_key, to_ffi_key, value_ty, \
|
||||
rust_value_ty, ffi_view_ty, ffi_value_ty, to_cpp_value, to_ffi_value) \
|
||||
google::protobuf::Map<key_ty, value_ty>* \
|
||||
proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_new() { \
|
||||
return new google::protobuf::Map<key_ty, value_ty>(); \
|
||||
} \
|
||||
void proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_free( \
|
||||
google::protobuf::Map<key_ty, value_ty>* m) { \
|
||||
delete m; \
|
||||
} \
|
||||
void proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_clear( \
|
||||
google::protobuf::Map<key_ty, value_ty>* m) { \
|
||||
m->clear(); \
|
||||
} \
|
||||
size_t proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_size( \
|
||||
const google::protobuf::Map<key_ty, value_ty>* m) { \
|
||||
return m->size(); \
|
||||
} \
|
||||
bool proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_insert( \
|
||||
google::protobuf::Map<key_ty, value_ty>* m, ffi_key_ty key, ffi_value_ty value) { \
|
||||
auto cleanup = google::protobuf::rust::MakeCleanup(value); \
|
||||
(void)cleanup; \
|
||||
auto iter_and_inserted = m->try_emplace(to_cpp_key, to_cpp_value); \
|
||||
if (!iter_and_inserted.second) { \
|
||||
iter_and_inserted.first->second = to_cpp_value; \
|
||||
} \
|
||||
return iter_and_inserted.second; \
|
||||
} \
|
||||
bool proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_get( \
|
||||
const google::protobuf::Map<key_ty, value_ty>* m, ffi_key_ty key, \
|
||||
ffi_view_ty* value) { \
|
||||
auto cpp_key = to_cpp_key; \
|
||||
auto it = m->find(cpp_key); \
|
||||
if (it == m->end()) { \
|
||||
return false; \
|
||||
} \
|
||||
auto& cpp_value = it->second; \
|
||||
*value = to_ffi_value; \
|
||||
return true; \
|
||||
} \
|
||||
google::protobuf::internal::UntypedMapIterator \
|
||||
proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_iter( \
|
||||
const google::protobuf::Map<key_ty, value_ty>* m) { \
|
||||
return google::protobuf::internal::UntypedMapIterator::FromTyped(m->cbegin()); \
|
||||
} \
|
||||
void proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_iter_get( \
|
||||
const google::protobuf::internal::UntypedMapIterator* iter, ffi_key_ty* key, \
|
||||
ffi_view_ty* value) { \
|
||||
auto typed_iter = \
|
||||
iter->ToTyped<google::protobuf::Map<key_ty, value_ty>::const_iterator>(); \
|
||||
const auto& cpp_key = typed_iter->first; \
|
||||
const auto& cpp_value = typed_iter->second; \
|
||||
*key = to_ffi_key; \
|
||||
*value = to_ffi_value; \
|
||||
} \
|
||||
bool proto2_rust_thunk_Map_##rust_key_ty##_##rust_value_ty##_remove( \
|
||||
google::protobuf::Map<key_ty, value_ty>* m, ffi_key_ty key, ffi_view_ty* value) { \
|
||||
auto cpp_key = to_cpp_key; \
|
||||
auto num_removed = m->erase(cpp_key); \
|
||||
return num_removed > 0; \
|
||||
} |
||||
|
||||
// Defines the map thunks for all supported key types for a given value type.
|
||||
#define __PB_RUST_EXPOSE_SCALAR_MAP_METHODS_FOR_VALUE_TYPE( \ |
||||
value_ty, rust_value_ty, ffi_view_ty, ffi_value_ty, to_cpp_value, \
|
||||
to_ffi_value) \
|
||||
__PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \
|
||||
int32_t, i32, int32_t, key, cpp_key, value_ty, rust_value_ty, \
|
||||
ffi_view_ty, ffi_value_ty, to_cpp_value, to_ffi_value); \
|
||||
__PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \
|
||||
uint32_t, u32, uint32_t, key, cpp_key, value_ty, rust_value_ty, \
|
||||
ffi_view_ty, ffi_value_ty, to_cpp_value, to_ffi_value); \
|
||||
__PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \
|
||||
bool, bool, bool, key, cpp_key, value_ty, rust_value_ty, ffi_view_ty, \
|
||||
ffi_value_ty, to_cpp_value, to_ffi_value); \
|
||||
__PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \
|
||||
uint64_t, u64, uint64_t, key, cpp_key, value_ty, rust_value_ty, \
|
||||
ffi_view_ty, ffi_value_ty, to_cpp_value, to_ffi_value); \
|
||||
__PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \
|
||||
int64_t, i64, int64_t, key, cpp_key, value_ty, rust_value_ty, \
|
||||
ffi_view_ty, ffi_value_ty, to_cpp_value, to_ffi_value); \
|
||||
__PB_RUST_EXPOSE_SCALAR_MAP_METHODS( \
|
||||
std::string, ProtoString, google::protobuf::rust::PtrAndLen, \
|
||||
std::string(key.ptr, key.len), \
|
||||
(google::protobuf::rust::PtrAndLen{cpp_key.data(), cpp_key.size()}), value_ty, \
|
||||
rust_value_ty, ffi_view_ty, ffi_value_ty, to_cpp_value, to_ffi_value); |
||||
|
||||
#endif // GOOGLE_PROTOBUF_RUST_CPP_KERNEL_MAP_H__
|
Loading…
Reference in new issue