/* * This source code is released into the public domain. */ module; #include #include #include #include export module nihil.ucl:real; import :object; namespace nihil::ucl { export struct real final : object { using contained_type = double; inline static constexpr object_type ucl_type = object_type::real; // Create a new real from a UCL object. real(ref_t, ::ucl_object_t const *uobj) : object(nihil::ucl::ref, uobj) { if (type() != ucl_type) throw type_mismatch(ucl_type, type()); } real(noref_t, ::ucl_object_t *uobj) : object(noref, uobj) { if (type() != ucl_type) throw type_mismatch(ucl_type, type()); } // Create a default-initialised real. real() : real(0) {} // Create a new real from a value. explicit real(contained_type value) : object(noref, ::ucl_object_fromdouble(value)) { if (_object == nullptr) throw error("failed to create UCL object"); } // Return the value of this real. auto value(this real const &self) -> contained_type { auto v = contained_type{}; auto const *uobj = self.get_ucl_object(); if (::ucl_object_todouble_safe(uobj, &v)) return v; std::abort(); } }; /* * Comparison operators. */ export auto operator== (real const &a, real const &b) -> bool { return a.value() == b.value(); } export auto operator<=> (real const &a, real const &b) -> std::partial_ordering { return a.value() <=> b.value(); } export auto operator== (real const &a, real::contained_type b) -> bool { return a.value() == b; } export auto operator<=> (real const &a, real::contained_type b) -> std::partial_ordering { return a.value() <=> b; } } // namespace nihil::ucl