blob: f13cef5fd67d113f0596bd9fb6fe252ad429fbf0 [file] [log] [blame] [edit]
// 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.
import 'package:compiler/src/common/elements.dart';
import 'package:compiler/src/deferred_load/output_unit.dart' show OutputUnit;
import 'package:compiler/src/elements/entities.dart';
import 'package:compiler/src/elements/names.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(ElementEnvironment 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(name, cls.library.canonicalUri),
);
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;
}
late final Map<LibraryEntity, LibraryData> _libraryMap = () {
final map = <LibraryEntity, LibraryData>{};
for (Fragment fragment in program.fragments) {
for (Library library in fragment.libraries) {
assert(!map.containsKey(library.element));
map[library.element] = LibraryData(library, fragment);
}
}
return map;
}();
LibraryData? getLibraryData(LibraryEntity element) {
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] = ClassData(cls);
}
for (StaticMethod method in library.statics) {
ClassEntity? enclosingClass = method.element?.enclosingClass;
if (enclosingClass != null) {
ClassData data = _classMap.putIfAbsent(
enclosingClass,
() => ClassData(null),
);
assert(!data._methodMap.containsKey(method.element));
data._methodMap[method.element as FunctionEntity] = method;
} else if (method.element != null) {
assert(!_methodMap.containsKey(method.element));
_methodMap[method.element as FunctionEntity] = method;
}
}
void addStaticField(StaticField field) {
ClassEntity? enclosingClass = field.element.enclosingClass;
if (enclosingClass != null) {
ClassData data = _classMap.putIfAbsent(
enclosingClass,
() => ClassData(null),
);
assert(!data._fieldMap.containsKey(field.element));
data._staticFieldMap[field.element] = field;
} else {
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 as FunctionEntity] = 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 as FieldEntity] = 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 = 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);
}
}