blob: b575f556ebb78f832d3946c8081c429004500475 [file] [log] [blame]
// Copyright (c) 2018, 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.7
import 'package:compiler/src/common/elements.dart';
import 'package:compiler/src/deferred_load/output_unit.dart';
import 'package:compiler/src/elements/entities.dart';
import 'package:compiler/src/js/js.dart' as js;
import 'package:compiler/src/js_backend/namer.dart';
import 'package:compiler/src/js_emitter/model.dart';
import 'package:compiler/src/js_model/js_strategy.dart';
import 'package:expect/expect.dart';
ClassEntity lookupClass(JElementEnvironment elementEnvironment, String name) {
ClassEntity cls =
elementEnvironment.lookupClass(elementEnvironment.mainLibrary, name);
Expect.isNotNull(cls, "No class '$name' found in the main library.");
return cls;
}
MemberEntity lookupMember(JElementEnvironment elementEnvironment, String name) {
MemberEntity member;
int dotIndex = name.indexOf('.');
if (dotIndex != -1) {
String className = name.substring(0, dotIndex);
name = name.substring(dotIndex + 1);
ClassEntity cls = elementEnvironment.lookupClass(
elementEnvironment.mainLibrary, className);
Expect.isNotNull(cls, "No class '$className' found in the main library.");
member = elementEnvironment.lookupClassMember(cls, name);
member ??= elementEnvironment.lookupConstructor(cls, name);
Expect.isNotNull(member, "No member '$name' found in $cls");
} else {
member = elementEnvironment.lookupLibraryMember(
elementEnvironment.mainLibrary, name);
Expect.isNotNull(member, "No member '$name' found in the main library.");
}
return member;
}
class ProgramLookup {
final Program program;
final Namer namer;
ProgramLookup(JsBackendStrategy backendStrategy)
: this.program = backendStrategy.emitterTask.emitter.programForTesting,
this.namer = backendStrategy.namerForTesting;
Fragment getFragment(OutputUnit outputUnit) {
for (Fragment fragment in program.fragments) {
if (fragment.outputUnit == outputUnit) {
return fragment;
}
}
return null;
}
Map<LibraryEntity, LibraryData> libraryMap;
LibraryData getLibraryData(LibraryEntity element) {
if (libraryMap == null) {
libraryMap = <LibraryEntity, LibraryData>{};
for (Fragment fragment in program.fragments) {
for (Library library in fragment.libraries) {
assert(!libraryMap.containsKey(library.element));
libraryMap[library.element] = new LibraryData(library, fragment);
}
}
}
return libraryMap[element];
}
Library getLibrary(LibraryEntity element) {
return getLibraryData(element)?.library;
}
ClassData getClassData(ClassEntity element) {
return getLibraryData(element.library)?.getClassData(element);
}
Class getClass(ClassEntity element) {
return getClassData(element)?.cls;
}
Method getMethod(FunctionEntity function) {
if (function.enclosingClass != null) {
return getClassData(function.enclosingClass)?.getMethod(function);
} else {
return getLibraryData(function.library)?.getMethod(function);
}
}
Field getField(FieldEntity field) {
if (field.enclosingClass != null) {
return getClassData(field.enclosingClass)?.getField(field);
} else {
return getLibraryData(field.library)?.getField(field);
}
}
StaticField getStaticField(FieldEntity field) {
if (field.enclosingClass != null) {
return getClassData(field.enclosingClass)?.getStaticField(field);
} else {
return getLibraryData(field.library)?.getStaticField(field);
}
}
}
class LibraryData {
final Library library;
final Map<ClassEntity, ClassData> _classMap = {};
final Map<FunctionEntity, StaticMethod> _methodMap = {};
final Map<FieldEntity, Field> _fieldMap = {};
final Map<FieldEntity, StaticField> _staticFieldMap = {};
LibraryData(this.library, Fragment fragment) {
for (Class cls in library.classes) {
assert(!_classMap.containsKey(cls.element));
_classMap[cls.element] = new ClassData(cls);
}
for (StaticMethod method in library.statics) {
ClassEntity enclosingClass = method.element?.enclosingClass;
if (enclosingClass != null) {
ClassData data =
_classMap.putIfAbsent(enclosingClass, () => new ClassData(null));
assert(!data._methodMap.containsKey(method.element));
data._methodMap[method.element] = method;
} else if (method.element != null) {
assert(!_methodMap.containsKey(method.element));
_methodMap[method.element] = method;
}
}
void addStaticField(StaticField field) {
ClassEntity enclosingClass = field.element?.enclosingClass;
if (enclosingClass != null) {
ClassData data =
_classMap.putIfAbsent(enclosingClass, () => new ClassData(null));
assert(!data._fieldMap.containsKey(field.element));
data._staticFieldMap[field.element] = field;
} else if (field.element != null) {
assert(!_fieldMap.containsKey(field.element));
_staticFieldMap[field.element] = field;
}
}
for (StaticField field in fragment.staticNonFinalFields) {
addStaticField(field);
}
for (StaticField field in fragment.staticLazilyInitializedFields) {
addStaticField(field);
}
}
ClassData getClassData(ClassEntity element) {
return _classMap[element];
}
StaticMethod getMethod(FunctionEntity function) {
return _methodMap[function];
}
Field getField(FieldEntity field) {
return _fieldMap[field];
}
StaticField getStaticField(FieldEntity field) {
return _staticFieldMap[field];
}
@override
String toString() => 'LibraryData(library=$library,_classMap=$_classMap,'
'_methodMap=$_methodMap,_fieldMap=$_fieldMap)';
}
class ClassData {
final Class cls;
final Map<FunctionEntity, Method> _methodMap = {};
final Map<FieldEntity, Field> _fieldMap = {};
final Map<FieldEntity, StaticField> _staticFieldMap = {};
final Map<FieldEntity, StubMethod> _checkedSetterMap = {};
ClassData(this.cls) {
if (cls != null) {
for (Method method in cls.methods) {
assert(!_methodMap.containsKey(method.element));
_methodMap[method.element] = method;
}
for (Field field in cls.fields) {
assert(!_fieldMap.containsKey(field.element));
_fieldMap[field.element] = field;
}
for (StubMethod checkedSetter in cls.checkedSetters) {
assert(!_checkedSetterMap.containsKey(checkedSetter.element));
_checkedSetterMap[checkedSetter.element] = checkedSetter;
}
}
}
Method getMethod(FunctionEntity function) {
return _methodMap[function];
}
Field getField(FieldEntity field) {
return _fieldMap[field];
}
StaticField getStaticField(FieldEntity field) {
return _staticFieldMap[field];
}
StubMethod getCheckedSetter(FieldEntity field) {
return _checkedSetterMap[field];
}
@override
String toString() => 'ClassData(cls=$cls,'
'_methodMap=$_methodMap,_fieldMap=$_fieldMap)';
}
void forEachNode(js.Node root,
{void Function(js.Call) onCall,
void Function(js.PropertyAccess) onPropertyAccess,
void Function(js.Assignment) onAssignment,
void Function(js.Switch) onSwitch}) {
CallbackVisitor visitor = new CallbackVisitor(
onCall: onCall,
onPropertyAccess: onPropertyAccess,
onAssignment: onAssignment,
onSwitch: onSwitch);
root.accept(visitor);
}
class CallbackVisitor extends js.BaseVisitorVoid {
final void Function(js.Call) onCall;
final void Function(js.PropertyAccess) onPropertyAccess;
final void Function(js.Assignment) onAssignment;
final void Function(js.Switch) onSwitch;
CallbackVisitor(
{this.onCall, this.onPropertyAccess, this.onAssignment, this.onSwitch});
@override
void visitCall(js.Call node) {
if (onCall != null) onCall(node);
super.visitCall(node);
}
@override
void visitAccess(js.PropertyAccess node) {
if (onPropertyAccess != null) onPropertyAccess(node);
super.visitAccess(node);
}
@override
void visitAssignment(js.Assignment node) {
if (onAssignment != null) onAssignment(node);
return super.visitAssignment(node);
}
@override
void visitSwitch(js.Switch node) {
if (onSwitch != null) onSwitch(node);
return super.visitSwitch(node);
}
}