Fix OkOrFail<status_t> conversion ambiguities
OkOrFail<status_t> has specialized conversions for Result<int, StatusT> to avoid ambiguous implicit conversion sequences. Since user conversion operators sequences can be followed by integral promotion, specializing for integral types is necessary. Specialize ResultError<StatusT> so calling code() returns a status_t instead of a StatusT and message() is implemented even when not carrying a string. Eventually, these classes should be combined. Add equality operators for ResultError<StatusT>. Bug: 219580167 Test: atest Errors_test.cpp Merged-In: I14acecfd2aef33c40e79ddb091e2f4af9291d837 Change-Id: Ifb5ed3c2d3452b10901e4aeb19368d873225d9ce
This commit is contained in:
parent
075b6d725e
commit
dde6034eac
|
@ -108,3 +108,65 @@ TEST(errors, result_in_status) {
|
||||||
status_t b = g(false);
|
status_t b = g(false);
|
||||||
EXPECT_EQ(PERMISSION_DENIED, b);
|
EXPECT_EQ(PERMISSION_DENIED, b);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(errors, conversion_promotion) {
|
||||||
|
constexpr size_t successVal = 10ull;
|
||||||
|
auto f = [&](bool success) -> Result<size_t, StatusT> {
|
||||||
|
OR_RETURN(success_or_fail(success));
|
||||||
|
return successVal;
|
||||||
|
};
|
||||||
|
auto s = f(true);
|
||||||
|
ASSERT_TRUE(s.ok());
|
||||||
|
EXPECT_EQ(s.value(), successVal);
|
||||||
|
auto r = f(false);
|
||||||
|
EXPECT_TRUE(!r.ok());
|
||||||
|
EXPECT_EQ(PERMISSION_DENIED, r.error().code());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(errors, conversion_promotion_bool) {
|
||||||
|
constexpr size_t successVal = true;
|
||||||
|
auto f = [&](bool success) -> Result<bool, StatusT> {
|
||||||
|
OR_RETURN(success_or_fail(success));
|
||||||
|
return successVal;
|
||||||
|
};
|
||||||
|
auto s = f(true);
|
||||||
|
ASSERT_TRUE(s.ok());
|
||||||
|
EXPECT_EQ(s.value(), successVal);
|
||||||
|
auto r = f(false);
|
||||||
|
EXPECT_TRUE(!r.ok());
|
||||||
|
EXPECT_EQ(PERMISSION_DENIED, r.error().code());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(errors, conversion_promotion_char) {
|
||||||
|
constexpr char successVal = 'a';
|
||||||
|
auto f = [&](bool success) -> Result<unsigned char, StatusT> {
|
||||||
|
OR_RETURN(success_or_fail(success));
|
||||||
|
return successVal;
|
||||||
|
};
|
||||||
|
auto s = f(true);
|
||||||
|
ASSERT_TRUE(s.ok());
|
||||||
|
EXPECT_EQ(s.value(), successVal);
|
||||||
|
auto r = f(false);
|
||||||
|
EXPECT_TRUE(!r.ok());
|
||||||
|
EXPECT_EQ(PERMISSION_DENIED, r.error().code());
|
||||||
|
}
|
||||||
|
|
||||||
|
struct IntContainer {
|
||||||
|
// Implicit conversion from int is desired
|
||||||
|
IntContainer(int val) : val_(val) {}
|
||||||
|
int val_;
|
||||||
|
};
|
||||||
|
|
||||||
|
TEST(errors, conversion_construct) {
|
||||||
|
constexpr int successVal = 10;
|
||||||
|
auto f = [&](bool success) -> Result<IntContainer, StatusT> {
|
||||||
|
OR_RETURN(success_or_fail(success));
|
||||||
|
return successVal;
|
||||||
|
};
|
||||||
|
auto s = f(true);
|
||||||
|
ASSERT_TRUE(s.ok());
|
||||||
|
EXPECT_EQ(s.value().val_, successVal);
|
||||||
|
auto r = f(false);
|
||||||
|
EXPECT_TRUE(!r.ok());
|
||||||
|
EXPECT_EQ(PERMISSION_DENIED, r.error().code());
|
||||||
|
}
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
// [1] build/soong/cc/config/global.go#commonGlobalIncludes
|
// [1] build/soong/cc/config/global.go#commonGlobalIncludes
|
||||||
#include <android-base/errors.h>
|
#include <android-base/errors.h>
|
||||||
#include <android-base/result.h>
|
#include <android-base/result.h>
|
||||||
|
#include <log/log_main.h>
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
|
@ -44,13 +45,58 @@ struct StatusT {
|
||||||
status_t val_;
|
status_t val_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
namespace base {
|
namespace base {
|
||||||
|
// TODO(b/221235365) StatusT fulfill ResultError contract and cleanup.
|
||||||
|
|
||||||
|
// Unlike typical ResultError types, the underlying code should be a status_t
|
||||||
|
// instead of a StatusT. We also special-case message generation.
|
||||||
|
template<>
|
||||||
|
struct ResultError<StatusT, false> {
|
||||||
|
ResultError(status_t s) : val_(s) {
|
||||||
|
LOG_FATAL_IF(s == OK, "Result error should not hold success");
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
operator expected<T, ResultError<StatusT, false>>() const {
|
||||||
|
return unexpected(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string message() const { return statusToString(val_); }
|
||||||
|
status_t code() const { return val_; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
const status_t val_;
|
||||||
|
};
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct ResultError<StatusT, true> {
|
||||||
|
template <typename T>
|
||||||
|
ResultError(T&& message, status_t s) : val_(s), message_(std::forward<T>(message)) {
|
||||||
|
LOG_FATAL_IF(s == OK, "Result error should not hold success");
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultError(status_t s) : val_(s) {}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
operator expected<T, ResultError<StatusT, true>>() const {
|
||||||
|
return unexpected(*this);
|
||||||
|
}
|
||||||
|
|
||||||
|
status_t code() const { return val_; }
|
||||||
|
|
||||||
|
std::string message() const { return statusToString(val_) + message_; }
|
||||||
|
private:
|
||||||
|
const status_t val_;
|
||||||
|
std::string message_;
|
||||||
|
};
|
||||||
|
|
||||||
// Specialization of android::base::OkOrFail<V> for V = status_t. This is used to use the OR_RETURN
|
// Specialization of android::base::OkOrFail<V> for V = status_t. This is used to use the OR_RETURN
|
||||||
// and OR_FATAL macros with statements that yields a value of status_t. See android-base/errors.h
|
// and OR_FATAL macros with statements that yields a value of status_t. See android-base/errors.h
|
||||||
// for the detailed contract.
|
// for the detailed contract.
|
||||||
template <>
|
template <>
|
||||||
struct OkOrFail<status_t> {
|
struct OkOrFail<status_t> {
|
||||||
|
static_assert(std::is_same_v<status_t, int>);
|
||||||
// Tests if status_t is a success value of not.
|
// Tests if status_t is a success value of not.
|
||||||
static bool IsOk(const status_t& s) { return s == OK; }
|
static bool IsOk(const status_t& s) { return s == OK; }
|
||||||
|
|
||||||
|
@ -71,16 +117,70 @@ struct OkOrFail<status_t> {
|
||||||
|
|
||||||
// Or converts into Result<T, StatusT>. This is used when OR_RETURN is used in a function whose
|
// Or converts into Result<T, StatusT>. This is used when OR_RETURN is used in a function whose
|
||||||
// return type is Result<T, StatusT>.
|
// return type is Result<T, StatusT>.
|
||||||
template <typename T, typename = std::enable_if_t<!std::is_same_v<T, status_t>>>
|
|
||||||
|
template <typename T>
|
||||||
operator Result<T, StatusT>() && {
|
operator Result<T, StatusT>() && {
|
||||||
return Error<StatusT>(std::move(val_));
|
return ResultError<StatusT>(std::move(val_));
|
||||||
}
|
}
|
||||||
|
|
||||||
operator Result<int, StatusT>() && { return Error<StatusT>(std::move(val_)); }
|
template<typename T>
|
||||||
|
operator Result<T, StatusT, false>() && {
|
||||||
|
return ResultError<StatusT, false>(std::move(val_));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Since user defined conversion can be followed by numeric conversion,
|
||||||
|
// we have to specialize all conversions to results holding numeric types to
|
||||||
|
// avoid conversion ambiguities with the constructor of expected.
|
||||||
|
#pragma push_macro("SPECIALIZED_CONVERSION")
|
||||||
|
#define SPECIALIZED_CONVERSION(type)\
|
||||||
|
operator Result<type, StatusT>() && { return ResultError<StatusT>(std::move(val_)); }\
|
||||||
|
operator Result<type, StatusT, false>() && { return ResultError<StatusT, false>(std::move(val_));}
|
||||||
|
|
||||||
|
SPECIALIZED_CONVERSION(int)
|
||||||
|
SPECIALIZED_CONVERSION(short int)
|
||||||
|
SPECIALIZED_CONVERSION(unsigned short int)
|
||||||
|
SPECIALIZED_CONVERSION(unsigned int)
|
||||||
|
SPECIALIZED_CONVERSION(long int)
|
||||||
|
SPECIALIZED_CONVERSION(unsigned long int)
|
||||||
|
SPECIALIZED_CONVERSION(long long int)
|
||||||
|
SPECIALIZED_CONVERSION(unsigned long long int)
|
||||||
|
SPECIALIZED_CONVERSION(bool)
|
||||||
|
SPECIALIZED_CONVERSION(char)
|
||||||
|
SPECIALIZED_CONVERSION(unsigned char)
|
||||||
|
SPECIALIZED_CONVERSION(signed char)
|
||||||
|
SPECIALIZED_CONVERSION(wchar_t)
|
||||||
|
SPECIALIZED_CONVERSION(char16_t)
|
||||||
|
SPECIALIZED_CONVERSION(char32_t)
|
||||||
|
SPECIALIZED_CONVERSION(float)
|
||||||
|
SPECIALIZED_CONVERSION(double)
|
||||||
|
SPECIALIZED_CONVERSION(long double)
|
||||||
|
#undef SPECIALIZED_CONVERSION
|
||||||
|
#pragma pop_macro("SPECIALIZED_CONVERSION")
|
||||||
// String representation of the error value.
|
// String representation of the error value.
|
||||||
static std::string ErrorMessage(const status_t& s) { return statusToString(s); }
|
static std::string ErrorMessage(const status_t& s) { return statusToString(s); }
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace base
|
} // namespace base
|
||||||
|
|
||||||
|
|
||||||
|
// These conversions make StatusT directly comparable to status_t in order to
|
||||||
|
// avoid calling code whenever comparisons are desired.
|
||||||
|
|
||||||
|
template <bool include_message>
|
||||||
|
bool operator==(const base::ResultError<StatusT, include_message>& l, const status_t& r) {
|
||||||
|
return (l.code() == r);
|
||||||
|
}
|
||||||
|
template <bool include_message>
|
||||||
|
bool operator==(const status_t& l, const base::ResultError<StatusT, include_message>& r) {
|
||||||
|
return (l == r.code());
|
||||||
|
}
|
||||||
|
|
||||||
|
template <bool include_message>
|
||||||
|
bool operator!=(const base::ResultError<StatusT, include_message>& l, const status_t& r) {
|
||||||
|
return (l.code() != r);
|
||||||
|
}
|
||||||
|
template <bool include_message>
|
||||||
|
bool operator!=(const status_t& l, const base::ResultError<StatusT, include_message>& r) {
|
||||||
|
return (l != r.code());
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace android
|
} // namespace android
|
||||||
|
|
Loading…
Reference in New Issue