Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(83)

Side by Side Diff: pkg/dev_compiler/lib/src/compiler/class_property_model.dart

Issue 2571363002: fixes #27385, implement virtual fields in DDC (Closed)
Patch Set: format Created 4 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright (c) 2016, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file.
4
5 import 'dart:collection' show HashSet;
6
7 import 'package:analyzer/dart/ast/ast.dart' show Identifier;
8 import 'package:analyzer/dart/element/element.dart';
9 import 'package:analyzer/src/dart/element/element.dart' show FieldElementImpl;
10
11 import '../js_ast/js_ast.dart' as JS;
12 import 'element_helpers.dart';
13 import 'js_names.dart' as JS;
14
15 /// Tracks how fields, getters and setters are represented when emitting JS.
16 ///
17 /// Dart classes have implicit features that must be made explicit:
18 ///
19 /// - virtual fields induce a getter and setter pair.
20 /// - getters and setters are independent.
21 /// - getters and setters can be overridden.
22 ///
23 class ClassPropertyModel {
24 /// Fields that are virtual, that is, they must be generated as a property
25 /// pair in JavaScript.
26 ///
27 /// The value property stores the symbol used for the field's storage slot.
28 final virtualFields = <FieldElement, JS.TemporaryId>{};
29
30 /// Static fields that are overridden, this does not matter for Dart but in
31 /// JS we need to take care initializing these because JS classes inherit
32 /// statics.
33 final staticFieldOverrides = new HashSet<FieldElement>();
34
35 /// The set of inherited getters, used because JS getters/setters are paired,
36 /// so if we're generating a setter we may need to emit a getter that calls
37 /// super.
38 final inheritedGetters = new HashSet<String>();
39
40 /// The set of inherited setters, used because JS getters/setters are paired,
41 /// so if we're generating a getter we may need to emit a setter that calls
42 /// super.
43 final inheritedSetters = new HashSet<String>();
44
45 ClassPropertyModel.build(
46 ClassElement classElem, Iterable<ExecutableElement> extensionMembers) {
47 // Visit superclasses to collect information about their fields/accessors.
48 // This is expensive so we try to collect everything in one pass.
49 for (var base in getSuperclasses(classElem)) {
50 for (var accessor in base.accessors) {
51 // For getter/setter pairs only process them once.
52 if (accessor.correspondingGetter != null) continue;
53
54 var field = accessor.variable;
55 var name = field.name;
56 // Ignore private names from other libraries.
57 if (Identifier.isPrivateName(name) &&
58 accessor.library != classElem.library) {
59 continue;
60 }
61
62 if (field.getter?.isAbstract == false) inheritedGetters.add(name);
63 if (field.setter?.isAbstract == false) inheritedSetters.add(name);
64 }
65 }
66
67 var extensionNames =
68 new HashSet<String>.from(extensionMembers.map((e) => e.name));
69
70 // Visit accessors in the current class, and see if they need to be
71 // generated differently based on the inherited fields/accessors.
72 for (var accessor in classElem.accessors) {
73 // For getter/setter pairs only process them once.
74 if (accessor.correspondingGetter != null) continue;
75 // Also ignore abstract fields.
76 if (accessor.isAbstract) continue;
77
78 var field = accessor.variable;
79 var name = field.name;
80 // Is it a field?
81 if (!field.isSynthetic && field is FieldElementImpl) {
82 if (inheritedGetters.contains(name) ||
83 inheritedSetters.contains(name) ||
84 extensionNames.contains(name) ||
85 field.isVirtual) {
86 if (field.isStatic) {
87 staticFieldOverrides.add(field);
88 } else {
89 virtualFields[field] = new JS.TemporaryId(name);
90 }
91 }
92 }
93 }
94 }
95 }
OLDNEW
« no previous file with comments | « pkg/dev_compiler/lib/js/legacy/dart_sdk.js ('k') | pkg/dev_compiler/lib/src/compiler/code_generator.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698