123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- /*************************************************************************
- *
- * Copyright 2021 Realm Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- **************************************************************************/
- #pragma once
- #include <atomic>
- #include <cstdint>
- #include <iosfwd>
- #include <string>
- #include "realm/error_codes.hpp"
- #include "realm/util/bind_ptr.hpp"
- #include "realm/util/features.h"
- namespace realm {
- class REALM_NODISCARD Status {
- public:
- /*
- * This is the best way to construct a Status that represents a non-error condition.
- */
- static inline Status OK();
- /*
- * You can construct a Status from anything that can construct a std::string_view.
- */
- template <typename Reason, std::enable_if_t<std::is_constructible_v<std::string_view, Reason>, int> = 0>
- Status(ErrorCodes::Error code, Reason&& reason)
- : m_error(ErrorInfo::create(code, std::string_view{reason}))
- {
- }
- template <typename Reason, std::enable_if_t<std::is_constructible_v<std::string_view, Reason>, int> = 0>
- Status(std::error_code code, Reason&& reason)
- {
- if (code) {
- m_error = ErrorInfo::create(ErrorCodes::SystemError, std::string_view{reason});
- m_error->m_std_error_code = code;
- }
- }
- std::error_code get_std_error_code() const
- {
- return m_error ? m_error->m_std_error_code : std::error_code{};
- }
- /*
- * Copying a Status is just copying an intrusive pointer - i.e. very cheap. Moving them is similarly cheap.
- */
- inline Status(const Status& other);
- inline Status& operator=(const Status& other);
- inline Status(Status&& other) noexcept;
- inline Status& operator=(Status&& other) noexcept;
- inline bool is_ok() const noexcept;
- inline const std::string& reason() const noexcept;
- inline ErrorCodes::Error code() const noexcept;
- inline std::string_view code_string() const noexcept;
- /*
- * This class is marked nodiscard so that we always handle errors. If there is a place where we need
- * to explicitly ignore an error, you can call this function, which does nothing, to satisfy the compiler.
- */
- void ignore() const noexcept {}
- private:
- friend struct SystemError;
- void set_std_error_code(std::error_code code)
- {
- m_error->m_std_error_code = code;
- }
- Status() = default;
- struct ErrorInfo {
- mutable std::atomic<uint32_t> m_refs;
- const ErrorCodes::Error m_code;
- const std::string m_reason;
- // The addition of this system error code may be temporary if we ever get all use of
- // std::error_code migrated to the unified exception system
- std::error_code m_std_error_code;
- static util::bind_ptr<ErrorInfo> create(ErrorCodes::Error code, std::string_view reason);
- protected:
- template <typename>
- friend class ::realm::util::bind_ptr;
- inline void bind_ptr() const noexcept
- {
- m_refs.fetch_add(1, std::memory_order_relaxed);
- }
- inline void unbind_ptr() const noexcept
- {
- if (m_refs.fetch_sub(1, std::memory_order_acq_rel) == 1) {
- delete this;
- }
- }
- private:
- ErrorInfo(ErrorCodes::Error code, std::string_view reason);
- };
- util::bind_ptr<ErrorInfo> m_error = {};
- };
- std::ostream& operator<<(std::ostream& out, const Status& val);
- inline bool operator==(const Status& lhs, const Status& rhs) noexcept
- {
- return lhs.code() == rhs.code();
- }
- inline bool operator!=(const Status& lhs, const Status& rhs) noexcept
- {
- return lhs.code() != rhs.code();
- }
- inline bool operator==(const Status& lhs, ErrorCodes::Error rhs) noexcept
- {
- return lhs.code() == rhs;
- }
- inline bool operator!=(const Status& lhs, ErrorCodes::Error rhs) noexcept
- {
- return lhs.code() != rhs;
- }
- inline Status Status::OK()
- {
- // Returns a status with m_error set to nullptr.
- return Status{};
- }
- inline Status::Status(const Status& other)
- : m_error(other.m_error)
- {
- }
- inline Status& Status::operator=(const Status& other)
- {
- m_error = other.m_error;
- return *this;
- }
- inline Status::Status(Status&& other) noexcept
- : m_error(std::move(other.m_error))
- {
- }
- inline Status& Status::operator=(Status&& other) noexcept
- {
- m_error = std::move(other.m_error);
- return *this;
- }
- inline bool Status::is_ok() const noexcept
- {
- return !m_error;
- }
- inline const std::string& Status::reason() const noexcept
- {
- static const std::string empty;
- return m_error ? m_error->m_reason : empty;
- }
- inline ErrorCodes::Error Status::code() const noexcept
- {
- return m_error ? m_error->m_code : ErrorCodes::OK;
- }
- inline std::string_view Status::code_string() const noexcept
- {
- return ErrorCodes::error_string(code());
- }
- } // namespace realm
|