[torque] use implicit parameters for template parameter inference

This allows templates to preserve the type of implicit parameters to
select a better ovleroad, without generally extending overload
resolution to implicit parameters, which could be confusing.

Bug: v8:7793
Change-Id: Ie57090a295b0b46d03789829b975fc16e2a9c5b9
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2329630
Commit-Queue: Tobias Tebbi <tebbi@chromium.org>
Reviewed-by: Seth Brenith <seth.brenith@microsoft.com>
Cr-Commit-Position: refs/heads/master@{#69177}
This commit is contained in:
Tobias Tebbi 2020-07-31 16:35:04 +02:00 committed by Commit Bot
parent 09db7cf9a5
commit 50d5c2bb07
10 changed files with 107 additions and 28 deletions

View File

@ -2,10 +2,12 @@
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "src/torque/declarable.h"
#include <fstream>
#include <iostream>
#include "src/torque/declarable.h"
#include "src/torque/ast.h"
#include "src/torque/global-context.h"
#include "src/torque/type-inference.h"
#include "src/torque/type-visitor.h"
@ -134,17 +136,13 @@ std::vector<TypeConstraint> ComputeConstraints(
TypeArgumentInference GenericCallable::InferSpecializationTypes(
const TypeVector& explicit_specialization_types,
const TypeVector& arguments) {
size_t implicit_count = declaration()->parameters.implicit_count;
const std::vector<base::Optional<const Type*>>& arguments) {
const std::vector<TypeExpression*>& parameters =
declaration()->parameters.types;
std::vector<TypeExpression*> explicit_parameters(
parameters.begin() + implicit_count, parameters.end());
CurrentScope::Scope generic_scope(ParentScope());
TypeArgumentInference inference(generic_parameters(),
explicit_specialization_types,
explicit_parameters, arguments);
explicit_specialization_types, parameters,
arguments);
if (!inference.HasFailed()) {
if (auto violation =
FindConstraintViolation(inference.GetResult(), Constraints())) {

View File

@ -567,7 +567,7 @@ class GenericCallable
TypeArgumentInference InferSpecializationTypes(
const TypeVector& explicit_specialization_types,
const TypeVector& arguments);
const std::vector<base::Optional<const Type*>>& arguments);
private:
friend class Declarations;

View File

@ -196,9 +196,10 @@ void DeclarationVisitor::Visit(SpecializationDeclaration* decl) {
// This argument inference is just to trigger constraint checking on the
// generic arguments.
TypeArgumentInference inference = generic->InferSpecializationTypes(
TypeVisitor::ComputeTypeVector(decl->generic_parameters),
signature_with_types.GetExplicitTypes());
if (inference.HasFailed()) continue;
TypeVisitor::ComputeTypeVector(decl->generic_parameters), {});
if (inference.HasFailed()) {
continue;
}
Signature generic_signature_with_types =
MakeSpecializedSignature(SpecializationKey<GenericCallable>{
generic, TypeVisitor::ComputeTypeVector(decl->generic_parameters)});

View File

@ -1786,6 +1786,25 @@ bool ImplementationVisitor::TestLookupCallable(
{}, {}, true) != nullptr;
}
TypeArgumentInference ImplementationVisitor::InferSpecializationTypes(
GenericCallable* generic, const TypeVector& explicit_specialization_types,
const TypeVector& explicit_arguments) {
std::vector<base::Optional<const Type*>> all_arguments;
const ParameterList& parameters = generic->declaration()->parameters;
for (size_t i = 0; i < parameters.implicit_count; ++i) {
base::Optional<Binding<LocalValue>*> val =
TryLookupLocalValue(parameters.names[i]->value);
all_arguments.push_back(
val ? (*val)->GetLocationReference(*val).ReferencedType()
: base::nullopt);
}
for (const Type* explicit_argument : explicit_arguments) {
all_arguments.push_back(explicit_argument);
}
return generic->InferSpecializationTypes(explicit_specialization_types,
all_arguments);
}
template <class Container>
Callable* ImplementationVisitor::LookupCallable(
const QualifiedName& name, const Container& declaration_container,
@ -1799,8 +1818,8 @@ Callable* ImplementationVisitor::LookupCallable(
std::vector<std::pair<GenericCallable*, std::string>> inapplicable_generics;
for (auto* declarable : declaration_container) {
if (GenericCallable* generic = GenericCallable::DynamicCast(declarable)) {
TypeArgumentInference inference = generic->InferSpecializationTypes(
specialization_types, parameter_types);
TypeArgumentInference inference = InferSpecializationTypes(
generic, specialization_types, parameter_types);
if (inference.HasFailed()) {
inapplicable_generics.push_back(
std::make_pair(generic, inference.GetFailureReason()));
@ -1861,8 +1880,8 @@ Callable* ImplementationVisitor::LookupCallable(
if (GenericCallable* generic =
GenericCallable::DynamicCast(overloads[best])) {
TypeArgumentInference inference = generic->InferSpecializationTypes(
specialization_types, parameter_types);
TypeArgumentInference inference = InferSpecializationTypes(
generic, specialization_types, parameter_types);
result = GetOrCreateSpecialization(
SpecializationKey<GenericCallable>{generic, inference.GetResult()});
} else {

View File

@ -682,6 +682,10 @@ class ImplementationVisitor {
const Arguments& arguments,
const TypeVector& specialization_types);
TypeArgumentInference InferSpecializationTypes(
GenericCallable* generic, const TypeVector& explicit_specialization_types,
const TypeVector& explicit_arguments);
const Type* GetCommonType(const Type* left, const Type* right);
VisitResult GenerateCopy(const VisitResult& to_copy);

View File

@ -12,7 +12,7 @@ TypeArgumentInference::TypeArgumentInference(
const GenericParameters& type_parameters,
const TypeVector& explicit_type_arguments,
const std::vector<TypeExpression*>& term_parameters,
const TypeVector& term_argument_types)
const std::vector<base::Optional<const Type*>>& term_argument_types)
: num_explicit_(explicit_type_arguments.size()),
type_parameter_from_name_(type_parameters.size()),
inferred_(type_parameters.size()) {
@ -20,8 +20,8 @@ TypeArgumentInference::TypeArgumentInference(
Fail("more explicit type arguments than expected");
return;
}
if (term_parameters.size() != term_argument_types.size()) {
Fail("number of term parameters does not match number of term arguments!");
if (term_argument_types.size() > term_parameters.size()) {
Fail("more arguments than expected");
return;
}
@ -32,8 +32,9 @@ TypeArgumentInference::TypeArgumentInference(
inferred_[i] = {explicit_type_arguments[i]};
}
for (size_t i = 0; i < term_parameters.size(); i++) {
Match(term_parameters[i], term_argument_types[i]);
for (size_t i = 0; i < term_argument_types.size(); i++) {
if (term_argument_types[i])
Match(term_parameters[i], *term_argument_types[i]);
if (HasFailed()) return;
}

View File

@ -55,10 +55,11 @@ namespace torque {
// should be instantiated in the appropriate scope.
class TypeArgumentInference {
public:
TypeArgumentInference(const GenericParameters& type_parameters,
const TypeVector& explicit_type_arguments,
const std::vector<TypeExpression*>& term_parameters,
const TypeVector& term_argument_types);
TypeArgumentInference(
const GenericParameters& type_parameters,
const TypeVector& explicit_type_arguments,
const std::vector<TypeExpression*>& term_parameters,
const std::vector<base::Optional<const Type*>>& term_argument_types);
bool HasFailed() const { return failure_reason_.has_value(); }
const std::string& GetFailureReason() { return *failure_reason_; }

View File

@ -500,9 +500,10 @@ const Type* TypeVisitor::ComputeTypeForStructExpression(
}
CurrentScope::Scope generic_scope(generic_type->ParentScope());
TypeArgumentInference inference(generic_type->generic_parameters(),
explicit_type_arguments, term_parameters,
term_argument_types);
TypeArgumentInference inference(
generic_type->generic_parameters(), explicit_type_arguments,
term_parameters,
TransformVector<base::Optional<const Type*>>(term_argument_types));
if (inference.HasFailed()) {
ReportError("failed to infer type arguments for struct ", basic->name,

View File

@ -5,6 +5,7 @@
#ifndef V8_TORQUE_UTILS_H_
#define V8_TORQUE_UTILS_H_
#include <algorithm>
#include <ostream>
#include <queue>
#include <streambuf>
@ -523,6 +524,17 @@ class Worklist {
std::unordered_set<T> contained_;
};
template <class T, class U, class F>
std::vector<T> TransformVector(const std::vector<U>& v, F f) {
std::vector<T> result;
std::transform(v.begin(), v.end(), std::back_inserter(result), f);
return result;
}
template <class T, class U>
std::vector<T> TransformVector(const std::vector<U>& v) {
return TransformVector<T>(v, [](const U& x) -> T { return x; });
}
} // namespace torque
} // namespace internal
} // namespace v8

View File

@ -915,6 +915,48 @@ TEST(Torque, UnusedImplicit) {
"Test6(Smi): Smi\n Test6(implicit Smi)(Smi): Smi"));
}
TEST(Torque, ImplicitTemplateParameterInference) {
ExpectSuccessfulCompilation(R"(
macro Foo(_x: Map) {}
macro Foo(_x: Smi) {}
macro GenericMacro<T: type>(implicit x: T)() {
Foo(x);
}
@export
macro Test1(implicit x: Smi)() { GenericMacro(); }
@export
macro Test2(implicit x: Map)() { GenericMacro(); }
)");
ExpectFailingCompilation(
R"(
// Wrap in namespace to avoid redeclaration error.
namespace foo {
macro Foo(implicit x: Map)() {}
}
macro Foo(implicit x: Smi)() {}
namespace foo{
@export
macro Test(implicit x: Smi)() { Foo(); }
}
)",
HasSubstr("ambiguous callable"));
ExpectFailingCompilation(
R"(
// Wrap in namespace to avoid redeclaration error.
namespace foo {
macro Foo(implicit x: Map)() {}
}
macro Foo(implicit x: Smi)() {}
namespace foo{
@export
macro Test(implicit x: Map)() { Foo(); }
}
)",
HasSubstr("ambiguous callable"));
}
} // namespace torque
} // namespace internal
} // namespace v8