blob: bbf02a209375099c179dba1a0db0849895fd028c [file] [log] [blame]
// Copyright (c) 2017, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
library fasta.dill_class_builder;
import 'package:kernel/ast.dart';
import 'package:kernel/class_hierarchy.dart';
import '../base/loader.dart';
import '../base/modifier.dart' show abstractMask, namedMixinApplicationMask;
import '../base/name_space.dart';
import '../base/problems.dart' show unimplemented;
import '../base/scope.dart';
import '../builder/builder.dart';
import '../builder/declaration_builders.dart';
import '../builder/library_builder.dart';
import '../builder/member_builder.dart';
import '../builder/name_iterator.dart';
import '../builder/type_builder.dart';
import 'dill_library_builder.dart' show DillLibraryBuilder;
import 'dill_member_builder.dart';
mixin DillClassMemberAccessMixin implements ClassMemberAccess {
NameSpace get nameSpace;
ConstructorScope get constructorScope;
@override
// Coverage-ignore(suite): Not run.
Iterator<T> fullConstructorIterator<T extends MemberBuilder>() =>
constructorScope.filteredIterator<T>(
includeAugmentations: true, includeDuplicates: false);
@override
NameIterator<T> fullConstructorNameIterator<T extends MemberBuilder>() =>
constructorScope.filteredNameIterator<T>(
includeAugmentations: true, includeDuplicates: false);
@override
Iterator<T> fullMemberIterator<T extends Builder>() =>
nameSpace.filteredIterator<T>(
includeAugmentations: true, includeDuplicates: false);
@override
// Coverage-ignore(suite): Not run.
NameIterator<T> fullMemberNameIterator<T extends Builder>() =>
nameSpace.filteredNameIterator<T>(
includeAugmentations: true, includeDuplicates: false);
}
class DillClassBuilder extends ClassBuilderImpl
with DillClassMemberAccessMixin {
@override
final Class cls;
late final LookupScope _scope;
final NameSpace _nameSpace;
@override
final ConstructorScope constructorScope;
List<NominalVariableBuilder>? _typeVariables;
TypeBuilder? _supertypeBuilder;
List<TypeBuilder>? _interfaceBuilders;
DillClassBuilder(this.cls, DillLibraryBuilder parent)
: _nameSpace = new NameSpaceImpl(),
constructorScope =
new ConstructorScope(cls.name, <String, MemberBuilder>{}),
super(/*metadata builders*/ null, computeModifiers(cls), cls.name,
parent, cls.fileOffset) {
_scope = new NameSpaceLookupScope(
_nameSpace, ScopeKind.declaration, "class ${cls.name}",
parent: parent.scope);
}
@override
// Coverage-ignore(suite): Not run.
LookupScope get scope => _scope;
@override
NameSpace get nameSpace => _nameSpace;
@override
bool get isEnum => cls.isEnum;
@override
DillClassBuilder get origin => this;
@override
DillLibraryBuilder get libraryBuilder =>
super.libraryBuilder as DillLibraryBuilder;
@override
bool get isMacro => cls.isMacro;
@override
bool get isMixinClass => cls.isMixinClass;
@override
bool get isMixinDeclaration => cls.isMixinDeclaration;
@override
bool get isSealed => cls.isSealed;
@override
bool get isBase => cls.isBase;
@override
bool get isInterface => cls.isInterface;
@override
bool get isFinal => cls.isFinal;
@override
List<NominalVariableBuilder>? get typeVariables {
List<NominalVariableBuilder>? typeVariables = _typeVariables;
if (typeVariables == null && cls.typeParameters.isNotEmpty) {
typeVariables = _typeVariables = computeTypeVariableBuilders(
cls.typeParameters, libraryBuilder.loader);
}
return typeVariables;
}
@override
Uri get fileUri => cls.fileUri;
@override
TypeBuilder? get supertypeBuilder {
TypeBuilder? supertype = _supertypeBuilder;
if (supertype == null) {
Supertype? targetSupertype = cls.supertype;
if (targetSupertype == null) return null;
_supertypeBuilder =
supertype = computeTypeBuilder(libraryBuilder, targetSupertype);
}
return supertype;
}
@override
// Coverage-ignore(suite): Not run.
List<TypeBuilder>? get onTypes => null;
void addField(Field field) {
DillFieldBuilder builder = new DillFieldBuilder(field, this);
String name = field.name.text;
nameSpace.addLocalMember(name, builder, setter: false);
}
void addConstructor(Constructor constructor, Procedure? constructorTearOff) {
DillConstructorBuilder builder =
new DillConstructorBuilder(constructor, constructorTearOff, this);
String name = constructor.name.text;
constructorScope.addLocalMember(name, builder);
}
void addFactory(Procedure factory, Procedure? factoryTearOff) {
String name = factory.name.text;
constructorScope.addLocalMember(
name, new DillFactoryBuilder(factory, factoryTearOff, this));
}
void addProcedure(Procedure procedure) {
String name = procedure.name.text;
switch (procedure.kind) {
case ProcedureKind.Factory:
// Coverage-ignore(suite): Not run.
throw new UnsupportedError("Use addFactory for adding factories");
case ProcedureKind.Setter:
nameSpace.addLocalMember(name, new DillSetterBuilder(procedure, this),
setter: true);
break;
case ProcedureKind.Getter:
nameSpace.addLocalMember(name, new DillGetterBuilder(procedure, this),
setter: false);
break;
case ProcedureKind.Operator:
nameSpace.addLocalMember(name, new DillOperatorBuilder(procedure, this),
setter: false);
break;
case ProcedureKind.Method:
nameSpace.addLocalMember(name, new DillMethodBuilder(procedure, this),
setter: false);
break;
}
}
@override
int get typeVariablesCount => cls.typeParameters.length;
@override
List<DartType> buildAliasedTypeArguments(LibraryBuilder library,
List<TypeBuilder>? arguments, ClassHierarchyBase? hierarchy) {
// For performance reasons, [typeVariables] aren't restored from [target].
// So, if [arguments] is null, the default types should be retrieved from
// [cls.typeParameters].
if (arguments == null) {
// TODO(johnniwinther): Use i2b here when needed.
return new List<DartType>.generate(cls.typeParameters.length,
(int i) => cls.typeParameters[i].defaultType,
growable: true);
}
// [arguments] != null
return new List<DartType>.generate(
arguments.length,
(int i) =>
arguments[i].buildAliased(library, TypeUse.typeArgument, hierarchy),
growable: true);
}
/// Returns true if this class is the result of applying a mixin to its
/// superclass.
@override
bool get isMixinApplication => cls.isMixinApplication;
@override
// Coverage-ignore(suite): Not run.
bool get declaresConstConstructor => cls.hasConstConstructor;
@override
TypeBuilder? get mixedInTypeBuilder {
return computeTypeBuilder(libraryBuilder, cls.mixedInType);
}
@override
// Coverage-ignore(suite): Not run.
void set mixedInTypeBuilder(TypeBuilder? mixin) {
unimplemented("mixedInType=", -1, null);
}
@override
List<TypeBuilder>? get interfaceBuilders {
if (cls.implementedTypes.isEmpty) return null;
List<TypeBuilder>? interfaceBuilders = _interfaceBuilders;
if (interfaceBuilders == null) {
interfaceBuilders = _interfaceBuilders = new List<TypeBuilder>.generate(
cls.implementedTypes.length,
(int i) =>
computeTypeBuilder(libraryBuilder, cls.implementedTypes[i])!,
growable: false);
}
return interfaceBuilders;
}
void clearCachedValues() {
_supertypeBuilder = null;
_interfaceBuilders = null;
_typeVariables = null;
}
}
int computeModifiers(Class cls) {
int modifiers = 0;
if (cls.isAbstract) {
modifiers |= abstractMask;
}
if (cls.isMixinApplication) {
modifiers |= namedMixinApplicationMask;
}
return modifiers;
}
TypeBuilder? computeTypeBuilder(
DillLibraryBuilder library, Supertype? supertype) {
return supertype == null
? null
: library.loader.computeTypeBuilder(supertype.asInterfaceType);
}
List<NominalVariableBuilder>? computeTypeVariableBuilders(
List<TypeParameter>? typeParameters, Loader loader) {
if (typeParameters == null || typeParameters.length == 0) return null;
return <NominalVariableBuilder>[
for (TypeParameter typeParameter in typeParameters)
new NominalVariableBuilder.fromKernel(typeParameter, loader: loader)
];
}