blob: 519886db59f7a991b6f95bd568f083bcd4b09a8b [file] [log] [blame]
// Copyright (c) 2016, 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.kernel_field_builder;
import 'package:kernel/ast.dart'
show DartType, Expression, Field, Name, Node, NullLiteral;
import '../../base/instrumentation.dart'
show Instrumentation, InstrumentationValueForType;
import '../../scanner/token.dart' show Token;
import '../fasta_codes.dart' show messageInternalProblemAlreadyInitialized;
import '../problems.dart' show internalProblem;
import '../type_inference/type_inference_listener.dart'
show TypeInferenceListener;
import 'kernel_body_builder.dart' show KernelBodyBuilder;
import 'kernel_builder.dart'
show
Declaration,
FieldBuilder,
KernelTypeBuilder,
LibraryBuilder,
MetadataBuilder;
import 'kernel_shadow_ast.dart' show ShadowField;
class KernelFieldBuilder extends FieldBuilder<Expression> {
final ShadowField field;
final List<MetadataBuilder> metadata;
final KernelTypeBuilder type;
final Token initializerTokenForInference;
final bool hasInitializer;
KernelFieldBuilder(
this.metadata,
this.type,
String name,
int modifiers,
Declaration compilationUnit,
int charOffset,
this.initializerTokenForInference,
this.hasInitializer)
: field = new ShadowField(null, type == null,
fileUri: compilationUnit?.fileUri)
..fileOffset = charOffset,
super(name, modifiers, compilationUnit, charOffset);
void set initializer(Expression value) {
if (!hasInitializer && value is! NullLiteral && !isConst && !isFinal) {
internalProblem(
messageInternalProblemAlreadyInitialized, charOffset, fileUri);
}
field.initializer = value..parent = field;
}
bool get isEligibleForInference =>
type == null && (hasInitializer || isInstanceMember);
Field build(LibraryBuilder library) {
field.name ??= new Name(name, library.target);
if (type != null) {
field.type = type.build(library);
}
bool isInstanceMember = !isStatic && !isTopLevel;
field
..isCovariant = isCovariant
..isFinal = isFinal
..isConst = isConst
..hasImplicitGetter = isInstanceMember
..hasImplicitSetter = isInstanceMember && !isConst && !isFinal
..isStatic = !isInstanceMember;
if (!library.disableTypeInference &&
isEligibleForInference &&
!isInstanceMember) {
library.loader.typeInferenceEngine
.recordStaticFieldInferenceCandidate(field, library);
}
return field;
}
Field get target => field;
@override
void prepareTopLevelInference() {
if (!isEligibleForInference) return;
var listener = new TypeInferenceListener<int, int, Node, int>();
var typeInferrer = library.loader.typeInferenceEngine
.createTopLevelTypeInferrer(
listener, field.enclosingClass?.thisType, field);
if (hasInitializer) {
initializer = new KernelBodyBuilder.forField(this, typeInferrer)
.parseFieldInitializer(initializerTokenForInference);
}
}
@override
void instrumentTopLevelInference(Instrumentation instrumentation) {
if (isEligibleForInference) {
instrumentation.record(field.fileUri, field.fileOffset, 'topType',
new InstrumentationValueForType(field.type));
}
}
@override
DartType get builtType => field.type;
@override
bool get hasTypeInferredFromInitializer =>
ShadowField.hasTypeInferredFromInitializer(field);
}