mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
When opening generic types with type parameter requirements, add information about requirement location to the locator of each generated constraint to make it easier to extract such information if needed.
256 lines
6.0 KiB
C++
256 lines
6.0 KiB
C++
//===--- ConstraintLocator.cpp - Constraint Locator -----------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements the \c ConstraintLocator class and its related types,
|
|
// which is used by the constraint-based type checker to describe how
|
|
// a particular constraint was derived.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
#include "ConstraintLocator.h"
|
|
#include "ConstraintSystem.h"
|
|
#include "swift/AST/Decl.h"
|
|
#include "swift/AST/Expr.h"
|
|
#include "swift/AST/Types.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
using namespace swift;
|
|
using namespace constraints;
|
|
|
|
void ConstraintLocator::Profile(llvm::FoldingSetNodeID &id, Expr *anchor,
|
|
ArrayRef<PathElement> path) {
|
|
id.AddPointer(anchor);
|
|
id.AddInteger(path.size());
|
|
for (auto elt : path) {
|
|
id.AddInteger(elt.getKind());
|
|
switch (elt.getKind()) {
|
|
case Archetype:
|
|
id.AddPointer(elt.getArchetype()->getCanonicalType().getPointer());
|
|
break;
|
|
|
|
case Requirement:
|
|
id.AddPointer(elt.getRequirement());
|
|
break;
|
|
|
|
case Witness:
|
|
id.AddPointer(elt.getWitness());
|
|
break;
|
|
|
|
case AssociatedType:
|
|
id.AddPointer(elt.getAssociatedType());
|
|
break;
|
|
|
|
case ApplyArgument:
|
|
case ApplyFunction:
|
|
case FunctionArgument:
|
|
case FunctionResult:
|
|
case OptionalPayload:
|
|
case Member:
|
|
case MemberRefBase:
|
|
case UnresolvedMember:
|
|
case SubscriptIndex:
|
|
case SubscriptMember:
|
|
case SubscriptResult:
|
|
case ConstructorMember:
|
|
case RvalueAdjustment:
|
|
case ClosureResult:
|
|
case ParentType:
|
|
case InstanceType:
|
|
case SequenceIteratorProtocol:
|
|
case GeneratorElementType:
|
|
case ArrayElementType:
|
|
case ScalarToTuple:
|
|
case Load:
|
|
case GenericArgument:
|
|
case NamedTupleElement:
|
|
case TupleElement:
|
|
case ApplyArgToParam:
|
|
case OpenedGeneric:
|
|
case KeyPathComponent:
|
|
case ConditionalRequirement:
|
|
case TypeParameterRequirement:
|
|
if (unsigned numValues = numNumericValuesInPathElement(elt.getKind())) {
|
|
id.AddInteger(elt.getValue());
|
|
if (numValues > 1)
|
|
id.AddInteger(elt.getValue2());
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConstraintLocator::dump(SourceManager *sm) {
|
|
dump(sm, llvm::errs());
|
|
llvm::errs() << "\n";
|
|
}
|
|
|
|
void ConstraintLocator::dump(ConstraintSystem *CS) {
|
|
dump(&CS->TC.Context.SourceMgr, llvm::errs());
|
|
llvm::errs() << "\n";
|
|
}
|
|
|
|
|
|
void ConstraintLocator::dump(SourceManager *sm, raw_ostream &out) {
|
|
out << "locator@" << (void*) this << " [";
|
|
|
|
if (anchor) {
|
|
out << Expr::getKindName(anchor->getKind());
|
|
if (sm) {
|
|
out << '@';
|
|
anchor->getLoc().print(out, *sm);
|
|
}
|
|
}
|
|
|
|
for (auto elt : getPath()) {
|
|
out << " -> ";
|
|
switch (elt.getKind()) {
|
|
case ArrayElementType:
|
|
out << "array element";
|
|
break;
|
|
|
|
case Archetype:
|
|
out << "archetype '" << elt.getArchetype()->getString() << "'";
|
|
break;
|
|
|
|
case AssociatedType:
|
|
out << "associated type '"
|
|
<< elt.getAssociatedType()->getNameStr() << "'";
|
|
break;
|
|
|
|
case ApplyArgument:
|
|
out << "apply argument";
|
|
break;
|
|
|
|
case ApplyFunction:
|
|
out << "apply function";
|
|
break;
|
|
|
|
case OptionalPayload:
|
|
out << "optional payload";
|
|
break;
|
|
|
|
case ApplyArgToParam:
|
|
out << "comparing call argument #" << llvm::utostr(elt.getValue())
|
|
<< " to parameter #" << llvm::utostr(elt.getValue2());
|
|
break;
|
|
|
|
case ClosureResult:
|
|
out << "closure result";
|
|
break;
|
|
|
|
case ConstructorMember:
|
|
out << "constructor member";
|
|
break;
|
|
|
|
case FunctionArgument:
|
|
out << "function argument";
|
|
break;
|
|
|
|
case FunctionResult:
|
|
out << "function result";
|
|
break;
|
|
|
|
case GeneratorElementType:
|
|
out << "generator element type";
|
|
break;
|
|
|
|
case GenericArgument:
|
|
out << "generic argument #" << llvm::utostr(elt.getValue());
|
|
break;
|
|
|
|
case InstanceType:
|
|
out << "instance type";
|
|
break;
|
|
|
|
case Load:
|
|
out << "load";
|
|
break;
|
|
|
|
case Member:
|
|
out << "member";
|
|
break;
|
|
|
|
case MemberRefBase:
|
|
out << "member reference base";
|
|
break;
|
|
|
|
case NamedTupleElement:
|
|
out << "named tuple element #" << llvm::utostr(elt.getValue());
|
|
break;
|
|
|
|
case UnresolvedMember:
|
|
out << "unresolved member";
|
|
break;
|
|
|
|
case ParentType:
|
|
out << "parent type";
|
|
break;
|
|
|
|
case RvalueAdjustment:
|
|
out << "rvalue adjustment";
|
|
break;
|
|
|
|
case ScalarToTuple:
|
|
out << "scalar to tuple";
|
|
break;
|
|
|
|
case SequenceIteratorProtocol:
|
|
out << "sequence iterator type";
|
|
break;
|
|
|
|
case SubscriptIndex:
|
|
out << "subscript index";
|
|
break;
|
|
|
|
case SubscriptMember:
|
|
out << "subscript member";
|
|
break;
|
|
|
|
case SubscriptResult:
|
|
out << "subscript result";
|
|
break;
|
|
|
|
case TupleElement:
|
|
out << "tuple element #" << llvm::utostr(elt.getValue());
|
|
break;
|
|
|
|
case KeyPathComponent:
|
|
out << "key path component #" << llvm::utostr(elt.getValue());
|
|
break;
|
|
|
|
case Requirement:
|
|
out << "requirement ";
|
|
elt.getRequirement()->dumpRef(out);
|
|
break;
|
|
|
|
case Witness:
|
|
out << "witness ";
|
|
elt.getWitness()->dumpRef(out);
|
|
break;
|
|
|
|
case OpenedGeneric:
|
|
out << "opened generic";
|
|
break;
|
|
|
|
case ConditionalRequirement:
|
|
out << "conditional requirement #" << llvm::utostr(elt.getValue());
|
|
break;
|
|
|
|
case TypeParameterRequirement:
|
|
out << "type parameter requirement #" << llvm::utostr(elt.getValue());
|
|
break;
|
|
}
|
|
}
|
|
|
|
out << ']';
|
|
}
|