blob: ea2e411f37399d3e7de0ca318d8a05865d5812e7 [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.
// @dart = 2.9
library fasta.dill_class_builder;
import 'package:kernel/ast.dart';
import '../builder/class_builder.dart';
import '../builder/library_builder.dart';
import '../builder/member_builder.dart';
import '../builder/type_builder.dart';
import '../builder/type_variable_builder.dart';
import '../scope.dart';
import '../problems.dart' show unimplemented;
import '../modifier.dart' show abstractMask, namedMixinApplicationMask;
import 'dill_library_builder.dart' show DillLibraryBuilder;
import 'dill_member_builder.dart';
class DillClassBuilder extends ClassBuilderImpl {
final Class cls;
DillClassBuilder(Class cls, DillLibraryBuilder parent)
: cls = cls,
super(
null,
computeModifiers(cls),
cls.name,
null,
null,
null,
null,
new Scope(
local: <String, MemberBuilder>{},
setters: <String, MemberBuilder>{},
parent: parent.scope,
debugName: "class ${cls.name}",
isModifiable: false),
new ConstructorScope(cls.name, <String, MemberBuilder>{}),
parent,
cls.fileOffset);
List<TypeVariableBuilder> get typeVariables {
List<TypeVariableBuilder> typeVariables = super.typeVariables;
if (typeVariables == null && cls.typeParameters.isNotEmpty) {
typeVariables = super.typeVariables =
computeTypeVariableBuilders(library, cls.typeParameters);
}
return typeVariables;
}
Uri get fileUri => cls.fileUri;
TypeBuilder get supertypeBuilder {
TypeBuilder supertype = super.supertypeBuilder;
if (supertype == null) {
Supertype targetSupertype = cls.supertype;
if (targetSupertype == null) return null;
super.supertypeBuilder =
supertype = computeTypeBuilder(library, targetSupertype);
}
return supertype;
}
@override
Class get actualCls => cls;
void addMember(Member member) {
if (member is Field) {
DillFieldBuilder builder = new DillFieldBuilder(member, this);
String name = member.name.text;
scopeBuilder.addMember(name, builder);
} else if (member is Procedure) {
String name = member.name.text;
switch (member.kind) {
case ProcedureKind.Factory:
constructorScopeBuilder.addMember(
name, new DillFactoryBuilder(member, this));
break;
case ProcedureKind.Setter:
scopeBuilder.addSetter(name, new DillSetterBuilder(member, this));
break;
case ProcedureKind.Getter:
scopeBuilder.addMember(name, new DillGetterBuilder(member, this));
break;
case ProcedureKind.Operator:
scopeBuilder.addMember(name, new DillOperatorBuilder(member, this));
break;
case ProcedureKind.Method:
scopeBuilder.addMember(name, new DillMethodBuilder(member, this));
break;
}
} else if (member is Constructor) {
DillConstructorBuilder builder = new DillConstructorBuilder(member, this);
String name = member.name.text;
constructorScopeBuilder.addMember(name, builder);
} else {
throw new UnsupportedError(
"Unexpected class member ${member} (${member.runtimeType})");
}
}
@override
int get typeVariablesCount => cls.typeParameters.length;
@override
List<DartType> buildTypeArguments(
LibraryBuilder library, List<TypeBuilder> arguments,
[bool notInstanceContext]) {
// 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) {
List<DartType> result = new List<DartType>.filled(
cls.typeParameters.length, null,
growable: true);
for (int i = 0; i < result.length; ++i) {
result[i] = cls.typeParameters[i].defaultType;
}
return result;
}
// [arguments] != null
List<DartType> result =
new List<DartType>.filled(arguments.length, null, growable: true);
for (int i = 0; i < result.length; ++i) {
result[i] = arguments[i].build(library, null, notInstanceContext);
}
return result;
}
/// Returns true if this class is the result of applying a mixin to its
/// superclass.
bool get isMixinApplication => cls.isMixinApplication;
@override
bool get declaresConstConstructor => cls.hasConstConstructor;
TypeBuilder get mixedInTypeBuilder {
return computeTypeBuilder(library, cls.mixedInType);
}
List<TypeBuilder> get interfaceBuilders {
if (cls.implementedTypes.isEmpty) return null;
if (super.interfaceBuilders == null) {
List<TypeBuilder> result =
new List<TypeBuilder>.filled(cls.implementedTypes.length, null);
for (int i = 0; i < result.length; i++) {
result[i] = computeTypeBuilder(library, cls.implementedTypes[i]);
}
super.interfaceBuilders = result;
}
return super.interfaceBuilders;
}
void set mixedInTypeBuilder(TypeBuilder mixin) {
unimplemented("mixedInType=", -1, null);
}
void clearCachedValues() {
supertypeBuilder = null;
interfaceBuilders = null;
}
}
int computeModifiers(Class cls) {
int modifiers = 0;
if (cls.isAbstract) {
modifiers |= abstractMask;
}
if (cls.isMixinApplication && cls.name != null) {
modifiers |= namedMixinApplicationMask;
}
return modifiers;
}
TypeBuilder computeTypeBuilder(
DillLibraryBuilder library, Supertype supertype) {
return supertype == null
? null
: library.loader.computeTypeBuilder(supertype.asInterfaceType);
}
List<TypeVariableBuilder> computeTypeVariableBuilders(
DillLibraryBuilder library, List<TypeParameter> typeParameters) {
if (typeParameters == null || typeParameters.length == 0) return null;
List<TypeVariableBuilder> result =
new List.filled(typeParameters.length, null);
for (int i = 0; i < result.length; i++) {
result[i] = new TypeVariableBuilder.fromKernel(typeParameters[i], library);
}
return result;
}