diff --git a/clang/lib/Analysis/FlowSensitive/Models/UncheckedOptionalAccessModel.cpp b/clang/lib/Analysis/FlowSensitive/Models/UncheckedOptionalAccessModel.cpp --- a/clang/lib/Analysis/FlowSensitive/Models/UncheckedOptionalAccessModel.cpp +++ b/clang/lib/Analysis/FlowSensitive/Models/UncheckedOptionalAccessModel.cpp @@ -78,6 +78,22 @@ argumentCountIs(1), hasArgument(0, unless(hasNulloptType()))); } +auto isOptionalValueOrConversionAssignment() { + return cxxOperatorCallExpr( + hasOverloadedOperatorName("="), + callee(cxxMethodDecl(ofClass(optionalClass()))), + unless(hasDeclaration(cxxMethodDecl( + anyOf(isCopyAssignmentOperator(), isMoveAssignmentOperator())))), + argumentCountIs(2), hasArgument(1, unless(hasNulloptType()))); +} + +auto isOptionalNulloptAssignment() { + return cxxOperatorCallExpr(hasOverloadedOperatorName("="), + callee(cxxMethodDecl(ofClass(optionalClass()))), + argumentCountIs(2), + hasArgument(1, hasNulloptType())); +} + /// Creates a symbolic value for an `optional` value using `HasValueVal` as the /// symbolic value of its "has_value" property. StructValue &createOptionalValue(Environment &Env, BoolValue &HasValueVal) { @@ -140,8 +156,8 @@ void transferUnwrapCall(const Expr *UnwrapExpr, const Expr *ObjectExpr, LatticeTransferState &State) { - if (auto *OptionalVal = cast_or_null( - State.Env.getValue(*ObjectExpr, SkipPast::ReferenceThenPointer))) { + if (auto *OptionalVal = cast_or_null(State.Env.getValue( + *ObjectExpr->IgnoreParens(), SkipPast::ReferenceThenPointer))) { auto *HasValueVal = getHasValue(OptionalVal); assert(HasValueVal != nullptr); @@ -213,6 +229,41 @@ assignOptionalValue(*E, State, *HasValueVal); } +void transferValueOrConversionAssignment( + const CXXOperatorCallExpr *E, const MatchFinder::MatchResult &MatchRes, + LatticeTransferState &State) { + assert(E->getDirectCallee()->getTemplateSpecializationArgs()->size() > 0); + assert(E->getNumArgs() > 1); + + auto *OptionalLoc = + State.Env.getStorageLocation(*E->getArg(0), SkipPast::Reference); + assert(OptionalLoc != nullptr); + + const int TemplateParamOptionalWrappersCount = countOptionalWrappers( + *MatchRes.Context, stripReference(E->getDirectCallee() + ->getTemplateSpecializationArgs() + ->get(0) + .getAsType())); + const int ArgTypeOptionalWrappersCount = countOptionalWrappers( + *MatchRes.Context, stripReference(E->getArg(1)->getType())); + auto *HasValueVal = + (TemplateParamOptionalWrappersCount == ArgTypeOptionalWrappersCount) + // This is a constructor call for optional with argument of type U + // such that T is constructible from U. + ? &State.Env.getBoolLiteralValue(true) + // This is a constructor call for optional with argument of type + // optional such that T is constructible from U. + : getHasValue(State.Env.getValue(*E->getArg(1), SkipPast::Reference)); + if (HasValueVal == nullptr) + HasValueVal = &State.Env.makeAtomicBoolValue(); + + State.Env.setValue(*OptionalLoc, + createOptionalValue(State.Env, *HasValueVal)); + + // Assign a storage location for the whole expression. + State.Env.setStorageLocation(*E, *OptionalLoc); +} + auto buildTransferMatchSwitch() { return MatchSwitchBuilder() // Attach a symbolic "has_value" state to optional values that we see for @@ -223,7 +274,7 @@ // make_optional .CaseOf(isMakeOptionalCall(), transferMakeOptionalCall) - // constructors: + // optional::optional .CaseOf( isOptionalInPlaceConstructor(), [](const CXXConstructExpr *E, const MatchFinder::MatchResult &, @@ -240,6 +291,26 @@ .CaseOf(isOptionalValueOrConversionConstructor(), transferValueOrConversionConstructor) + // optional::operator= + .CaseOf(isOptionalValueOrConversionAssignment(), + transferValueOrConversionAssignment) + .CaseOf( + isOptionalNulloptAssignment(), + [](const CXXOperatorCallExpr *E, const MatchFinder::MatchResult &, + LatticeTransferState &State) { + auto *OptionalLoc = State.Env.getStorageLocation( + *E->getArg(0), SkipPast::Reference); + assert(OptionalLoc != nullptr); + + State.Env.setValue( + *OptionalLoc, + createOptionalValue(State.Env, + State.Env.getBoolLiteralValue(false))); + + // Assign a storage location for the whole expression. + State.Env.setStorageLocation(*E, *OptionalLoc); + }) + // optional::value .CaseOf( isOptionalMemberCallWithName("value"), diff --git a/clang/unittests/Analysis/FlowSensitive/UncheckedOptionalAccessModelTest.cpp b/clang/unittests/Analysis/FlowSensitive/UncheckedOptionalAccessModelTest.cpp --- a/clang/unittests/Analysis/FlowSensitive/UncheckedOptionalAccessModelTest.cpp +++ b/clang/unittests/Analysis/FlowSensitive/UncheckedOptionalAccessModelTest.cpp @@ -30,13 +30,28 @@ using ::testing::UnorderedElementsAre; // FIXME: Move header definitions in separate file(s). +static constexpr char CSDtdDefHeader[] = R"( +#ifndef CSTDDEF_H +#define CSTDDEF_H + +namespace std { + +typedef decltype(sizeof(char)) size_t; + +using nullptr_t = decltype(nullptr); + +} // namespace std + +#endif // CSTDDEF_H +)"; + static constexpr char StdTypeTraitsHeader[] = R"( #ifndef STD_TYPE_TRAITS_H #define STD_TYPE_TRAITS_H -namespace std { +#include "cstddef.h" -typedef decltype(sizeof(char)) size_t; +namespace std { template struct integral_constant { @@ -287,6 +302,9 @@ template using _And = decltype(__and_helper<_Pred...>(0)); +template +struct _Not : _BoolConstant {}; + struct __check_tuple_constructor_fail { static constexpr bool __enable_explicit_default() { return false; } static constexpr bool __enable_implicit_default() { return false; } @@ -300,6 +318,150 @@ } }; +template +struct __select_2nd { + typedef _Tp type; +}; +template +typename __select_2nd() = declval<_Arg>())), + true_type>::type +__is_assignable_test(int); +template +false_type __is_assignable_test(...); +template ::value || is_void<_Arg>::value> +struct __is_assignable_imp + : public decltype((__is_assignable_test<_Tp, _Arg>(0))) {}; +template +struct __is_assignable_imp<_Tp, _Arg, true> : public false_type {}; +template +struct is_assignable : public __is_assignable_imp<_Tp, _Arg> {}; + +template +struct __libcpp_is_integral : public false_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; // NOLINT +template <> +struct __libcpp_is_integral : public true_type {}; // NOLINT +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; +template <> +struct __libcpp_is_integral : public true_type {}; // NOLINT +template <> +struct __libcpp_is_integral : public true_type {}; // NOLINT +template <> +struct __libcpp_is_integral : public true_type {}; // NOLINT +template <> // NOLINTNEXTLINE +struct __libcpp_is_integral : public true_type {}; +template +struct is_integral + : public __libcpp_is_integral::type> {}; + +template +struct __libcpp_is_floating_point : public false_type {}; +template <> +struct __libcpp_is_floating_point : public true_type {}; +template <> +struct __libcpp_is_floating_point : public true_type {}; +template <> +struct __libcpp_is_floating_point : public true_type {}; +template +struct is_floating_point + : public __libcpp_is_floating_point::type> {}; + +template +struct is_arithmetic + : public integral_constant::value || + is_floating_point<_Tp>::value> {}; + +template +struct __libcpp_is_pointer : public false_type {}; +template +struct __libcpp_is_pointer<_Tp*> : public true_type {}; +template +struct is_pointer : public __libcpp_is_pointer::type> { +}; + +template +struct __libcpp_is_member_pointer : public false_type {}; +template +struct __libcpp_is_member_pointer<_Tp _Up::*> : public true_type {}; +template +struct is_member_pointer + : public __libcpp_is_member_pointer::type> {}; + +template +struct __libcpp_union : public false_type {}; +template +struct is_union : public __libcpp_union::type> {}; + +template +struct is_reference : false_type {}; +template +struct is_reference : true_type {}; +template +struct is_reference : true_type {}; + +template +inline constexpr bool is_reference_v = is_reference::value; + +struct __two { + char __lx[2]; +}; + +namespace __is_class_imp { +template +char __test(int _Tp::*); +template +__two __test(...); +} // namespace __is_class_imp +template +struct is_class + : public integral_constant(0)) == 1 && + !is_union<_Tp>::value> {}; + +template +struct __is_nullptr_t_impl : public false_type {}; +template <> +struct __is_nullptr_t_impl : public true_type {}; +template +struct __is_nullptr_t + : public __is_nullptr_t_impl::type> {}; +template +struct is_null_pointer + : public __is_nullptr_t_impl::type> {}; + +template +struct is_enum + : public integral_constant< + bool, !is_void<_Tp>::value && !is_integral<_Tp>::value && + !is_floating_point<_Tp>::value && !is_array<_Tp>::value && + !is_pointer<_Tp>::value && !is_reference<_Tp>::value && + !is_member_pointer<_Tp>::value && !is_union<_Tp>::value && + !is_class<_Tp>::value && !is_function<_Tp>::value> {}; + +template +struct is_scalar + : public integral_constant< + bool, is_arithmetic<_Tp>::value || is_member_pointer<_Tp>::value || + is_pointer<_Tp>::value || __is_nullptr_t<_Tp>::value || + is_enum<_Tp>::value> {}; +template <> +struct is_scalar : public true_type {}; + } // namespace std #endif // STD_TYPE_TRAITS_H @@ -442,6 +604,18 @@ _CheckOptionalLikeConstructor<_QualUp>, __check_tuple_constructor_fail>; + + template + using _CheckOptionalLikeAssign = _If< + _And< + _IsNotSame<_Up, _Tp>, + is_constructible<_Tp, _QualUp>, + is_assignable<_Tp&, _QualUp> + >::value, + _CheckOptionalLikeConstructor<_QualUp>, + __check_tuple_constructor_fail + >; + public: constexpr optional() noexcept {} constexpr optional(const optional&) = default; @@ -492,6 +666,30 @@ int> = 0> constexpr explicit optional(optional<_Up>&& __v); + constexpr optional& operator=(nullopt_t) noexcept; + + optional& operator=(const optional&); + + optional& operator=(optional&&); + + template , optional>, + _Or<_IsNotSame<__uncvref_t<_Up>, value_type>, + _Not>>, + is_constructible, + is_assignable>::value>> + constexpr optional& operator=(_Up&& __v); + + template :: + template __enable_assign<_Up>(), + int> = 0> + constexpr optional& operator=(const optional<_Up>& __v); + + template :: + template __enable_assign<_Up>(), + int> = 0> + constexpr optional& operator=(optional<_Up>&& __v); + const _Tp& operator*() const&; _Tp& operator*() &; const _Tp&& operator*() const&&; @@ -556,7 +754,6 @@ namespace optional_internal { -// Whether T is constructible or convertible from optional. template struct is_constructible_convertible_from_optional : std::integral_constant< @@ -569,6 +766,15 @@ std::is_convertible&, T>::value || std::is_convertible&&, T>::value> {}; +template +struct is_constructible_convertible_assignable_from_optional + : std::integral_constant< + bool, is_constructible_convertible_from_optional::value || + std::is_assignable&>::value || + std::is_assignable&&>::value || + std::is_assignable&>::value || + std::is_assignable&&>::value> {}; + } // namespace optional_internal template @@ -666,6 +872,44 @@ bool>::type = false> explicit optional(optional&& rhs); + optional& operator=(nullopt_t) noexcept; + + optional& operator=(const optional& src); + + optional& operator=(optional&& src); + + template < + typename U = T, + typename = typename std::enable_if, typename std::decay::type>>, + absl::negation< + absl::conjunction, + std::is_same::type>>>, + std::is_constructible, std::is_assignable>::value>::type> + optional& operator=(U&& v); + + template < + typename U, + typename = typename std::enable_if>, + std::is_constructible, std::is_assignable, + absl::negation< + optional_internal:: + is_constructible_convertible_assignable_from_optional< + T, U>>>::value>::type> + optional& operator=(const optional& rhs); + + template >, std::is_constructible, + std::is_assignable, + absl::negation< + optional_internal:: + is_constructible_convertible_assignable_from_optional< + T, U>>>::value>::type> + optional& operator=(optional&& rhs); + const T& operator*() const&; T& operator*() &; const T&& operator*() const&&; @@ -744,6 +988,15 @@ std::is_convertible&&, T>::value || std::is_convertible&&, T>::value> {}; +template +struct IsAssignableFromOptional + : std::integral_constant< + bool, IsConvertibleFromOptional::value || + std::is_assignable&>::value || + std::is_assignable&>::value || + std::is_assignable&&>::value || + std::is_assignable&&>::value> {}; + } // namespace internal template @@ -818,6 +1071,36 @@ bool>::type = false> constexpr explicit Optional(U&& value); + Optional& operator=(const Optional& other) noexcept; + + Optional& operator=(Optional&& other) noexcept; + + Optional& operator=(nullopt_t); + + template + typename std::enable_if< + !std::is_same, Optional>::value && + std::is_constructible::value && + std::is_assignable::value && + (!std::is_scalar::value || + !std::is_same::type, T>::value), + Optional&>::type + operator=(U&& value) noexcept; + + template + typename std::enable_if::value && + std::is_constructible::value && + std::is_assignable::value, + Optional&>::type + operator=(const Optional& other) noexcept; + + template + typename std::enable_if::value && + std::is_constructible::value && + std::is_assignable::value, + Optional&>::type + operator=(Optional&& other) noexcept; + const T& operator*() const&; T& operator*() &; const T&& operator*() const&&; @@ -904,6 +1187,7 @@ ReplaceAllOccurrences(SourceCode, "$optional", GetParam().TypeName); std::vector> Headers; + Headers.emplace_back("cstddef.h", CSDtdDefHeader); Headers.emplace_back("std_initializer_list.h", StdInitializerListHeader); Headers.emplace_back("std_type_traits.h", StdTypeTraitsHeader); Headers.emplace_back("std_utility.h", StdUtilityHeader); @@ -1475,9 +1759,157 @@ // FIXME: Add tests that call `reset` in conditional branches. } +TEST_P(UncheckedOptionalAccessTest, ValueAssignment) { + ExpectLatticeChecksFor(R"( + #include "unchecked_optional_access_test.h" + + struct Foo {}; + + void target() { + $ns::$optional opt; + opt = Foo(); + opt.value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "safe"))); + + ExpectLatticeChecksFor(R"( + #include "unchecked_optional_access_test.h" + + struct Foo {}; + + void target() { + $ns::$optional opt; + (opt = Foo()).value(); + (void)0; + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "safe"))); + + ExpectLatticeChecksFor(R"( + #include "unchecked_optional_access_test.h" + + struct MyString { + MyString(const char*); + }; + + void target() { + $ns::$optional opt; + opt = "foo"; + opt.value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "safe"))); + + ExpectLatticeChecksFor(R"( + #include "unchecked_optional_access_test.h" + + struct MyString { + MyString(const char*); + }; + + void target() { + $ns::$optional opt; + (opt = "foo").value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "safe"))); +} + +TEST_P(UncheckedOptionalAccessTest, OptionalConversionAssignment) { + ExpectLatticeChecksFor( + R"( + #include "unchecked_optional_access_test.h" + + struct Foo {}; + + struct Bar { + Bar(const Foo&); + }; + + void target() { + $ns::$optional opt; + opt = Make<$ns::$optional>(); + opt.value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "unsafe: input.cc:13:7"))); + + ExpectLatticeChecksFor( + R"( + #include "unchecked_optional_access_test.h" + + struct Foo {}; + + struct Bar { + Bar(const Foo&); + }; + + void target() { + $ns::$optional opt1 = $ns::nullopt; + $ns::$optional opt2; + opt2 = opt1; + opt2.value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "unsafe: input.cc:14:7"))); + + ExpectLatticeChecksFor(R"( + #include "unchecked_optional_access_test.h" + + struct Foo {}; + + struct Bar { + Bar(const Foo&); + }; + + void target() { + $ns::$optional opt1 = Foo(); + $ns::$optional opt2; + opt2 = opt1; + opt2.value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "safe"))); +} + +TEST_P(UncheckedOptionalAccessTest, NulloptAssignment) { + ExpectLatticeChecksFor( + R"( + #include "unchecked_optional_access_test.h" + + void target() { + $ns::$optional opt = 3; + opt = $ns::nullopt; + opt.value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "unsafe: input.cc:7:7"))); + + ExpectLatticeChecksFor( + R"( + #include "unchecked_optional_access_test.h" + + void target() { + $ns::$optional opt = 3; + (opt = $ns::nullopt).value(); + /*[[check]]*/ + } + )", + UnorderedElementsAre(Pair("check", "unsafe: input.cc:6:7"))); +} + // FIXME: Add support for: // - constructors (copy, move) -// - assignment operators (default, copy, move, non-standard) +// - assignment operators (default, copy, move) // - swap // - invalidation (passing optional by non-const reference/pointer) // - `value_or(nullptr) != nullptr`, `value_or(0) != 0`, `value_or("").empty()`