status.hpp 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. /*************************************************************************
  2. *
  3. * Copyright 2021 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #pragma once
  19. #include <atomic>
  20. #include <cstdint>
  21. #include <iosfwd>
  22. #include <string>
  23. #include "realm/error_codes.hpp"
  24. #include "realm/util/bind_ptr.hpp"
  25. #include "realm/util/features.h"
  26. namespace realm {
  27. class REALM_NODISCARD Status {
  28. public:
  29. /*
  30. * This is the best way to construct a Status that represents a non-error condition.
  31. */
  32. static inline Status OK();
  33. /*
  34. * You can construct a Status from anything that can construct a std::string_view.
  35. */
  36. template <typename Reason, std::enable_if_t<std::is_constructible_v<std::string_view, Reason>, int> = 0>
  37. Status(ErrorCodes::Error code, Reason&& reason)
  38. : m_error(ErrorInfo::create(code, std::string_view{reason}))
  39. {
  40. }
  41. template <typename Reason, std::enable_if_t<std::is_constructible_v<std::string_view, Reason>, int> = 0>
  42. Status(std::error_code code, Reason&& reason)
  43. {
  44. if (code) {
  45. m_error = ErrorInfo::create(ErrorCodes::SystemError, std::string_view{reason});
  46. m_error->m_std_error_code = code;
  47. }
  48. }
  49. std::error_code get_std_error_code() const
  50. {
  51. return m_error ? m_error->m_std_error_code : std::error_code{};
  52. }
  53. /*
  54. * Copying a Status is just copying an intrusive pointer - i.e. very cheap. Moving them is similarly cheap.
  55. */
  56. inline Status(const Status& other);
  57. inline Status& operator=(const Status& other);
  58. inline Status(Status&& other) noexcept;
  59. inline Status& operator=(Status&& other) noexcept;
  60. inline bool is_ok() const noexcept;
  61. inline const std::string& reason() const noexcept;
  62. inline ErrorCodes::Error code() const noexcept;
  63. inline std::string_view code_string() const noexcept;
  64. /*
  65. * This class is marked nodiscard so that we always handle errors. If there is a place where we need
  66. * to explicitly ignore an error, you can call this function, which does nothing, to satisfy the compiler.
  67. */
  68. void ignore() const noexcept {}
  69. private:
  70. friend struct SystemError;
  71. void set_std_error_code(std::error_code code)
  72. {
  73. m_error->m_std_error_code = code;
  74. }
  75. Status() = default;
  76. struct ErrorInfo {
  77. mutable std::atomic<uint32_t> m_refs;
  78. const ErrorCodes::Error m_code;
  79. const std::string m_reason;
  80. // The addition of this system error code may be temporary if we ever get all use of
  81. // std::error_code migrated to the unified exception system
  82. std::error_code m_std_error_code;
  83. static util::bind_ptr<ErrorInfo> create(ErrorCodes::Error code, std::string_view reason);
  84. protected:
  85. template <typename>
  86. friend class ::realm::util::bind_ptr;
  87. inline void bind_ptr() const noexcept
  88. {
  89. m_refs.fetch_add(1, std::memory_order_relaxed);
  90. }
  91. inline void unbind_ptr() const noexcept
  92. {
  93. if (m_refs.fetch_sub(1, std::memory_order_acq_rel) == 1) {
  94. delete this;
  95. }
  96. }
  97. private:
  98. ErrorInfo(ErrorCodes::Error code, std::string_view reason);
  99. };
  100. util::bind_ptr<ErrorInfo> m_error = {};
  101. };
  102. std::ostream& operator<<(std::ostream& out, const Status& val);
  103. inline bool operator==(const Status& lhs, const Status& rhs) noexcept
  104. {
  105. return lhs.code() == rhs.code();
  106. }
  107. inline bool operator!=(const Status& lhs, const Status& rhs) noexcept
  108. {
  109. return lhs.code() != rhs.code();
  110. }
  111. inline bool operator==(const Status& lhs, ErrorCodes::Error rhs) noexcept
  112. {
  113. return lhs.code() == rhs;
  114. }
  115. inline bool operator!=(const Status& lhs, ErrorCodes::Error rhs) noexcept
  116. {
  117. return lhs.code() != rhs;
  118. }
  119. inline Status Status::OK()
  120. {
  121. // Returns a status with m_error set to nullptr.
  122. return Status{};
  123. }
  124. inline Status::Status(const Status& other)
  125. : m_error(other.m_error)
  126. {
  127. }
  128. inline Status& Status::operator=(const Status& other)
  129. {
  130. m_error = other.m_error;
  131. return *this;
  132. }
  133. inline Status::Status(Status&& other) noexcept
  134. : m_error(std::move(other.m_error))
  135. {
  136. }
  137. inline Status& Status::operator=(Status&& other) noexcept
  138. {
  139. m_error = std::move(other.m_error);
  140. return *this;
  141. }
  142. inline bool Status::is_ok() const noexcept
  143. {
  144. return !m_error;
  145. }
  146. inline const std::string& Status::reason() const noexcept
  147. {
  148. static const std::string empty;
  149. return m_error ? m_error->m_reason : empty;
  150. }
  151. inline ErrorCodes::Error Status::code() const noexcept
  152. {
  153. return m_error ? m_error->m_code : ErrorCodes::OK;
  154. }
  155. inline std::string_view Status::code_string() const noexcept
  156. {
  157. return ErrorCodes::error_string(code());
  158. }
  159. } // namespace realm