| // Copyright 2018 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef LIB_FIDL_CPP_STRING_H_ |
| #define LIB_FIDL_CPP_STRING_H_ |
| |
| #include <lib/stdcompat/optional.h> |
| #include <lib/stdcompat/string_view.h> |
| #include <zircon/assert.h> |
| |
| #include <iosfwd> |
| #include <string> |
| #include <utility> |
| |
| #include "lib/fidl/cpp/coding_traits.h" |
| #include "lib/fidl/cpp/traits.h" |
| |
| namespace fidl { |
| |
| class StringPtr final : public cpp17::optional<std::string> { |
| public: |
| constexpr StringPtr() = default; |
| |
| constexpr StringPtr(cpp17::nullopt_t) noexcept {} |
| // Deprecated in favor of cpp17::nullopt_t. |
| constexpr StringPtr(std::nullptr_t) noexcept {} |
| |
| StringPtr(const StringPtr&) = default; |
| StringPtr& operator=(const StringPtr&) = default; |
| |
| StringPtr(StringPtr&&) = default; |
| StringPtr& operator=(StringPtr&&) = default; |
| |
| // Move construct and move assignment from the value type |
| constexpr StringPtr(std::string&& value) : cpp17::optional<std::string>(std::move(value)) {} |
| constexpr StringPtr& operator=(std::string&& value) { |
| cpp17::optional<std::string>::operator=(std::move(value)); |
| return *this; |
| } |
| |
| // Copy construct and copy assignment from the value type |
| constexpr StringPtr(const std::string& value) : cpp17::optional<std::string>(value) {} |
| constexpr StringPtr& operator=(const std::string& value) { |
| cpp17::optional<std::string>::operator=(value); |
| return *this; |
| } |
| |
| // Construct from string literals |
| template <size_t N> |
| constexpr StringPtr(const char (&literal)[N]) : cpp17::optional<std::string>(literal) {} |
| template <size_t N> |
| constexpr StringPtr& operator=(const char (&literal)[N]) { |
| cpp17::optional<std::string>::operator=(literal); |
| return *this; |
| } |
| |
| // Construct from string pointers |
| StringPtr(const char* value) : cpp17::optional<std::string>(value) {} |
| StringPtr(const char* value, size_t size) |
| : cpp17::optional<std::string>(std::string(value, size)) {} |
| StringPtr& operator=(const char* value) { |
| cpp17::optional<std::string>::operator=(value); |
| return *this; |
| } |
| |
| // Construct from string views. |
| StringPtr(cpp17::string_view value) |
| : cpp17::optional<std::string>(std::string(value.data(), value.size())) {} |
| StringPtr& operator=(cpp17::string_view value) { |
| cpp17::optional<std::string>::operator=(std::string(value.data(), value.size())); |
| return *this; |
| } |
| |
| // Override unchecked accessors with versions that check. |
| constexpr std::string* operator->() { |
| if (!cpp17::optional<std::string>::has_value()) { |
| __builtin_trap(); |
| } |
| return cpp17::optional<std::string>::operator->(); |
| } |
| constexpr const std::string* operator->() const { |
| if (!cpp17::optional<std::string>::has_value()) { |
| __builtin_trap(); |
| } |
| return cpp17::optional<std::string>::operator->(); |
| } |
| |
| // Destructor. |
| ~StringPtr() = default; |
| }; |
| |
| template <> |
| struct Equality<StringPtr> { |
| bool operator()(const StringPtr& lhs, const StringPtr& rhs) const { |
| if (!lhs.has_value()) { |
| return !rhs.has_value(); |
| } |
| return rhs.has_value() && lhs.value() == rhs.value(); |
| } |
| }; |
| |
| inline bool operator==(const StringPtr& lhs, const StringPtr& rhs) { |
| return ::fidl::Equality<StringPtr>{}(lhs, rhs); |
| } |
| |
| inline bool operator==(const char* lhs, const StringPtr& rhs) { |
| if (lhs == nullptr) { |
| return !rhs.has_value(); |
| } |
| return rhs.has_value() && lhs == rhs.value(); |
| } |
| |
| inline bool operator==(const StringPtr& lhs, const char* rhs) { |
| if (!lhs.has_value()) { |
| return rhs == nullptr; |
| } |
| return rhs != nullptr && lhs.value() == rhs; |
| } |
| |
| inline bool operator!=(const StringPtr& lhs, const StringPtr& rhs) { return !(lhs == rhs); } |
| |
| inline bool operator!=(const char* lhs, const StringPtr& rhs) { return !(lhs == rhs); } |
| |
| inline bool operator!=(const StringPtr& lhs, const char* rhs) { return !(lhs == rhs); } |
| |
| inline bool operator<(const StringPtr& lhs, const StringPtr& rhs) { |
| if (!lhs.has_value() || !rhs.has_value()) { |
| return rhs.has_value(); |
| } |
| return *lhs < *rhs; |
| } |
| |
| inline bool operator<(const char* lhs, const StringPtr& rhs) { |
| if (lhs == nullptr || !rhs.has_value()) { |
| return rhs.has_value(); |
| } |
| return lhs < *rhs; |
| } |
| |
| inline bool operator<(const StringPtr& lhs, const char* rhs) { |
| if (!lhs.has_value() || rhs == nullptr) { |
| return rhs != nullptr; |
| } |
| return *lhs < rhs; |
| } |
| |
| inline bool operator>(const StringPtr& lhs, const StringPtr& rhs) { |
| if (!lhs.has_value() || !rhs.has_value()) { |
| return !!lhs.has_value(); |
| } |
| return *lhs > *rhs; |
| } |
| |
| inline bool operator>(const char* lhs, const StringPtr& rhs) { |
| if (lhs == nullptr || !rhs.has_value()) { |
| return lhs != nullptr; |
| } |
| return lhs > *rhs; |
| } |
| |
| inline bool operator>(const StringPtr& lhs, const char* rhs) { |
| if (!lhs.has_value() || rhs == nullptr) { |
| return lhs != nullptr; |
| } |
| return *lhs > rhs; |
| } |
| |
| inline bool operator<=(const StringPtr& lhs, const StringPtr& rhs) { return !(lhs > rhs); } |
| |
| inline bool operator<=(const char* lhs, const StringPtr& rhs) { return !(lhs > rhs); } |
| |
| inline bool operator<=(const StringPtr& lhs, const char* rhs) { return !(lhs > rhs); } |
| |
| inline bool operator>=(const StringPtr& lhs, const StringPtr& rhs) { return !(lhs < rhs); } |
| |
| inline bool operator>=(const char* lhs, const StringPtr& rhs) { return !(lhs < rhs); } |
| |
| inline bool operator>=(const StringPtr& lhs, const char* rhs) { return !(lhs < rhs); } |
| |
| inline std::ostream& operator<<(std::ostream& out, const StringPtr& str) { |
| return out << str.value_or(""); |
| } |
| |
| template <> |
| struct CodingTraits<::std::string> { |
| static constexpr size_t inline_size_v2 = sizeof(fidl_string_t); |
| template <class EncoderImpl> |
| static void Encode(EncoderImpl* encoder, std::string* value, size_t offset, |
| cpp17::optional<HandleInformation> maybe_handle_info = cpp17::nullopt) { |
| ZX_DEBUG_ASSERT(!maybe_handle_info); |
| const size_t size = value->size(); |
| fidl_string_t* string = encoder->template GetPtr<fidl_string_t>(offset); |
| string->size = size; |
| string->data = reinterpret_cast<char*>(FIDL_ALLOC_PRESENT); |
| size_t base = encoder->Alloc(size); |
| char* payload = encoder->template GetPtr<char>(base); |
| memcpy(payload, value->data(), size); |
| } |
| template <class DecoderImpl> |
| static void Decode(DecoderImpl* decoder, std::string* value, size_t offset) { |
| fidl_string_t* string = decoder->template GetPtr<fidl_string_t>(offset); |
| ZX_ASSERT(string->data != nullptr); |
| *value = std::string(string->data, string->size); |
| } |
| }; |
| |
| template <> |
| struct CodingTraits<StringPtr> { |
| static constexpr size_t inline_size_v2 = sizeof(fidl_string_t); |
| template <class EncoderImpl> |
| static void Encode(EncoderImpl* encoder, StringPtr* value, size_t offset, |
| cpp17::optional<HandleInformation> maybe_handle_info = cpp17::nullopt) { |
| ZX_DEBUG_ASSERT(!maybe_handle_info); |
| if (value->has_value()) { |
| ::fidl::CodingTraits<std::string>::Encode(encoder, &value->value(), offset, cpp17::nullopt); |
| } else { |
| fidl_string_t* string = encoder->template GetPtr<fidl_string_t>(offset); |
| string->size = 0u; |
| string->data = reinterpret_cast<char*>(FIDL_ALLOC_ABSENT); |
| } |
| } |
| template <class DecoderImpl> |
| static void Decode(DecoderImpl* decoder, StringPtr* value, size_t offset) { |
| fidl_string_t* string = decoder->template GetPtr<fidl_string_t>(offset); |
| if (string->data) { |
| std::string string_value; |
| ::fidl::CodingTraits<std::string>::Decode(decoder, &string_value, offset); |
| (*value) = std::move(string_value); |
| } else { |
| value->reset(); |
| } |
| } |
| }; |
| |
| } // namespace fidl |
| |
| namespace fit { |
| |
| inline std::ostream& operator<<(std::ostream& out, const cpp17::optional<std::string>& str) { |
| return out << str.value_or(""); |
| } |
| |
| } // namespace fit |
| |
| #endif // LIB_FIDL_CPP_STRING_H_ |