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.
396 lines
13 KiB
396 lines
13 KiB
// Protocol Buffers - Google's data interchange format |
|
// Copyright 2023 Google LLC. All rights reserved. |
|
// https://developers.google.com/protocol-buffers/ |
|
// |
|
// Redistribution and use in source and binary forms, with or without |
|
// modification, are permitted provided that the following conditions are |
|
// met: |
|
// |
|
// * Redistributions of source code must retain the above copyright |
|
// notice, this list of conditions and the following disclaimer. |
|
// * Redistributions in binary form must reproduce the above |
|
// copyright notice, this list of conditions and the following disclaimer |
|
// in the documentation and/or other materials provided with the |
|
// distribution. |
|
// * Neither the name of Google LLC nor the names of its |
|
// contributors may be used to endorse or promote products derived from |
|
// this software without specific prior written permission. |
|
// |
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
#ifndef UPB_PROTOS_REPEATED_FIELD_ITERATOR_H_ |
|
#define UPB_PROTOS_REPEATED_FIELD_ITERATOR_H_ |
|
|
|
#include <cstddef> |
|
#include <cstring> |
|
#include <iterator> |
|
#include <type_traits> |
|
|
|
#include "absl/strings/string_view.h" |
|
#include "upb/protos/protos.h" |
|
#include "upb/upb/base/string_view.h" |
|
#include "upb/upb/collections/array.h" |
|
#include "upb/upb/mem/arena.h" |
|
#include "upb/upb/message/copy.h" |
|
|
|
// Must be last: |
|
#include "upb/upb/message/types.h" |
|
#include "upb/upb/port/def.inc" |
|
|
|
namespace protos { |
|
namespace internal { |
|
|
|
// TODO: Implement std iterator for messages |
|
template <typename T> |
|
class RepeatedFieldScalarProxy; |
|
template <typename T> |
|
class RepeatedFieldStringProxy; |
|
|
|
struct IteratorTestPeer; |
|
|
|
template <typename T> |
|
class Iterator; |
|
|
|
template <typename PolicyT> |
|
class ReferenceProxy; |
|
|
|
template <typename PolicyT> |
|
class InjectedRelationalsImpl { |
|
using RP = ReferenceProxy<PolicyT>; |
|
using V = typename PolicyT::value_type; |
|
friend bool operator==(RP a, V b) { return static_cast<V>(a) == b; } |
|
friend bool operator==(V a, RP b) { return a == static_cast<V>(b); } |
|
friend bool operator==(RP a, RP b) { |
|
return static_cast<V>(a) == static_cast<V>(b); |
|
} |
|
friend bool operator!=(RP a, V b) { return static_cast<V>(a) != b; } |
|
friend bool operator!=(V a, RP b) { return a != static_cast<V>(b); } |
|
friend bool operator!=(RP a, RP b) { |
|
return static_cast<V>(a) != static_cast<V>(b); |
|
} |
|
friend bool operator<(RP a, V b) { return static_cast<V>(a) < b; } |
|
friend bool operator<(V a, RP b) { return a < static_cast<V>(b); } |
|
friend bool operator<(RP a, RP b) { |
|
return static_cast<V>(a) < static_cast<V>(b); |
|
} |
|
friend bool operator<=(RP a, V b) { return static_cast<V>(a) <= b; } |
|
friend bool operator<=(V a, RP b) { return a <= static_cast<V>(b); } |
|
friend bool operator<=(RP a, RP b) { |
|
return static_cast<V>(a) <= static_cast<V>(b); |
|
} |
|
friend bool operator>(RP a, V b) { return static_cast<V>(a) > b; } |
|
friend bool operator>(V a, RP b) { return a > static_cast<V>(b); } |
|
friend bool operator>(RP a, RP b) { |
|
return static_cast<V>(a) > static_cast<V>(b); |
|
} |
|
friend bool operator>=(RP a, V b) { return static_cast<V>(a) >= b; } |
|
friend bool operator>=(V a, RP b) { return a >= static_cast<V>(b); } |
|
friend bool operator>=(RP a, RP b) { |
|
return static_cast<V>(a) >= static_cast<V>(b); |
|
} |
|
}; |
|
class NoInjectedRelationalsImpl {}; |
|
|
|
// We need to inject relationals for the string references because the |
|
// relationals for string_view are templates and won't allow for implicit |
|
// conversions from ReferenceProxy to string_view before deduction. |
|
template <typename PolicyT> |
|
using InjectedRelationals = std::conditional_t< |
|
std::is_same_v<std::remove_const_t<typename PolicyT::value_type>, |
|
absl::string_view>, |
|
InjectedRelationalsImpl<PolicyT>, NoInjectedRelationalsImpl>; |
|
|
|
template <typename PolicyT> |
|
class ReferenceProxy : InjectedRelationals<PolicyT> { |
|
using value_type = typename PolicyT::value_type; |
|
|
|
public: |
|
ReferenceProxy(const ReferenceProxy&) = default; |
|
ReferenceProxy& operator=(const ReferenceProxy& other) { |
|
// Assign through the references |
|
// TODO: Make this better for strings to avoid the copy. |
|
it_.Set(other.it_.Get()); |
|
return *this; |
|
} |
|
friend void swap(ReferenceProxy a, ReferenceProxy b) { a.it_.swap(b.it_); } |
|
|
|
operator value_type() const { return it_.Get(); } |
|
void operator=(const value_type& value) const { it_.Set(value); } |
|
void operator=(value_type&& value) const { it_.Set(std::move(value)); } |
|
Iterator<PolicyT> operator&() const { return Iterator<PolicyT>(it_); } |
|
|
|
private: |
|
friend IteratorTestPeer; |
|
friend ReferenceProxy<typename PolicyT::AddConst>; |
|
friend Iterator<PolicyT>; |
|
|
|
explicit ReferenceProxy(typename PolicyT::Payload elem) : it_(elem) {} |
|
typename PolicyT::Payload it_; |
|
}; |
|
|
|
template <template <typename> class PolicyTemplate, typename T> |
|
class ReferenceProxy<PolicyTemplate<const T>> |
|
: InjectedRelationals<PolicyTemplate<const T>> { |
|
using PolicyT = PolicyTemplate<const T>; |
|
using value_type = typename PolicyT::value_type; |
|
|
|
public: |
|
ReferenceProxy(ReferenceProxy<PolicyTemplate<T>> p) : it_(p.it_) {} |
|
ReferenceProxy(const ReferenceProxy&) = default; |
|
ReferenceProxy& operator=(const ReferenceProxy&) = delete; |
|
|
|
operator value_type() const { return it_.Get(); } |
|
Iterator<PolicyT> operator&() const { return Iterator<PolicyT>(it_); } |
|
|
|
private: |
|
friend IteratorTestPeer; |
|
friend Iterator<PolicyT>; |
|
|
|
explicit ReferenceProxy(typename PolicyT::Payload elem) : it_(elem) {} |
|
typename PolicyT::Payload it_; |
|
}; |
|
|
|
template <typename PolicyT> |
|
class Iterator { |
|
public: |
|
using iterator_category = std::random_access_iterator_tag; |
|
using value_type = std::remove_const_t<typename PolicyT::value_type>; |
|
using difference_type = std::ptrdiff_t; |
|
using pointer = Iterator; |
|
using reference = |
|
std::conditional_t<PolicyT::kUseReferenceProxy, ReferenceProxy<PolicyT>, |
|
typename PolicyT::value_type>; |
|
|
|
constexpr Iterator() noexcept : it_(nullptr) {} |
|
Iterator(const Iterator& other) = default; |
|
Iterator& operator=(const Iterator& other) = default; |
|
template < |
|
typename P = PolicyT, |
|
typename = std::enable_if_t<std::is_const<typename P::value_type>::value>> |
|
Iterator(const Iterator<typename P::RemoveConst>& other) : it_(other.it_) {} |
|
|
|
constexpr reference operator*() const noexcept { |
|
if constexpr (PolicyT::kUseReferenceProxy) { |
|
return reference(it_); |
|
} else { |
|
return it_.Get(); |
|
} |
|
} |
|
// No operator-> needed because T is a scalar. |
|
|
|
private: |
|
// Hide the internal type. |
|
using iterator = Iterator; |
|
|
|
public: |
|
// {inc,dec}rementable |
|
constexpr iterator& operator++() noexcept { |
|
it_.AddOffset(1); |
|
return *this; |
|
} |
|
constexpr iterator operator++(int) noexcept { |
|
auto copy = *this; |
|
++*this; |
|
return copy; |
|
} |
|
constexpr iterator& operator--() noexcept { |
|
it_.AddOffset(-1); |
|
return *this; |
|
} |
|
constexpr iterator operator--(int) noexcept { |
|
auto copy = *this; |
|
--*this; |
|
return copy; |
|
} |
|
|
|
// equality_comparable |
|
friend constexpr bool operator==(const iterator& x, |
|
const iterator& y) noexcept { |
|
return x.it_.Index() == y.it_.Index(); |
|
} |
|
friend constexpr bool operator!=(const iterator& x, |
|
const iterator& y) noexcept { |
|
return !(x == y); |
|
} |
|
|
|
// less_than_comparable |
|
friend constexpr bool operator<(const iterator& x, |
|
const iterator& y) noexcept { |
|
return x.it_.Index() < y.it_.Index(); |
|
} |
|
friend constexpr bool operator<=(const iterator& x, |
|
const iterator& y) noexcept { |
|
return !(y < x); |
|
} |
|
friend constexpr bool operator>(const iterator& x, |
|
const iterator& y) noexcept { |
|
return y < x; |
|
} |
|
friend constexpr bool operator>=(const iterator& x, |
|
const iterator& y) noexcept { |
|
return !(x < y); |
|
} |
|
|
|
constexpr iterator& operator+=(difference_type d) noexcept { |
|
it_.AddOffset(d); |
|
return *this; |
|
} |
|
constexpr iterator operator+(difference_type d) const noexcept { |
|
auto copy = *this; |
|
copy += d; |
|
return copy; |
|
} |
|
friend constexpr iterator operator+(const difference_type d, |
|
iterator it) noexcept { |
|
return it + d; |
|
} |
|
|
|
constexpr iterator& operator-=(difference_type d) noexcept { |
|
it_.AddOffset(-d); |
|
return *this; |
|
} |
|
constexpr iterator operator-(difference_type d) const noexcept { |
|
auto copy = *this; |
|
copy -= d; |
|
return copy; |
|
} |
|
|
|
// indexable |
|
constexpr reference operator[](difference_type d) const noexcept { |
|
auto copy = *this; |
|
copy += d; |
|
return *copy; |
|
} |
|
|
|
// random access iterator |
|
friend constexpr difference_type operator-(iterator x, iterator y) noexcept { |
|
return x.it_.Index() - y.it_.Index(); |
|
} |
|
|
|
private: |
|
friend IteratorTestPeer; |
|
friend ReferenceProxy<PolicyT>; |
|
friend Iterator<typename PolicyT::AddConst>; |
|
template <typename U> |
|
friend class RepeatedFieldScalarProxy; |
|
template <typename U> |
|
friend class RepeatedFieldStringProxy; |
|
template <typename U> |
|
friend class RepeatedFieldProxy; |
|
|
|
// Create from internal::RepeatedFieldScalarProxy. |
|
explicit Iterator(typename PolicyT::Payload it) noexcept : it_(it) {} |
|
|
|
// The internal iterator. |
|
typename PolicyT::Payload it_; |
|
}; |
|
|
|
template <typename T> |
|
struct ScalarIteratorPolicy { |
|
static constexpr bool kUseReferenceProxy = true; |
|
using value_type = T; |
|
using RemoveConst = ScalarIteratorPolicy<std::remove_const_t<T>>; |
|
using AddConst = ScalarIteratorPolicy<const T>; |
|
|
|
struct Payload { |
|
T* value; |
|
void AddOffset(ptrdiff_t offset) { value += offset; } |
|
T Get() const { return *value; } |
|
void Set(T new_value) const { *value = new_value; } |
|
T* Index() const { return value; } |
|
|
|
void swap(Payload& other) { |
|
using std::swap; |
|
swap(*value, *other.value); |
|
} |
|
|
|
operator typename ScalarIteratorPolicy<const T>::Payload() const { |
|
return {value}; |
|
} |
|
}; |
|
}; |
|
|
|
template <typename T> |
|
struct StringIteratorPolicy { |
|
static constexpr bool kUseReferenceProxy = true; |
|
using value_type = T; |
|
using RemoveConst = StringIteratorPolicy<std::remove_const_t<T>>; |
|
using AddConst = StringIteratorPolicy<const T>; |
|
|
|
struct Payload { |
|
using Array = |
|
std::conditional_t<std::is_const_v<T>, const upb_Array, upb_Array>; |
|
Array* arr; |
|
upb_Arena* arena; |
|
size_t index; |
|
|
|
void AddOffset(ptrdiff_t offset) { index += offset; } |
|
absl::string_view Get() const { |
|
upb_MessageValue message_value = upb_Array_Get(arr, index); |
|
return absl::string_view(message_value.str_val.data, |
|
message_value.str_val.size); |
|
} |
|
void Set(absl::string_view new_value) const { |
|
char* data = |
|
static_cast<char*>(upb_Arena_Malloc(arena, new_value.size())); |
|
memcpy(data, new_value.data(), new_value.size()); |
|
upb_MessageValue message_value; |
|
message_value.str_val = |
|
upb_StringView_FromDataAndSize(data, new_value.size()); |
|
upb_Array_Set(arr, index, message_value); |
|
} |
|
size_t Index() const { return index; } |
|
|
|
void swap(Payload& other) { |
|
upb_MessageValue a = upb_Array_Get(this->arr, this->index); |
|
upb_MessageValue b = upb_Array_Get(other.arr, other.index); |
|
upb_Array_Set(this->arr, this->index, b); |
|
upb_Array_Set(other.arr, other.index, a); |
|
} |
|
|
|
operator typename StringIteratorPolicy<const T>::Payload() const { |
|
return {arr, arena, index}; |
|
} |
|
}; |
|
}; |
|
|
|
template <typename T> |
|
struct MessageIteratorPolicy { |
|
static constexpr bool kUseReferenceProxy = false; |
|
using value_type = std::conditional_t<std::is_const_v<T>, typename T::CProxy, |
|
typename T::Proxy>; |
|
using RemoveConst = MessageIteratorPolicy<std::remove_const_t<T>>; |
|
using AddConst = MessageIteratorPolicy<const T>; |
|
|
|
struct Payload { |
|
using Array = |
|
std::conditional_t<std::is_const_v<T>, const upb_Array, upb_Array>; |
|
upb_Message** arr; |
|
upb_Arena* arena; |
|
|
|
void AddOffset(ptrdiff_t offset) { arr += offset; } |
|
auto Get() const { |
|
if constexpr (std::is_const_v<T>) { |
|
return ::protos::internal::CreateMessage< |
|
typename std::remove_const_t<T>>(*arr, arena); |
|
} else { |
|
return ::protos::internal::CreateMessageProxy<T>(*arr, arena); |
|
} |
|
} |
|
auto Index() const { return arr; } |
|
}; |
|
}; |
|
|
|
} // namespace internal |
|
} // namespace protos |
|
|
|
#endif // UPB_PROTOS_REPEATED_FIELD_ITERATOR_H_
|
|
|