Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
122 lines
7.7 KiB
122 lines
7.7 KiB
#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__
|
|
|