// Copyright 2018 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_TORQUE_PARAMETER_DIFFERENCE_H_ #define V8_TORQUE_PARAMETER_DIFFERENCE_H_ #include <vector> #include "src/torque/types.h" namespace v8 { namespace internal { namespace torque { class ParameterDifference { public: ParameterDifference(const TypeVector& to, const TypeVector& from) { DCHECK_EQ(to.size(), from.size()); for (size_t i = 0; i < to.size(); ++i) { AddParameter(to[i], from[i]); } } // An overload is selected if it is strictly better than all alternatives. // This means that it has to be strictly better in at least one parameter, // and better or equally good in all others. // // When comparing a pair of corresponding parameters of two overloads... // ... they are considered equally good if: // - They are equal. // - Both require some implicit conversion. // ... one is considered better if: // - It is a strict subtype of the other. // - It doesn't require an implicit conversion, while the other does. bool StrictlyBetterThan(const ParameterDifference& other) const { DCHECK_EQ(difference_.size(), other.difference_.size()); bool better_parameter_found = false; for (size_t i = 0; i < difference_.size(); ++i) { base::Optional<const Type*> a = difference_[i]; base::Optional<const Type*> b = other.difference_[i]; if (a == b) { continue; } else if (a && b && a != b && (*a)->IsSubtypeOf(*b)) { DCHECK(!(*b)->IsSubtypeOf(*a)); better_parameter_found = true; } else if (a && !b) { better_parameter_found = true; } else { return false; } } return better_parameter_found; } private: // Pointwise difference between call arguments and a signature. // {base::nullopt} means that an implicit conversion was necessary, // otherwise we store the supertype found in the signature. std::vector<base::Optional<const Type*>> difference_; void AddParameter(const Type* to, const Type* from) { if (from->IsSubtypeOf(to)) { difference_.push_back(to); } else if (IsAssignableFrom(to, from)) { difference_.push_back(base::nullopt); } else { UNREACHABLE(); } } }; } // namespace torque } // namespace internal } // namespace v8 #endif // V8_TORQUE_PARAMETER_DIFFERENCE_H_