| Index: sdk/lib/_internal/compiler/implementation/mirrors/dart2js_mirrors.dart
|
| diff --git a/sdk/lib/_internal/compiler/implementation/mirrors/dart2js_mirrors.dart b/sdk/lib/_internal/compiler/implementation/mirrors/dart2js_mirrors.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..ed26e941c870ae567d4f7eebf1794a2be3ef06a4
|
| --- /dev/null
|
| +++ b/sdk/lib/_internal/compiler/implementation/mirrors/dart2js_mirrors.dart
|
| @@ -0,0 +1,468 @@
|
| +// Copyright (c) 2012, 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 dart2js.mirrors;
|
| +
|
| +import 'dart:collection' show UnmodifiableListView;
|
| +
|
| +import '../elements/elements.dart';
|
| +import '../scanner/scannerlib.dart';
|
| +import '../resolution/resolution.dart' show Scope;
|
| +import '../dart2jslib.dart';
|
| +import '../dart_types.dart';
|
| +import '../tree/tree.dart';
|
| +import '../util/util.dart' show Spannable, Link, LinkBuilder;
|
| +import '../util/characters.dart' show $CR, $LF;
|
| +
|
| +import 'source_mirrors.dart';
|
| +import 'mirrors_util.dart';
|
| +import 'util.dart';
|
| +
|
| +part 'dart2js_library_mirror.dart';
|
| +part 'dart2js_type_mirrors.dart';
|
| +part 'dart2js_member_mirrors.dart';
|
| +part 'dart2js_instance_mirrors.dart';
|
| +
|
| +//------------------------------------------------------------------------------
|
| +// Utility types and functions for the dart2js mirror system
|
| +//------------------------------------------------------------------------------
|
| +
|
| +bool _isPrivate(String name) {
|
| + return name.startsWith('_');
|
| +}
|
| +
|
| +List<ParameterMirror> _parametersFromFunctionSignature(
|
| + Dart2JsDeclarationMirror owner,
|
| + FunctionSignature signature) {
|
| + var parameters = <ParameterMirror>[];
|
| + Link<Element> link = signature.requiredParameters;
|
| + while (!link.isEmpty) {
|
| + parameters.add(new Dart2JsParameterMirror(
|
| + owner, link.head, isOptional: false, isNamed: false));
|
| + link = link.tail;
|
| + }
|
| + link = signature.optionalParameters;
|
| + bool isNamed = signature.optionalParametersAreNamed;
|
| + while (!link.isEmpty) {
|
| + parameters.add(new Dart2JsParameterMirror(
|
| + owner, link.head, isOptional: true, isNamed: isNamed));
|
| + link = link.tail;
|
| + }
|
| + return parameters;
|
| +}
|
| +
|
| +MethodMirror _convertElementMethodToMethodMirror(
|
| + Dart2JsDeclarationMirror library, Element element) {
|
| + if (element is FunctionElement) {
|
| + return new Dart2JsMethodMirror(library, element);
|
| + } else {
|
| + return null;
|
| + }
|
| +}
|
| +
|
| +//------------------------------------------------------------------------------
|
| +// Dart2Js specific extensions of mirror interfaces
|
| +//------------------------------------------------------------------------------
|
| +
|
| +abstract class Dart2JsMirror implements Mirror {
|
| + Dart2JsMirrorSystem get mirrorSystem;
|
| +}
|
| +
|
| +abstract class Dart2JsDeclarationMirror extends Dart2JsMirror
|
| + implements DeclarationSourceMirror {
|
| +
|
| + bool get isTopLevel => owner != null && owner is LibraryMirror;
|
| +
|
| + bool get isPrivate => _isPrivate(_simpleNameString);
|
| +
|
| + String get _simpleNameString;
|
| +
|
| + String get _qualifiedNameString {
|
| + var parent = owner;
|
| + if (parent is Dart2JsDeclarationMirror) {
|
| + return '${parent._qualifiedNameString}.${_simpleNameString}';
|
| + }
|
| + assert(parent == null);
|
| + return _simpleNameString;
|
| + }
|
| +
|
| + Symbol get simpleName => symbolOf(_simpleNameString, getLibrary(this));
|
| +
|
| + Symbol get qualifiedName => symbolOf(_qualifiedNameString, getLibrary(this));
|
| +
|
| + /**
|
| + * Returns the first token for the source of this declaration, not including
|
| + * metadata annotations.
|
| + */
|
| + Token getBeginToken();
|
| +
|
| + /**
|
| + * Returns the last token for the source of this declaration.
|
| + */
|
| + Token getEndToken();
|
| +
|
| + /**
|
| + * Returns the script for the source of this declaration.
|
| + */
|
| + Script getScript();
|
| +
|
| + /// Returns the type mirror for [type] in the context of this declaration.
|
| + TypeMirror _getTypeMirror(DartType type, [FunctionSignature signature]) {
|
| + return mirrorSystem._convertTypeToTypeMirror(type, signature);
|
| + }
|
| +
|
| + /// Returns a list of the declaration mirrorSystem for [element].
|
| + Iterable<Dart2JsMemberMirror> _getDeclarationMirrors(Element element) {
|
| + if (element.isSynthesized) {
|
| + return const <Dart2JsMemberMirror>[];
|
| + } else if (element is VariableElement) {
|
| + return <Dart2JsMemberMirror>[new Dart2JsFieldMirror(this, element)];
|
| + } else if (element is FunctionElement) {
|
| + return <Dart2JsMemberMirror>[new Dart2JsMethodMirror(this, element)];
|
| + } else if (element is AbstractFieldElement) {
|
| + var members = <Dart2JsMemberMirror>[];
|
| + AbstractFieldElement field = element;
|
| + if (field.getter != null) {
|
| + members.add(new Dart2JsMethodMirror(this, field.getter));
|
| + }
|
| + if (field.setter != null) {
|
| + members.add(new Dart2JsMethodMirror(this, field.setter));
|
| + }
|
| + return members;
|
| + }
|
| + mirrorSystem.compiler.internalError(
|
| + "Unexpected member type $element ${element.kind}");
|
| + }
|
| +
|
| +}
|
| +
|
| +abstract class Dart2JsElementMirror extends Dart2JsDeclarationMirror {
|
| + final Dart2JsMirrorSystem mirrorSystem;
|
| + final Element _element;
|
| + List<InstanceMirror> _metadata;
|
| +
|
| + Dart2JsElementMirror(this.mirrorSystem, this._element) {
|
| + assert (mirrorSystem != null);
|
| + assert (_element != null);
|
| + }
|
| +
|
| + /**
|
| + * Returns the element to be used to determine the begin token of this
|
| + * declaration and the metadata associated with this declaration.
|
| + *
|
| + * This indirection is needed to use the [VariableListElement] as the location
|
| + * for type and metadata information on a [VariableElement].
|
| + */
|
| + Element get _beginElement => _element;
|
| +
|
| + String get _simpleNameString => _element.name;
|
| +
|
| + bool get isNameSynthetic => false;
|
| +
|
| + /**
|
| + * Computes the first token for this declaration using the begin token of the
|
| + * element node or element position as indicator.
|
| + */
|
| + Token getBeginToken() {
|
| + // TODO(johnniwinther): Avoid calling [parseNode].
|
| + Node node = _beginElement.parseNode(mirrorSystem.compiler);
|
| + if (node == null) {
|
| + return _beginElement.position();
|
| + }
|
| + return node.getBeginToken();
|
| + }
|
| +
|
| + /**
|
| + * Computes the last token for this declaration using the end token of the
|
| + * element node or element position as indicator.
|
| + */
|
| + Token getEndToken() {
|
| + // TODO(johnniwinther): Avoid calling [parseNode].
|
| + Node node = _element.parseNode(mirrorSystem.compiler);
|
| + if (node == null) {
|
| + return _element.position();
|
| + }
|
| + return node.getEndToken();
|
| + }
|
| +
|
| + /**
|
| + * Returns the first token for the source of this declaration, including
|
| + * metadata annotations.
|
| + */
|
| + Token getFirstToken() {
|
| + if (!_beginElement.metadata.isEmpty) {
|
| + for (MetadataAnnotation metadata in _beginElement.metadata) {
|
| + if (metadata.beginToken != null) {
|
| + return metadata.beginToken;
|
| + }
|
| + }
|
| + }
|
| + return getBeginToken();
|
| + }
|
| +
|
| + Script getScript() => _element.getCompilationUnit().script;
|
| +
|
| + SourceLocation get location {
|
| + Token beginToken = getFirstToken();
|
| + Script script = getScript();
|
| + SourceSpan span;
|
| + if (beginToken == null) {
|
| + span = new SourceSpan(script.uri, 0, 0);
|
| + } else {
|
| + Token endToken = getEndToken();
|
| + span = mirrorSystem.compiler.spanFromTokens(
|
| + beginToken, endToken, script.uri);
|
| + }
|
| + return new Dart2JsSourceLocation(script, span);
|
| + }
|
| +
|
| + String toString() => _element.toString();
|
| +
|
| + void _appendCommentTokens(Token commentToken) {
|
| + while (commentToken != null && commentToken.kind == COMMENT_TOKEN) {
|
| + _metadata.add(new Dart2JsCommentInstanceMirror(
|
| + mirrorSystem, commentToken.value));
|
| + commentToken = commentToken.next;
|
| + }
|
| + }
|
| +
|
| + List<InstanceMirror> get metadata {
|
| + if (_metadata == null) {
|
| + _metadata = <InstanceMirror>[];
|
| + for (MetadataAnnotation metadata in _element.metadata) {
|
| + _appendCommentTokens(
|
| + mirrorSystem.compiler.commentMap[metadata.beginToken]);
|
| + metadata.ensureResolved(mirrorSystem.compiler);
|
| + _metadata.add(
|
| + _convertConstantToInstanceMirror(mirrorSystem, metadata.value));
|
| + }
|
| + _appendCommentTokens(mirrorSystem.compiler.commentMap[getBeginToken()]);
|
| + }
|
| + // TODO(johnniwinther): Return an unmodifiable list instead.
|
| + return new List<InstanceMirror>.from(_metadata);
|
| + }
|
| +
|
| + DeclarationMirror lookupInScope(String name) {
|
| + // TODO(11653): Support lookup of constructors.
|
| + Scope scope = _element.buildScope();
|
| + Element result;
|
| + int index = name.indexOf('.');
|
| + if (index != -1) {
|
| + // Lookup [: prefix.id :].
|
| + String prefix = name.substring(0, index);
|
| + String id = name.substring(index+1);
|
| + result = scope.lookup(prefix);
|
| + if (result != null && result.isPrefix()) {
|
| + PrefixElement prefix = result;
|
| + result = prefix.lookupLocalMember(id);
|
| + } else {
|
| + result = null;
|
| + }
|
| + } else {
|
| + // Lookup [: id :].
|
| + result = scope.lookup(name);
|
| + }
|
| + if (result == null || result.isPrefix()) return null;
|
| + return _convertElementToDeclarationMirror(mirrorSystem, result);
|
| + }
|
| +
|
| + bool operator ==(var other) {
|
| + if (identical(this, other)) return true;
|
| + if (other == null) return false;
|
| + if (other is! Dart2JsElementMirror) return false;
|
| + return _element == other._element &&
|
| + owner == other.owner;
|
| + }
|
| +
|
| + int get hashCode {
|
| + return 13 * _element.hashCode + 17 * owner.hashCode;
|
| + }
|
| +}
|
| +
|
| +//------------------------------------------------------------------------------
|
| +// Mirror system implementation.
|
| +//------------------------------------------------------------------------------
|
| +
|
| +class Dart2JsMirrorSystem extends MirrorSystem {
|
| + final Compiler compiler;
|
| + Map<Uri, Dart2JsLibraryMirror> _libraries;
|
| + Map<LibraryElement, Dart2JsLibraryMirror> _libraryMap;
|
| +
|
| + Dart2JsMirrorSystem(this.compiler)
|
| + : _libraryMap = new Map<LibraryElement, Dart2JsLibraryMirror>();
|
| +
|
| + IsolateMirror get isolate => null;
|
| +
|
| + void _ensureLibraries() {
|
| + if (_libraries == null) {
|
| + _libraries = new Map<Uri, Dart2JsLibraryMirror>();
|
| + compiler.libraries.forEach((_, LibraryElement v) {
|
| + var mirror = new Dart2JsLibraryMirror(mirrorSystem, v);
|
| + _libraries[mirror.uri] = mirror;
|
| + _libraryMap[v] = mirror;
|
| + });
|
| + }
|
| + }
|
| +
|
| + Map<Uri, LibraryMirror> get libraries {
|
| + _ensureLibraries();
|
| + return new FilteredImmutableMap<Uri, LibraryMirror>(_libraries,
|
| + (library) => new bool.fromEnvironment("list_all_libraries") ||
|
| + !library._element.isInternalLibrary);
|
| + }
|
| +
|
| + Dart2JsLibraryMirror _getLibrary(LibraryElement element) =>
|
| + _libraryMap[element];
|
| +
|
| + Dart2JsMirrorSystem get mirrorSystem => this;
|
| +
|
| + TypeMirror get dynamicType =>
|
| + _convertTypeToTypeMirror(compiler.types.dynamicType);
|
| +
|
| + TypeMirror get voidType =>
|
| + _convertTypeToTypeMirror(compiler.types.voidType);
|
| +
|
| + TypeMirror _convertTypeToTypeMirror(DartType type,
|
| + [FunctionSignature signature]) {
|
| + assert(type != null);
|
| + if (type.treatAsDynamic) {
|
| + return new Dart2JsDynamicMirror(this, type);
|
| + } else if (type is InterfaceType) {
|
| + if (type.typeArguments.isEmpty) {
|
| + return _getTypeDeclarationMirror(type.element);
|
| + } else {
|
| + return new Dart2JsInterfaceTypeMirror(this, type);
|
| + }
|
| + } else if (type is TypeVariableType) {
|
| + return new Dart2JsTypeVariableMirror(this, type);
|
| + } else if (type is FunctionType) {
|
| + return new Dart2JsFunctionTypeMirror(this, type, signature);
|
| + } else if (type is VoidType) {
|
| + return new Dart2JsVoidMirror(this, type);
|
| + } else if (type is TypedefType) {
|
| + if (type.typeArguments.isEmpty) {
|
| + return _getTypeDeclarationMirror(type.element);
|
| + } else {
|
| + return new Dart2JsTypedefMirror(this, type);
|
| + }
|
| + }
|
| + compiler.internalError("Unexpected type $type of kind ${type.kind}");
|
| + }
|
| +
|
| + DeclarationMirror _getTypeDeclarationMirror(Element element) {
|
| + if (element.isClass()) {
|
| + return new Dart2JsClassDeclarationMirror(
|
| + this, element.computeType(compiler));
|
| + } else if (element.isTypedef()) {
|
| + return new Dart2JsTypedefDeclarationMirror(this,
|
| + element.computeType(compiler));
|
| + }
|
| + }
|
| +}
|
| +
|
| +abstract class ContainerMixin {
|
| + Map<Symbol, DeclarationMirror> _declarations;
|
| +
|
| + void _ensureDeclarations() {
|
| + if (_declarations == null) {
|
| + _declarations = <Symbol, DeclarationMirror>{};
|
| + _forEachElement((Element element) {
|
| + for (DeclarationMirror mirror in _getDeclarationMirrors(element)) {
|
| + assert(invariant(_element,
|
| + !_declarations.containsKey(mirror.simpleName),
|
| + message: "Declaration name '${nameOf(mirror)}' "
|
| + "is not unique in $_element."));
|
| + _declarations[mirror.simpleName] = mirror;
|
| + }
|
| + });
|
| + }
|
| + }
|
| +
|
| + Element get _element;
|
| +
|
| + void _forEachElement(f(Element element));
|
| +
|
| + Iterable<Dart2JsMemberMirror> _getDeclarationMirrors(Element element);
|
| +
|
| + Map<Symbol, DeclarationMirror> get declarations {
|
| + _ensureDeclarations();
|
| + return new ImmutableMapWrapper<Symbol, DeclarationMirror>(_declarations);
|
| + }
|
| +}
|
| +
|
| +/**
|
| + * Converts [element] into its corresponding [DeclarationMirror], if any.
|
| + *
|
| + * If [element] is an [AbstractFieldElement] the mirror for its getter is
|
| + * returned or, if not present, the mirror for its setter.
|
| + */
|
| +DeclarationMirror _convertElementToDeclarationMirror(Dart2JsMirrorSystem system,
|
| + Element element) {
|
| + if (element.isTypeVariable()) {
|
| + return new Dart2JsTypeVariableMirror(
|
| + system, element.computeType(system.compiler));
|
| + }
|
| +
|
| + Dart2JsLibraryMirror library = system._libraryMap[element.getLibrary()];
|
| + if (element.isLibrary()) return library;
|
| + if (element.isTypedef()) {
|
| + return new Dart2JsTypedefMirror.fromLibrary(
|
| + library, element.computeType(system.compiler));
|
| + }
|
| +
|
| + Dart2JsDeclarationMirror container = library;
|
| + if (element.getEnclosingClass() != null) {
|
| + container = system._getTypeDeclarationMirror(element.getEnclosingClass());
|
| + }
|
| + if (element.isClass()) return container;
|
| + if (element.isParameter()) {
|
| + Dart2JsMethodMirror method = _convertElementMethodToMethodMirror(
|
| + container, element.getOutermostEnclosingMemberOrTopLevel());
|
| + // TODO(johnniwinther): Find the right info for [isOptional] and [isNamed].
|
| + return new Dart2JsParameterMirror(
|
| + method, element, isOptional: false, isNamed: false);
|
| + }
|
| + Iterable<DeclarationMirror> members =
|
| + container._getDeclarationMirrors(element);
|
| + if (members.isEmpty) return null;
|
| + return members.first;
|
| +}
|
| +
|
| +/**
|
| + * Experimental API for accessing compilation units defined in a
|
| + * library.
|
| + */
|
| +// TODO(ahe): Superclasses? Is this really a mirror?
|
| +class Dart2JsCompilationUnitMirror extends Dart2JsMirror
|
| + with ContainerMixin {
|
| + final Dart2JsLibraryMirror _library;
|
| + final CompilationUnitElement _element;
|
| +
|
| + Dart2JsCompilationUnitMirror(this._element, this._library);
|
| +
|
| + Dart2JsMirrorSystem get mirrorSystem => _library.mirrorSystem;
|
| +
|
| + // TODO(johnniwinther): make sure that these are returned in declaration
|
| + // order.
|
| + void _forEachElement(f(Element element)) => _element.forEachLocalMember(f);
|
| +
|
| + Iterable<DeclarationMirror> _getDeclarationMirrors(Element element) =>
|
| + _library._getDeclarationMirrors(element);
|
| +
|
| + Uri get uri => _element.script.uri;
|
| +}
|
| +
|
| +/**
|
| + * Transitional class that allows access to features that have not yet
|
| + * made it to the mirror API.
|
| + *
|
| + * All API in this class is experimental.
|
| + */
|
| +class BackDoor {
|
| + /// Return the compilation units comprising [library].
|
| + static List<Mirror> compilationUnitsOf(Dart2JsLibraryMirror library) {
|
| + return library._element.compilationUnits.toList().map(
|
| + (cu) => new Dart2JsCompilationUnitMirror(cu, library)).toList();
|
| + }
|
| +}
|
|
|