| // Copyright (c) 2020, 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. |
| |
| import 'dart:typed_data'; |
| |
| import 'package:analyzer/dart/analysis/features.dart'; |
| import 'package:analyzer/dart/ast/ast.dart'; |
| import 'package:analyzer/dart/element/element.dart'; |
| import 'package:analyzer/dart/element/nullability_suffix.dart'; |
| import 'package:analyzer/dart/element/type.dart'; |
| import 'package:analyzer/src/dart/analysis/experiments.dart'; |
| import 'package:analyzer/src/dart/ast/ast.dart'; |
| import 'package:analyzer/src/dart/element/element.dart'; |
| import 'package:analyzer/src/dart/element/member.dart'; |
| import 'package:analyzer/src/dart/element/type.dart'; |
| import 'package:analyzer/src/dart/element/type_algebra.dart'; |
| import 'package:analyzer/src/dart/resolver/variance.dart'; |
| import 'package:analyzer/src/generated/source.dart'; |
| import 'package:analyzer/src/generated/utilities_dart.dart'; |
| import 'package:analyzer/src/summary2/ast_binary_reader.dart'; |
| import 'package:analyzer/src/summary2/ast_binary_tag.dart'; |
| import 'package:analyzer/src/summary2/data_reader.dart'; |
| import 'package:analyzer/src/summary2/element_flags.dart'; |
| import 'package:analyzer/src/summary2/informative_data.dart'; |
| import 'package:analyzer/src/summary2/linked_element_factory.dart'; |
| import 'package:analyzer/src/summary2/reference.dart'; |
| import 'package:analyzer/src/task/inference_error.dart'; |
| import 'package:pub_semver/pub_semver.dart'; |
| |
| class BundleReader { |
| final SummaryDataReader _reader; |
| final Map<Uri, Uint8List> _unitsInformativeBytes; |
| |
| final Map<String, LibraryReader> libraryMap = {}; |
| |
| BundleReader({ |
| required LinkedElementFactory elementFactory, |
| required Uint8List resolutionBytes, |
| Map<Uri, Uint8List> unitsInformativeBytes = const {}, |
| }) : _reader = SummaryDataReader(resolutionBytes), |
| _unitsInformativeBytes = unitsInformativeBytes { |
| _reader.offset = _reader.bytes.length - 4 * 4; |
| var baseResolutionOffset = _reader.readUInt32(); |
| var librariesOffset = _reader.readUInt32(); |
| var referencesOffset = _reader.readUInt32(); |
| var stringsOffset = _reader.readUInt32(); |
| _reader.createStringTable(stringsOffset); |
| |
| var referenceReader = _ReferenceReader( |
| elementFactory, |
| _reader, |
| referencesOffset, |
| ); |
| |
| _reader.offset = librariesOffset; |
| var libraryHeaderList = _reader.readTypedList(() { |
| return _LibraryHeader( |
| uriStr: _reader.readStringReference(), |
| offset: _reader.readUInt30(), |
| classMembersLengths: _reader.readUInt30List(), |
| ); |
| }); |
| |
| for (var libraryHeader in libraryHeaderList) { |
| _reader.offset = libraryHeader.offset; |
| var uriStr = libraryHeader.uriStr; |
| var reference = elementFactory.rootReference.getChild(uriStr); |
| libraryMap[uriStr] = LibraryReader._( |
| elementFactory: elementFactory, |
| reader: _reader, |
| unitsInformativeBytes: _unitsInformativeBytes, |
| baseResolutionOffset: baseResolutionOffset, |
| referenceReader: referenceReader, |
| reference: reference, |
| offset: _reader.offset, |
| classMembersLengths: libraryHeader.classMembersLengths, |
| ); |
| } |
| } |
| } |
| |
| class ClassElementLinkedData extends ElementLinkedData<ClassElementImpl> { |
| void Function()? _readMembers; |
| void Function()? applyInformativeDataToMembers; |
| |
| ClassElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| /// Ensure that all members of the [element] are available. This includes |
| /// being able to ask them for example using [ClassElement.methods], and |
| /// as well access them through their [Reference]s. For a class declaration |
| /// this means reading them, for a named mixin application this means |
| /// computing constructors. |
| void readMembers(ClassElementImpl element) { |
| if (element.isMixinApplication) { |
| element.constructors; |
| } else { |
| _readMembers?.call(); |
| _readMembers = null; |
| |
| applyInformativeDataToMembers?.call(); |
| applyInformativeDataToMembers = null; |
| } |
| } |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| _readTypeParameters(reader, element.typeParameters); |
| element.supertype = reader._readOptionalInterfaceType(); |
| element.mixins = reader._readInterfaceTypeList(); |
| element.interfaces = reader._readInterfaceTypeList(); |
| } |
| } |
| |
| class CompilationUnitElementLinkedData |
| extends ElementLinkedData<CompilationUnitElementImpl> { |
| CompilationUnitElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| } |
| } |
| |
| class ConstructorElementLinkedData |
| extends ElementLinkedData<ConstructorElementImpl> { |
| ConstructorElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| _addEnclosingElementTypeParameters(reader, element); |
| |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| reader._addFormalParameters(element.parameters); |
| _readFormalParameters(reader, element.parameters); |
| if (element.isConst || element.isFactory) { |
| element.redirectedConstructor = |
| reader.readElement() as ConstructorElement?; |
| element.constantInitializers = reader._readNodeList(); |
| } |
| } |
| } |
| |
| /// Lazy reader of resolution information. |
| abstract class ElementLinkedData<E extends ElementImpl> { |
| final Reference reference; |
| final LibraryReader _libraryReader; |
| final CompilationUnitElementImpl unitElement; |
| |
| /// When this object is created, this offset is the offset of the resolution |
| /// information in the [_libraryReader]. After reading is done, this offset |
| /// is set to `-1`. |
| int _offset; |
| |
| ElementLinkedData( |
| this.reference, LibraryReader libraryReader, this.unitElement, int offset) |
| : _libraryReader = libraryReader, |
| _offset = offset; |
| |
| void read(ElementImpl element) { |
| if (_offset == -1) { |
| return null; |
| } |
| |
| var dataReader = _libraryReader._reader.fork(_offset); |
| _offset = -1; |
| |
| var reader = ResolutionReader( |
| _libraryReader._elementFactory, |
| _libraryReader._referenceReader, |
| dataReader, |
| ); |
| |
| _read(element as E, reader); |
| } |
| |
| void _addEnclosingElementTypeParameters( |
| ResolutionReader reader, |
| ElementImpl element, |
| ) { |
| var enclosing = element.enclosingElement; |
| if (enclosing is ClassElement) { |
| reader._addTypeParameters(enclosing.typeParameters); |
| } else if (enclosing is CompilationUnitElement) { |
| // Nothing. |
| } else if (enclosing is ExtensionElement) { |
| reader._addTypeParameters(enclosing.typeParameters); |
| } else { |
| throw UnimplementedError('${enclosing.runtimeType}'); |
| } |
| } |
| |
| void _read(E element, ResolutionReader reader); |
| |
| void _readFormalParameters( |
| ResolutionReader reader, |
| List<ParameterElement> parameters, |
| ) { |
| for (var parameter in parameters) { |
| parameter as ParameterElementImpl; |
| parameter.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| _readTypeParameters(reader, parameter.typeParameters); |
| _readFormalParameters(reader, parameter.parameters); |
| parameter.type = reader.readRequiredType(); |
| if (parameter is ConstVariableElement) { |
| var defaultParameter = parameter as ConstVariableElement; |
| var initializer = reader._readOptionalExpression(); |
| if (initializer != null) { |
| defaultParameter.constantInitializer = initializer; |
| } |
| } |
| if (parameter is FieldFormalParameterElementImpl) { |
| parameter.field = reader.readElement() as FieldElement?; |
| } |
| } |
| } |
| |
| void _readTypeParameters( |
| ResolutionReader reader, |
| List<TypeParameterElement> typeParameters, |
| ) { |
| reader._addTypeParameters(typeParameters); |
| for (var typeParameter in typeParameters) { |
| typeParameter as TypeParameterElementImpl; |
| typeParameter.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| typeParameter.bound = reader.readType(); |
| typeParameter.defaultType = reader.readType(); |
| } |
| } |
| } |
| |
| class EnumElementLinkedData extends ElementLinkedData<EnumElementImpl> { |
| EnumElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| var typeProvider = element.library.typeProvider; |
| |
| element.metadata = reader._readAnnotationList( |
| unitElement: element.enclosingElement, |
| ); |
| |
| var indexField = element.getField('index') as FieldElementImpl; |
| indexField.type = typeProvider.intType; |
| |
| var toStringMethod = element.getMethod('toString') as MethodElementImpl; |
| toStringMethod.returnType = typeProvider.stringType; |
| |
| for (var constant in element.constants) { |
| constant as FieldElementImpl; |
| constant.metadata = reader._readAnnotationList( |
| unitElement: element.enclosingElement, |
| ); |
| } |
| } |
| } |
| |
| class ExtensionElementLinkedData |
| extends ElementLinkedData<ExtensionElementImpl> { |
| ExtensionElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: element.enclosingElement, |
| ); |
| _readTypeParameters(reader, element.typeParameters); |
| element.extendedType = reader.readRequiredType(); |
| } |
| } |
| |
| class FieldElementLinkedData extends ElementLinkedData<FieldElementImpl> { |
| FieldElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| _addEnclosingElementTypeParameters(reader, element); |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| element.type = reader.readRequiredType(); |
| if (element is ConstFieldElementImpl) { |
| var initializer = reader._readOptionalExpression(); |
| if (initializer != null) { |
| element.constantInitializer = initializer; |
| ConstantContextForExpressionImpl(initializer); |
| } |
| } |
| } |
| } |
| |
| class FunctionElementLinkedData extends ElementLinkedData<FunctionElementImpl> { |
| FunctionElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| _readTypeParameters(reader, element.typeParameters); |
| element.returnType = reader.readRequiredType(); |
| _readFormalParameters(reader, element.parameters); |
| } |
| } |
| |
| class LibraryElementLinkedData extends ElementLinkedData<LibraryElementImpl> { |
| LibraryElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| LinkedElementFactory get elementFactory { |
| return _libraryReader._elementFactory; |
| } |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| |
| for (var import in element.imports) { |
| import as ImportElementImpl; |
| import.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| import.importedLibrary = reader.readElement() as LibraryElementImpl?; |
| } |
| |
| for (var export in element.exports) { |
| export as ExportElementImpl; |
| export.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| export.exportedLibrary = reader.readElement() as LibraryElementImpl?; |
| } |
| |
| element.entryPoint = reader.readElement() as FunctionElement?; |
| } |
| } |
| |
| class LibraryReader { |
| final LinkedElementFactory _elementFactory; |
| final SummaryDataReader _reader; |
| final Map<Uri, Uint8List> _unitsInformativeBytes; |
| final int _baseResolutionOffset; |
| final _ReferenceReader _referenceReader; |
| final Reference _reference; |
| final int _offset; |
| |
| final Uint32List _classMembersLengths; |
| int _classMembersLengthsIndex = 0; |
| |
| late List<Reference> exports; |
| |
| LibraryReader._({ |
| required LinkedElementFactory elementFactory, |
| required SummaryDataReader reader, |
| required Map<Uri, Uint8List> unitsInformativeBytes, |
| required int baseResolutionOffset, |
| required _ReferenceReader referenceReader, |
| required Reference reference, |
| required int offset, |
| required Uint32List classMembersLengths, |
| }) : _elementFactory = elementFactory, |
| _reader = reader, |
| _unitsInformativeBytes = unitsInformativeBytes, |
| _baseResolutionOffset = baseResolutionOffset, |
| _referenceReader = referenceReader, |
| _reference = reference, |
| _offset = offset, |
| _classMembersLengths = classMembersLengths; |
| |
| LibraryElementImpl readElement({required Source librarySource}) { |
| var analysisContext = _elementFactory.analysisContext; |
| var analysisSession = _elementFactory.analysisSession; |
| var sourceFactory = analysisContext.sourceFactory; |
| |
| _reader.offset = _offset; |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| |
| var name = _reader.readStringReference(); |
| var featureSet = _readFeatureSet(); |
| |
| var libraryElement = LibraryElementImpl( |
| analysisContext, analysisSession, name, -1, 0, featureSet); |
| _reference.element = libraryElement; |
| libraryElement.reference = _reference; |
| |
| libraryElement.languageVersion = _readLanguageVersion(); |
| libraryElement.imports = _reader.readTypedList(_readImportElement); |
| libraryElement.exports = _reader.readTypedList(_readExportElement); |
| LibraryElementFlags.read(_reader, libraryElement); |
| |
| var unitContainerRef = _reference.getChild('@unit'); |
| var unitCount = _reader.readUInt30(); |
| var units = <CompilationUnitElementImpl>[]; |
| for (var i = 0; i < unitCount; i++) { |
| var unitElement = _readUnitElement( |
| sourceFactory: sourceFactory, |
| unitContainerRef: unitContainerRef, |
| libraryElement: libraryElement, |
| librarySource: librarySource, |
| ); |
| units.add(unitElement); |
| } |
| |
| var exportsIndexList = _reader.readUInt30List(); |
| exports = exportsIndexList |
| .map((index) => _referenceReader.referenceOfIndex(index)) |
| .toList(); |
| |
| libraryElement.definingCompilationUnit = units[0]; |
| libraryElement.parts = units.skip(1).toList(); |
| |
| libraryElement.linkedData = LibraryElementLinkedData( |
| reference: _reference, |
| libraryReader: this, |
| unitElement: units[0], |
| offset: resolutionOffset, |
| ); |
| |
| InformativeDataApplier(_elementFactory, _unitsInformativeBytes) |
| .applyTo(libraryElement); |
| |
| return libraryElement; |
| } |
| |
| ClassElementImpl _readClassElement( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = unitReference.getChild('@class').getChild(name); |
| |
| var element = ClassElementImpl(name, -1); |
| |
| var linkedData = ClassElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| ClassElementFlags.read(_reader, element); |
| |
| element.typeParameters = _readTypeParameters(); |
| |
| if (!element.isMixinApplication) { |
| var membersOffset = _reader.offset; |
| linkedData._readMembers = () { |
| _reader.offset = membersOffset; |
| _readClassElementMembers(unitElement, element, reference); |
| }; |
| _reader.offset += _classMembersLengths[_classMembersLengthsIndex++]; |
| } |
| |
| return element; |
| } |
| |
| void _readClassElementMembers( |
| CompilationUnitElementImpl unitElement, |
| ClassElementImpl element, |
| Reference reference, |
| ) { |
| var accessors = <PropertyAccessorElementImpl>[]; |
| var fields = <FieldElement>[]; |
| _readFields(unitElement, element, reference, accessors, fields); |
| _readPropertyAccessors( |
| unitElement, element, reference, accessors, fields, '@field'); |
| element.fields = fields; |
| element.accessors = accessors; |
| |
| element.constructors = _readConstructors(unitElement, element, reference); |
| element.methods = _readMethods(unitElement, element, reference); |
| } |
| |
| void _readClasses( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var length = _reader.readUInt30(); |
| unitElement.types = List.generate(length, (index) { |
| return _readClassElement(unitElement, unitReference); |
| }); |
| } |
| |
| List<ConstructorElementImpl> _readConstructors( |
| CompilationUnitElementImpl unitElement, |
| ClassElementImpl classElement, |
| Reference classReference, |
| ) { |
| var containerRef = classReference.getChild('@constructor'); |
| var length = _reader.readUInt30(); |
| return List.generate(length, (_) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = containerRef.getChild(name); |
| var element = ConstructorElementImpl(name, -1); |
| var linkedData = ConstructorElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| ConstructorElementFlags.read(_reader, element); |
| element.parameters = _readParameters(element, reference); |
| return element; |
| }); |
| } |
| |
| EnumElementImpl _readEnumElement( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = unitReference.getChild('@enum').getChild(name); |
| |
| var element = EnumElementImpl(name, -1); |
| |
| var linkedData = EnumElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| var fields = <FieldElement>[]; |
| var getters = <PropertyAccessorElement>[]; |
| |
| // Build the 'index' field. |
| { |
| var field = FieldElementImpl('index', -1) |
| ..enclosingElement = element |
| ..isSynthetic = true |
| ..isFinal = true; |
| fields.add(field); |
| getters.add( |
| PropertyAccessorElementImpl_ImplicitGetter(field, |
| reference: reference.getChild('@getter').getChild('index')) |
| ..enclosingElement = element, |
| ); |
| } |
| |
| // Build the 'values' field. |
| { |
| var field = ConstFieldElementImpl_EnumValues(element); |
| fields.add(field); |
| getters.add( |
| PropertyAccessorElementImpl_ImplicitGetter(field, |
| reference: reference.getChild('@getter').getChild('values')) |
| ..enclosingElement = element, |
| ); |
| } |
| |
| // Build fields for all enum constants. |
| var containerRef = reference.getChild('@constant'); |
| var constantCount = _reader.readUInt30(); |
| for (var i = 0; i < constantCount; i++) { |
| var constantName = _reader.readStringReference(); |
| var field = ConstFieldElementImpl_EnumValue(element, constantName, i); |
| var constantRef = containerRef.getChild(name); |
| field.reference = constantRef; |
| constantRef.element = field; |
| fields.add(field); |
| getters.add( |
| PropertyAccessorElementImpl_ImplicitGetter(field, |
| reference: reference.getChild('@getter').getChild(constantName)) |
| ..enclosingElement = element, |
| ); |
| } |
| |
| element.fields = fields; |
| element.accessors = getters; |
| element.createToStringMethodElement(); |
| |
| return element; |
| } |
| |
| void _readEnums( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var count = _reader.readUInt30(); |
| unitElement.enums = List.generate(count, (_) { |
| return _readEnumElement(unitElement, unitReference); |
| }); |
| } |
| |
| ExportElementImpl _readExportElement() { |
| var element = ExportElementImpl(-1); |
| element.uri = _reader.readOptionalStringReference(); |
| element.combinators = _reader.readTypedList(_readNamespaceCombinator); |
| return element; |
| } |
| |
| ExtensionElementImpl _readExtensionElement( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readOptionalStringReference(); |
| var refName = _reader.readStringReference(); |
| var reference = unitReference.getChild('@extension').getChild(refName); |
| |
| var element = ExtensionElementImpl(name, -1); |
| element.setLinkedData( |
| reference, |
| ExtensionElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ), |
| ); |
| |
| element.typeParameters = _readTypeParameters(); |
| |
| var accessors = <PropertyAccessorElement>[]; |
| var fields = <FieldElement>[]; |
| _readPropertyAccessors( |
| unitElement, element, reference, accessors, fields, '@field'); |
| _readFields(unitElement, element, reference, accessors, fields); |
| element.accessors = accessors; |
| element.fields = fields; |
| |
| element.methods = _readMethods(unitElement, element, reference); |
| |
| return element; |
| } |
| |
| void _readExtensions( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var count = _reader.readUInt30(); |
| unitElement.extensions = List.generate(count, (_) { |
| return _readExtensionElement(unitElement, unitReference); |
| }); |
| } |
| |
| FeatureSet _readFeatureSet() { |
| var featureSetEncoded = _reader.readUint8List(); |
| return ExperimentStatus.fromStorage(featureSetEncoded); |
| } |
| |
| FieldElementImpl _readFieldElement( |
| CompilationUnitElementImpl unitElement, |
| ElementImpl classElement, |
| Reference classReference, |
| Reference containerRef, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var isConstElement = _reader.readBool(); |
| var reference = containerRef.getChild(name); |
| |
| FieldElementImpl element; |
| if (isConstElement) { |
| element = ConstFieldElementImpl(name, -1); |
| } else { |
| element = FieldElementImpl(name, -1); |
| } |
| |
| var linkedData = FieldElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| FieldElementFlags.read(_reader, element); |
| element.typeInferenceError = _readTopLevelInferenceError(); |
| element.createImplicitAccessors(classReference, name); |
| |
| return element; |
| } |
| |
| void _readFields( |
| CompilationUnitElementImpl unitElement, |
| ElementImpl classElement, |
| Reference classReference, |
| List<PropertyAccessorElement> accessors, |
| List<FieldElement> variables, |
| ) { |
| var containerRef = classReference.getChild('@field'); |
| var createdElements = <FieldElement>[]; |
| var variableElementCount = _reader.readUInt30(); |
| for (var i = 0; i < variableElementCount; i++) { |
| var variable = _readFieldElement( |
| unitElement, classElement, classReference, containerRef); |
| createdElements.add(variable); |
| variables.add(variable); |
| |
| var getter = variable.getter; |
| if (getter is PropertyAccessorElementImpl) { |
| accessors.add(getter); |
| } |
| |
| var setter = variable.setter; |
| if (setter is PropertyAccessorElementImpl) { |
| accessors.add(setter); |
| } |
| } |
| } |
| |
| void _readFunctions( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var count = _reader.readUInt30(); |
| unitElement.functions = List.generate(count, (_) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = unitReference.getChild('@function').getChild(name); |
| |
| var element = FunctionElementImpl(name, -1); |
| |
| var linkedData = FunctionElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| FunctionElementFlags.read(_reader, element); |
| element.typeParameters = _readTypeParameters(); |
| element.parameters = _readParameters(element, reference); |
| |
| return element; |
| }); |
| } |
| |
| ImportElementImpl _readImportElement() { |
| var element = ImportElementImpl(-1); |
| ImportElementFlags.read(_reader, element); |
| element.uri = _reader.readOptionalStringReference(); |
| var prefixName = _reader.readOptionalStringReference(); |
| if (prefixName != null) { |
| var reference = _reference.getChild('@prefix').getChild(prefixName); |
| var prefixElement = |
| PrefixElementImpl(prefixName, -1, reference: reference); |
| element.prefix = prefixElement; |
| } |
| element.combinators = _reader.readTypedList(_readNamespaceCombinator); |
| return element; |
| } |
| |
| LibraryLanguageVersion _readLanguageVersion() { |
| var packageMajor = _reader.readUInt30(); |
| var packageMinor = _reader.readUInt30(); |
| var package = Version(packageMajor, packageMinor, 0); |
| |
| Version? override; |
| if (_reader.readBool()) { |
| var overrideMajor = _reader.readUInt30(); |
| var overrideMinor = _reader.readUInt30(); |
| override = Version(overrideMajor, overrideMinor, 0); |
| } |
| |
| return LibraryLanguageVersion(package: package, override: override); |
| } |
| |
| List<MethodElementImpl> _readMethods( |
| CompilationUnitElementImpl unitElement, |
| ElementImpl enclosingElement, |
| Reference enclosingReference, |
| ) { |
| var containerRef = enclosingReference.getChild('@method'); |
| var length = _reader.readUInt30(); |
| return List.generate(length, (_) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = containerRef.getChild(name); |
| var element = MethodElementImpl(name, -1); |
| var linkedData = MethodElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| MethodElementFlags.read(_reader, element); |
| element.typeParameters = _readTypeParameters(); |
| element.parameters = _readParameters(element, reference); |
| element.typeInferenceError = _readTopLevelInferenceError(); |
| return element; |
| }); |
| } |
| |
| MixinElementImpl _readMixinElement( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = unitReference.getChild('@mixin').getChild(name); |
| |
| var element = MixinElementImpl(name, -1); |
| |
| var linkedData = MixinElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| element.typeParameters = _readTypeParameters(); |
| |
| var accessors = <PropertyAccessorElement>[]; |
| var fields = <FieldElement>[]; |
| _readPropertyAccessors( |
| unitElement, element, reference, accessors, fields, '@field'); |
| _readFields(unitElement, element, reference, accessors, fields); |
| element.accessors = accessors; |
| element.fields = fields; |
| |
| element.constructors = _readConstructors(unitElement, element, reference); |
| element.methods = _readMethods(unitElement, element, reference); |
| element.superInvokedNames = _reader.readStringReferenceList(); |
| |
| return element; |
| } |
| |
| void _readMixins( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var length = _reader.readUInt30(); |
| unitElement.mixins = List.generate(length, (index) { |
| return _readMixinElement(unitElement, unitReference); |
| }); |
| } |
| |
| NamespaceCombinator _readNamespaceCombinator() { |
| var tag = _reader.readByte(); |
| if (tag == Tag.HideCombinator) { |
| var combinator = HideElementCombinatorImpl(); |
| combinator.hiddenNames = _reader.readStringReferenceList(); |
| return combinator; |
| } else if (tag == Tag.ShowCombinator) { |
| var combinator = ShowElementCombinatorImpl(); |
| combinator.shownNames = _reader.readStringReferenceList(); |
| return combinator; |
| } else { |
| throw UnimplementedError('tag: $tag'); |
| } |
| } |
| |
| List<ParameterElementImpl> _readParameters( |
| ElementImpl enclosingElement, |
| Reference enclosingReference, |
| ) { |
| var containerRef = enclosingReference.getChild('@parameter'); |
| var length = _reader.readUInt30(); |
| return List.generate(length, (_) { |
| var name = _reader.readStringReference(); |
| var isInitializingFormal = _reader.readBool(); |
| var reference = containerRef.getChild(name); |
| |
| var kindIndex = _reader.readByte(); |
| var kind = ResolutionReader._formalParameterKind(kindIndex); |
| |
| ParameterElementImpl element; |
| if (kind.isRequiredPositional) { |
| if (isInitializingFormal) { |
| element = FieldFormalParameterElementImpl(name, -1); |
| } else { |
| element = ParameterElementImpl(name, -1); |
| } |
| } else { |
| if (isInitializingFormal) { |
| element = DefaultFieldFormalParameterElementImpl(name, -1); |
| } else { |
| element = DefaultParameterElementImpl(name, -1); |
| } |
| element.reference = reference; |
| reference.element = element; |
| } |
| element.parameterKind = kind; |
| ParameterElementFlags.read(_reader, element); |
| element.typeParameters = _readTypeParameters(); |
| element.parameters = _readParameters(element, reference); |
| return element; |
| }); |
| } |
| |
| PropertyAccessorElementImpl _readPropertyAccessorElement( |
| CompilationUnitElementImpl unitElement, |
| ElementImpl classElement, |
| Reference classReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| |
| var name = _reader.readStringReference(); |
| |
| var element = PropertyAccessorElementImpl(name, -1); |
| PropertyAccessorElementFlags.read(_reader, element); |
| |
| var reference = classReference |
| .getChild(element.isGetter ? '@getter' : '@setter') |
| .getChild(name); |
| var linkedData = PropertyAccessorElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| element.parameters = _readParameters(element, reference); |
| return element; |
| } |
| |
| void _readPropertyAccessors( |
| CompilationUnitElementImpl unitElement, |
| ElementImpl enclosingElement, |
| Reference enclosingReference, |
| List<PropertyAccessorElement> accessors, |
| List<PropertyInducingElement> properties, |
| String containerRefName, |
| ) { |
| var containerRef = enclosingReference.getChild(containerRefName); |
| |
| var accessorCount = _reader.readUInt30(); |
| for (var i = 0; i < accessorCount; i++) { |
| var accessor = _readPropertyAccessorElement( |
| unitElement, |
| enclosingElement, |
| enclosingReference, |
| ); |
| accessors.add(accessor); |
| |
| var name = accessor.displayName; |
| var isGetter = accessor.isGetter; |
| |
| var reference = containerRef.getChild(name); |
| |
| PropertyInducingElementImpl property; |
| if (enclosingElement is CompilationUnitElementImpl) { |
| var existing = reference.element; |
| if (existing is TopLevelVariableElementImpl) { |
| property = existing; |
| } else { |
| var field = TopLevelVariableElementImpl(name, -1); |
| field.isFinal = true; |
| property = field; |
| } |
| } else { |
| var existing = reference.element; |
| if (existing is FieldElementImpl) { |
| property = existing; |
| } else { |
| var field = FieldElementImpl(name, -1); |
| field.isFinal = true; |
| field.isStatic = accessor.isStatic; |
| property = field; |
| } |
| } |
| |
| if (reference.element == null) { |
| reference.element = property; |
| properties.add(property); |
| |
| property.enclosingElement = enclosingElement; |
| property.isSynthetic = true; |
| } |
| |
| accessor.variable = property; |
| if (isGetter) { |
| property.getter = accessor; |
| } else { |
| property.setter = accessor; |
| if (property.isSynthetic) { |
| property.isFinal = false; |
| } |
| } |
| } |
| } |
| |
| TopLevelInferenceError? _readTopLevelInferenceError() { |
| var kindIndex = _reader.readByte(); |
| var kind = TopLevelInferenceErrorKind.values[kindIndex]; |
| if (kind == TopLevelInferenceErrorKind.none) { |
| return null; |
| } |
| return TopLevelInferenceError( |
| kind: kind, |
| arguments: _reader.readStringReferenceList(), |
| ); |
| } |
| |
| TopLevelVariableElementImpl _readTopLevelVariableElement( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var isConst = _reader.readBool(); |
| var reference = unitReference.getChild('@variable').getChild(name); |
| |
| TopLevelVariableElementImpl element; |
| if (isConst) { |
| element = ConstTopLevelVariableElementImpl(name, -1); |
| } else { |
| element = TopLevelVariableElementImpl(name, -1); |
| } |
| |
| var linkedData = TopLevelVariableElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| element.isConst = isConst; |
| TopLevelVariableElementFlags.read(_reader, element); |
| element.typeInferenceError = _readTopLevelInferenceError(); |
| element.createImplicitAccessors(unitReference, name); |
| |
| return element; |
| } |
| |
| void _readTopLevelVariables( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| List<PropertyAccessorElementImpl> accessors, |
| List<TopLevelVariableElementImpl> variables, |
| ) { |
| var variableElementCount = _reader.readUInt30(); |
| for (var i = 0; i < variableElementCount; i++) { |
| var variable = _readTopLevelVariableElement(unitElement, unitReference); |
| variables.add(variable); |
| |
| var getter = variable.getter; |
| if (getter is PropertyAccessorElementImpl) { |
| accessors.add(getter); |
| } |
| |
| var setter = variable.setter; |
| if (setter is PropertyAccessorElementImpl) { |
| accessors.add(setter); |
| } |
| } |
| } |
| |
| TypeAliasElementImpl _readTypeAliasElement( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var name = _reader.readStringReference(); |
| var reference = unitReference.getChild('@typeAlias').getChild(name); |
| |
| var isFunctionTypeAliasBased = _reader.readBool(); |
| |
| TypeAliasElementImpl element; |
| if (isFunctionTypeAliasBased) { |
| element = |
| // ignore: deprecated_member_use_from_same_package |
| FunctionTypeAliasElementImpl(name, -1); |
| } else { |
| element = TypeAliasElementImpl(name, -1); |
| } |
| |
| var linkedData = TypeAliasElementLinkedData( |
| reference: reference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ); |
| element.setLinkedData(reference, linkedData); |
| |
| element.isFunctionTypeAliasBased = isFunctionTypeAliasBased; |
| TypeAliasElementFlags.read(_reader, element); |
| |
| element.typeParameters = _readTypeParameters(); |
| |
| return element; |
| } |
| |
| void _readTypeAliases( |
| CompilationUnitElementImpl unitElement, |
| Reference unitReference, |
| ) { |
| var length = _reader.readUInt30(); |
| unitElement.typeAliases = List.generate(length, (_) { |
| return _readTypeAliasElement(unitElement, unitReference); |
| }); |
| } |
| |
| List<TypeParameterElementImpl> _readTypeParameters() { |
| var length = _reader.readUInt30(); |
| return List.generate(length, (_) { |
| var name = _reader.readStringReference(); |
| var varianceEncoding = _reader.readByte(); |
| var variance = _decodeVariance(varianceEncoding); |
| var element = TypeParameterElementImpl(name, -1); |
| element.variance = variance; |
| return element; |
| }); |
| } |
| |
| CompilationUnitElementImpl _readUnitElement({ |
| required SourceFactory sourceFactory, |
| required Reference unitContainerRef, |
| required LibraryElementImpl libraryElement, |
| required Source librarySource, |
| }) { |
| var resolutionOffset = _baseResolutionOffset + _reader.readUInt30(); |
| var unitUriStr = _reader.readStringReference(); |
| var unitUri = Uri.parse(unitUriStr); |
| var unitSource = sourceFactory.forUri2(unitUri)!; |
| |
| var unitElement = CompilationUnitElementImpl(); |
| unitElement.source = unitSource; |
| unitElement.librarySource = librarySource; |
| |
| var unitReference = unitContainerRef.getChild(unitUriStr); |
| unitElement.setLinkedData( |
| unitReference, |
| CompilationUnitElementLinkedData( |
| reference: unitReference, |
| libraryReader: this, |
| unitElement: unitElement, |
| offset: resolutionOffset, |
| ), |
| ); |
| |
| unitElement.uri = _reader.readOptionalStringReference(); |
| unitElement.isSynthetic = _reader.readBool(); |
| |
| _readClasses(unitElement, unitReference); |
| _readEnums(unitElement, unitReference); |
| _readExtensions(unitElement, unitReference); |
| _readFunctions(unitElement, unitReference); |
| _readMixins(unitElement, unitReference); |
| _readTypeAliases(unitElement, unitReference); |
| |
| var accessors = <PropertyAccessorElementImpl>[]; |
| var variables = <TopLevelVariableElementImpl>[]; |
| _readTopLevelVariables(unitElement, unitReference, accessors, variables); |
| _readPropertyAccessors(unitElement, unitElement, unitReference, accessors, |
| variables, '@variable'); |
| unitElement.accessors = accessors; |
| unitElement.topLevelVariables = variables; |
| return unitElement; |
| } |
| |
| static Variance? _decodeVariance(int index) { |
| var tag = TypeParameterVarianceTag.values[index]; |
| switch (tag) { |
| case TypeParameterVarianceTag.legacy: |
| return null; |
| case TypeParameterVarianceTag.unrelated: |
| return Variance.unrelated; |
| case TypeParameterVarianceTag.covariant: |
| return Variance.covariant; |
| case TypeParameterVarianceTag.contravariant: |
| return Variance.contravariant; |
| case TypeParameterVarianceTag.invariant: |
| return Variance.invariant; |
| } |
| } |
| } |
| |
| class MethodElementLinkedData extends ElementLinkedData<MethodElementImpl> { |
| MethodElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| _addEnclosingElementTypeParameters(reader, element); |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| _readTypeParameters(reader, element.typeParameters); |
| _readFormalParameters(reader, element.parameters); |
| element.returnType = reader.readRequiredType(); |
| } |
| } |
| |
| class MixinElementLinkedData extends ElementLinkedData<MixinElementImpl> { |
| MixinElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: element.enclosingElement, |
| ); |
| _readTypeParameters(reader, element.typeParameters); |
| element.superclassConstraints = reader._readInterfaceTypeList(); |
| element.interfaces = reader._readInterfaceTypeList(); |
| } |
| } |
| |
| class PropertyAccessorElementLinkedData |
| extends ElementLinkedData<PropertyAccessorElementImpl> { |
| PropertyAccessorElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| _addEnclosingElementTypeParameters(reader, element); |
| |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| |
| element.returnType = reader.readRequiredType(); |
| _readFormalParameters(reader, element.parameters); |
| } |
| } |
| |
| /// Helper for reading elements and types from their binary encoding. |
| class ResolutionReader { |
| final LinkedElementFactory _elementFactory; |
| final _ReferenceReader _referenceReader; |
| final SummaryDataReader _reader; |
| |
| /// The stack of [TypeParameterElement]s and [ParameterElement] that are |
| /// available in the scope of [readElement] and [readType]. |
| /// |
| /// This stack is shared with the client of the reader, and update mostly |
| /// by the client. However it is also updated during [_readFunctionType]. |
| final List<Element> _localElements = []; |
| |
| ResolutionReader( |
| this._elementFactory, |
| this._referenceReader, |
| this._reader, |
| ); |
| |
| int readByte() { |
| return _reader.readByte(); |
| } |
| |
| double readDouble() { |
| return _reader.readDouble(); |
| } |
| |
| Element? readElement() { |
| var memberFlags = _reader.readByte(); |
| var element = _readRawElement(); |
| |
| if (element == null) { |
| return null; |
| } |
| |
| if (memberFlags == Tag.RawElement) { |
| return element; |
| } |
| |
| if (memberFlags == Tag.MemberLegacyWithTypeArguments || |
| memberFlags == Tag.MemberWithTypeArguments) { |
| var arguments = _readTypeList(); |
| // TODO(scheglov) why to check for empty? If we have this flags. |
| if (arguments.isNotEmpty) { |
| var typeParameters = |
| (element.enclosingElement as TypeParameterizedElement) |
| .typeParameters; |
| var substitution = Substitution.fromPairs(typeParameters, arguments); |
| element = |
| ExecutableMember.from2(element as ExecutableElement, substitution); |
| } |
| } |
| |
| if (memberFlags == Tag.MemberLegacyWithTypeArguments) { |
| return Member.legacy(element); |
| } |
| |
| if (memberFlags == Tag.MemberWithTypeArguments) { |
| return element; |
| } |
| |
| throw UnimplementedError('memberFlags: $memberFlags'); |
| } |
| |
| DartType readRequiredType() { |
| return readType()!; |
| } |
| |
| String readStringReference() { |
| return _reader.readStringReference(); |
| } |
| |
| List<String> readStringReferenceList() { |
| return _reader.readStringReferenceList(); |
| } |
| |
| DartType? readType() { |
| var tag = _reader.readByte(); |
| if (tag == Tag.NullType) { |
| return null; |
| } else if (tag == Tag.DynamicType) { |
| var type = DynamicTypeImpl.instance; |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.FunctionType) { |
| var type = _readFunctionType(); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.InterfaceType) { |
| var element = readElement() as ClassElement; |
| var typeArguments = _readTypeList(); |
| var nullability = _readNullability(); |
| var type = InterfaceTypeImpl( |
| element: element, |
| typeArguments: typeArguments, |
| nullabilitySuffix: nullability, |
| ); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.InterfaceType_noTypeArguments_none) { |
| var element = readElement() as ClassElement; |
| var type = InterfaceTypeImpl( |
| element: element, |
| typeArguments: const <DartType>[], |
| nullabilitySuffix: NullabilitySuffix.none, |
| ); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.InterfaceType_noTypeArguments_question) { |
| var element = readElement() as ClassElement; |
| var type = InterfaceTypeImpl( |
| element: element, |
| typeArguments: const <DartType>[], |
| nullabilitySuffix: NullabilitySuffix.question, |
| ); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.InterfaceType_noTypeArguments_star) { |
| var element = readElement() as ClassElement; |
| var type = InterfaceTypeImpl( |
| element: element, |
| typeArguments: const <DartType>[], |
| nullabilitySuffix: NullabilitySuffix.star, |
| ); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.NeverType) { |
| var nullability = _readNullability(); |
| var type = NeverTypeImpl.instance.withNullability(nullability); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.TypeParameterType) { |
| var element = readElement() as TypeParameterElement; |
| var nullability = _readNullability(); |
| var type = TypeParameterTypeImpl( |
| element: element, |
| nullabilitySuffix: nullability, |
| ); |
| return _readAliasElementArguments(type); |
| } else if (tag == Tag.VoidType) { |
| var type = VoidTypeImpl.instance; |
| return _readAliasElementArguments(type); |
| } else { |
| throw UnimplementedError('$tag'); |
| } |
| } |
| |
| List<T> readTypedList<T>(T Function() read) { |
| var length = readUInt30(); |
| return List<T>.generate(length, (_) { |
| return read(); |
| }); |
| } |
| |
| int readUInt30() { |
| return _reader.readUInt30(); |
| } |
| |
| int readUInt32() { |
| return _reader.readUInt32(); |
| } |
| |
| void _addFormalParameters(List<ParameterElement> parameters) { |
| for (var parameter in parameters) { |
| _localElements.add(parameter); |
| } |
| } |
| |
| void _addTypeParameters(List<TypeParameterElement> typeParameters) { |
| for (var typeParameter in typeParameters) { |
| _localElements.add(typeParameter); |
| } |
| } |
| |
| ElementImpl? _readAliasedElement(CompilationUnitElementImpl unitElement) { |
| var tag = _reader.readByte(); |
| if (tag == AliasedElementTag.nothing) { |
| return null; |
| } else if (tag == AliasedElementTag.genericFunctionElement) { |
| var typeParameters = _readTypeParameters(unitElement); |
| var formalParameters = _readFormalParameters(unitElement); |
| var returnType = readRequiredType(); |
| |
| _localElements.length -= typeParameters.length; |
| |
| return GenericFunctionTypeElementImpl.forOffset(-1) |
| ..typeParameters = typeParameters |
| ..parameters = formalParameters |
| ..returnType = returnType; |
| } else { |
| throw UnimplementedError('tag: $tag'); |
| } |
| } |
| |
| DartType _readAliasElementArguments(DartType type) { |
| var aliasElement = _readRawElement(); |
| if (aliasElement is TypeAliasElement) { |
| var aliasArguments = _readTypeList(); |
| if (type is DynamicType) { |
| // TODO(scheglov) add support for `dynamic` aliasing |
| return type; |
| } else if (type is FunctionType) { |
| return FunctionTypeImpl( |
| typeFormals: type.typeFormals, |
| parameters: type.parameters, |
| returnType: type.returnType, |
| nullabilitySuffix: type.nullabilitySuffix, |
| aliasElement: aliasElement, |
| aliasArguments: aliasArguments, |
| ); |
| } else if (type is InterfaceType) { |
| return InterfaceTypeImpl( |
| element: type.element, |
| typeArguments: type.typeArguments, |
| nullabilitySuffix: type.nullabilitySuffix, |
| aliasElement: aliasElement, |
| aliasArguments: aliasArguments, |
| ); |
| } else if (type is TypeParameterType) { |
| return TypeParameterTypeImpl( |
| element: type.element, |
| nullabilitySuffix: type.nullabilitySuffix, |
| aliasElement: aliasElement, |
| aliasArguments: aliasArguments, |
| ); |
| } else if (type is VoidType) { |
| // TODO(scheglov) add support for `void` aliasing |
| return type; |
| } else { |
| throw UnimplementedError('${type.runtimeType}'); |
| } |
| } |
| return type; |
| } |
| |
| List<ElementAnnotationImpl> _readAnnotationList({ |
| required CompilationUnitElementImpl unitElement, |
| }) { |
| var length = _reader.readUInt30(); |
| if (length == 0) { |
| return const <ElementAnnotationImpl>[]; |
| } |
| return List.generate(length, (_) { |
| var ast = _readRequiredNode() as Annotation; |
| return ElementAnnotationImpl(unitElement) |
| ..annotationAst = ast |
| ..element = ast.element; |
| }); |
| } |
| |
| List<ParameterElementImpl> _readFormalParameters( |
| CompilationUnitElementImpl? unitElement, |
| ) { |
| var formalParameterCount = _reader.readUInt30(); |
| return List.generate(formalParameterCount, (_) { |
| var kindIndex = _reader.readByte(); |
| var kind = _formalParameterKind(kindIndex); |
| var hasImplicitType = _reader.readBool(); |
| var isInitializingFormal = _reader.readBool(); |
| var typeParameters = _readTypeParameters(unitElement); |
| var type = readRequiredType(); |
| var name = readStringReference(); |
| if (kind.isRequiredPositional) { |
| ParameterElementImpl element; |
| if (isInitializingFormal) { |
| element = FieldFormalParameterElementImpl(name, -1) |
| ..parameterKind = kind |
| ..type = type; |
| } else { |
| element = ParameterElementImpl(name, -1) |
| ..parameterKind = kind |
| ..type = type; |
| } |
| element.hasImplicitType = hasImplicitType; |
| element.typeParameters = typeParameters; |
| element.parameters = _readFormalParameters(unitElement); |
| // TODO(scheglov) reuse for formal parameters |
| _localElements.length -= typeParameters.length; |
| if (unitElement != null) { |
| element.metadata = _readAnnotationList(unitElement: unitElement); |
| } |
| return element; |
| } else { |
| var element = DefaultParameterElementImpl(name, -1) |
| ..parameterKind = kind |
| ..type = type; |
| element.hasImplicitType = hasImplicitType; |
| element.typeParameters = typeParameters; |
| element.parameters = _readFormalParameters(unitElement); |
| // TODO(scheglov) reuse for formal parameters |
| _localElements.length -= typeParameters.length; |
| if (unitElement != null) { |
| element.metadata = _readAnnotationList(unitElement: unitElement); |
| } |
| return element; |
| } |
| }); |
| } |
| |
| /// TODO(scheglov) Optimize for write/read of types without type parameters. |
| FunctionType _readFunctionType() { |
| // TODO(scheglov) reuse for formal parameters |
| var typeParameters = _readTypeParameters(null); |
| var returnType = readRequiredType(); |
| var formalParameters = _readFormalParameters(null); |
| |
| var nullability = _readNullability(); |
| |
| _localElements.length -= typeParameters.length; |
| |
| return FunctionTypeImpl( |
| typeFormals: typeParameters, |
| parameters: formalParameters, |
| returnType: returnType, |
| nullabilitySuffix: nullability, |
| ); |
| } |
| |
| InterfaceType _readInterfaceType() { |
| var element = _readRawElement() as ClassElement; |
| var typeArguments = _readTypeList(); |
| var nullability = _readNullability(); |
| var type = InterfaceTypeImpl( |
| element: element, |
| typeArguments: typeArguments, |
| nullabilitySuffix: nullability, |
| ); |
| return type; |
| } |
| |
| List<InterfaceType> _readInterfaceTypeList() { |
| var length = _reader.readUInt30(); |
| if (length == 0) { |
| return const <InterfaceType>[]; |
| } |
| return List.generate(length, (_) => _readInterfaceType()); |
| } |
| |
| List<T> _readNodeList<T>() { |
| var length = _reader.readUInt30(); |
| return List<T>.generate(length, (_) { |
| return _readRequiredNode() as T; |
| }); |
| } |
| |
| NullabilitySuffix _readNullability() { |
| var index = _reader.readByte(); |
| return NullabilitySuffix.values[index]; |
| } |
| |
| ExpressionImpl? _readOptionalExpression() { |
| if (_reader.readBool()) { |
| return _readRequiredNode() as ExpressionImpl; |
| } |
| } |
| |
| InterfaceType? _readOptionalInterfaceType() { |
| var hasSuperType = _reader.readByte() != 0; |
| if (hasSuperType) { |
| return _readInterfaceType(); |
| } |
| } |
| |
| Element? _readRawElement() { |
| var index = _reader.readUInt30(); |
| |
| if ((index & 0x1) == 0x1) { |
| return _localElements[index >> 1]; |
| } |
| |
| var referenceIndex = index >> 1; |
| var reference = _referenceReader.referenceOfIndex(referenceIndex); |
| |
| return _elementFactory.elementOfReference(reference); |
| } |
| |
| AstNode _readRequiredNode() { |
| var astReader = AstBinaryReader(reader: this); |
| return astReader.readNode(); |
| } |
| |
| List<DartType> _readTypeList() { |
| var types = <DartType>[]; |
| var length = _reader.readUInt30(); |
| for (var i = 0; i < length; i++) { |
| var argument = readType()!; |
| types.add(argument); |
| } |
| return types; |
| } |
| |
| List<TypeParameterElementImpl> _readTypeParameters( |
| CompilationUnitElementImpl? unitElement, |
| ) { |
| var typeParameterCount = _reader.readUInt30(); |
| var typeParameters = List.generate(typeParameterCount, (_) { |
| var name = readStringReference(); |
| var typeParameter = TypeParameterElementImpl(name, -1); |
| _localElements.add(typeParameter); |
| return typeParameter; |
| }); |
| |
| for (var typeParameter in typeParameters) { |
| typeParameter.bound = readType(); |
| if (unitElement != null) { |
| typeParameter.metadata = _readAnnotationList(unitElement: unitElement); |
| } |
| } |
| return typeParameters; |
| } |
| |
| static ParameterKind _formalParameterKind(int encoding) { |
| if (encoding == Tag.ParameterKindRequiredPositional) { |
| return ParameterKind.REQUIRED; |
| } else if (encoding == Tag.ParameterKindOptionalPositional) { |
| return ParameterKind.POSITIONAL; |
| } else if (encoding == Tag.ParameterKindRequiredNamed) { |
| return ParameterKind.NAMED_REQUIRED; |
| } else if (encoding == Tag.ParameterKindOptionalNamed) { |
| return ParameterKind.NAMED; |
| } else { |
| throw StateError('Unexpected parameter kind encoding: $encoding'); |
| } |
| } |
| } |
| |
| class TopLevelVariableElementLinkedData |
| extends ElementLinkedData<TopLevelVariableElementImpl> { |
| TopLevelVariableElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| element.type = reader.readRequiredType(); |
| if (element is ConstTopLevelVariableElementImpl) { |
| var initializer = reader._readOptionalExpression(); |
| if (initializer != null) { |
| element.constantInitializer = initializer; |
| ConstantContextForExpressionImpl(initializer); |
| } |
| } |
| } |
| } |
| |
| class TypeAliasElementLinkedData |
| extends ElementLinkedData<TypeAliasElementImpl> { |
| TypeAliasElementLinkedData({ |
| required Reference reference, |
| required LibraryReader libraryReader, |
| required CompilationUnitElementImpl unitElement, |
| required int offset, |
| }) : super(reference, libraryReader, unitElement, offset); |
| |
| @override |
| void _read(element, reader) { |
| element.metadata = reader._readAnnotationList( |
| unitElement: unitElement, |
| ); |
| _readTypeParameters(reader, element.typeParameters); |
| element.aliasedElement = reader._readAliasedElement(unitElement); |
| element.aliasedType = reader.readRequiredType(); |
| } |
| } |
| |
| /// Information that we need to know about each library before reading it, |
| /// and without reading it. |
| /// |
| /// Specifically, the [offset] allows us to know the location of each library, |
| /// so that when we need to read this library, we know where it starts without |
| /// reading previous libraries. |
| class _LibraryHeader { |
| final String uriStr; |
| final int offset; |
| |
| /// We don't read class members when reading libraries, by performance |
| /// reasons - in many cases only some classes of a library are used. But |
| /// we need to know how much data to skip for each class. |
| final Uint32List classMembersLengths; |
| |
| _LibraryHeader({ |
| required this.uriStr, |
| required this.offset, |
| required this.classMembersLengths, |
| }); |
| } |
| |
| class _ReferenceReader { |
| final LinkedElementFactory elementFactory; |
| final SummaryDataReader _reader; |
| late final Uint32List _parents; |
| late final Uint32List _names; |
| late final List<Reference?> _references; |
| |
| _ReferenceReader(this.elementFactory, this._reader, int offset) { |
| _reader.offset = offset; |
| _parents = _reader.readUInt30List(); |
| _names = _reader.readUInt30List(); |
| assert(_parents.length == _names.length); |
| |
| _references = List.filled(_names.length, null); |
| } |
| |
| Reference referenceOfIndex(int index) { |
| var reference = _references[index]; |
| if (reference != null) { |
| return reference; |
| } |
| |
| if (index == 0) { |
| reference = elementFactory.rootReference; |
| _references[index] = reference; |
| return reference; |
| } |
| |
| var nameIndex = _names[index]; |
| var name = _reader.stringOfIndex(nameIndex); |
| |
| var parentIndex = _parents[index]; |
| var parent = referenceOfIndex(parentIndex); |
| |
| reference = parent.getChild(name); |
| _references[index] = reference; |
| |
| return reference; |
| } |
| } |