OLD | NEW |
| (Empty) |
1 // Copyright (c) 2012, 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 class ScannerTask extends CompilerTask { | |
6 ScannerTask(Compiler compiler) : super(compiler); | |
7 String get name() => 'Scanner'; | |
8 | |
9 void scan(CompilationUnitElement compilationUnit) { | |
10 measure(() { | |
11 if (compilationUnit.kind === ElementKind.LIBRARY) { | |
12 compiler.log("scanning library ${compilationUnit.script.name}"); | |
13 } | |
14 scanElements(compilationUnit); | |
15 if (compilationUnit.kind === ElementKind.LIBRARY) { | |
16 processScriptTags(compilationUnit); | |
17 } | |
18 }); | |
19 } | |
20 | |
21 void processScriptTags(LibraryElement library) { | |
22 int tagState = TagState.NO_TAG_SEEN; | |
23 | |
24 /** | |
25 * If [value] is less than [tagState] complain and return | |
26 * [tagState]. Otherwise return the new value for [tagState] | |
27 * (transition function for state machine). | |
28 */ | |
29 int checkTag(int value, ScriptTag tag) { | |
30 if (tagState > value) { | |
31 compiler.reportError(tag, 'out of order'); | |
32 return tagState; | |
33 } | |
34 return TagState.NEXT[value]; | |
35 } | |
36 | |
37 LinkBuilder<ScriptTag> imports = new LinkBuilder<ScriptTag>(); | |
38 Uri cwd = new Uri(scheme: 'file', path: compiler.currentDirectory); | |
39 Uri base = cwd.resolve(library.script.name.toString()); | |
40 for (ScriptTag tag in library.tags.reverse()) { | |
41 StringNode argument = tag.argument; | |
42 // TODO(lrn): Support interpolations here. We need access to the | |
43 // special constants that can be inserted into script tag strings. | |
44 Uri resolved = base.resolve(argument.dartString.slowToString()); | |
45 if (tag.isImport()) { | |
46 tagState = checkTag(TagState.IMPORT, tag); | |
47 // It is not safe to import other libraries at this point as | |
48 // another library could then observe the current library | |
49 // before it fully declares all the members that are sourced | |
50 // in. | |
51 imports.addLast(tag); | |
52 } else if (tag.isLibrary()) { | |
53 tagState = checkTag(TagState.LIBRARY, tag); | |
54 if (library.libraryTag !== null) { | |
55 compiler.cancel("duplicated library declaration", node: tag); | |
56 } else { | |
57 library.libraryTag = tag; | |
58 } | |
59 } else if (tag.isSource()) { | |
60 tagState = checkTag(TagState.SOURCE, tag); | |
61 Script script = compiler.readScript(resolved, tag); | |
62 CompilationUnitElement unit = | |
63 new CompilationUnitElement(script, library); | |
64 compiler.withCurrentElement(unit, () => scan(unit)); | |
65 } else if (tag.isResource()) { | |
66 tagState = checkTag(TagState.RESOURCE, tag); | |
67 compiler.reportWarning(tag, 'ignoring resource tag'); | |
68 } else { | |
69 compiler.cancel("illegal script tag: ${tag.tag}", node: tag); | |
70 } | |
71 } | |
72 // TODO(ahe): During Compiler.scanBuiltinLibraries, | |
73 // compiler.coreLibrary is null. Clean this up when there is a | |
74 // better way to access "dart:core". | |
75 bool implicitlyImportCoreLibrary = compiler.coreLibrary !== null; | |
76 for (ScriptTag tag in imports.toLink()) { | |
77 // Now that we have processed all the source tags, it is safe to | |
78 // start loading other libraries. | |
79 StringNode argument = tag.argument; | |
80 Uri resolved = base.resolve(argument.dartString.slowToString()); | |
81 if (resolved.toString() == "dart:core") { | |
82 implicitlyImportCoreLibrary = false; | |
83 } | |
84 importLibrary(library, loadLibrary(resolved, tag), tag); | |
85 } | |
86 if (implicitlyImportCoreLibrary) { | |
87 importLibrary(library, compiler.coreLibrary, null); | |
88 } | |
89 } | |
90 | |
91 void scanElements(CompilationUnitElement compilationUnit) { | |
92 Script script = compilationUnit.script; | |
93 Token tokens; | |
94 try { | |
95 tokens = new StringScanner(script.text).tokenize(); | |
96 } catch (MalformedInputException ex) { | |
97 Token token; | |
98 var message; | |
99 if (ex.position is num) { | |
100 // TODO(ahe): Always use tokens in MalformedInputException. | |
101 token = new Token(EOF_INFO, ex.position); | |
102 } else { | |
103 token = ex.position; | |
104 } | |
105 compiler.cancel(ex.message, token: token); | |
106 } | |
107 compiler.dietParser.dietParse(compilationUnit, tokens); | |
108 } | |
109 | |
110 LibraryElement loadLibrary(Uri uri, ScriptTag node) { | |
111 bool newLibrary = false; | |
112 LibraryElement library = | |
113 compiler.universe.libraries.putIfAbsent(uri.toString(), () { | |
114 newLibrary = true; | |
115 Script script = compiler.readScript(uri, node); | |
116 LibraryElement element = new LibraryElement(script); | |
117 native.maybeEnableNative(compiler, element, uri); | |
118 return element; | |
119 }); | |
120 if (newLibrary) { | |
121 compiler.withCurrentElement(library, () => scan(library)); | |
122 compiler.onLibraryLoaded(library, uri); | |
123 } | |
124 return library; | |
125 } | |
126 | |
127 void importLibrary(LibraryElement library, LibraryElement imported, | |
128 ScriptTag tag) { | |
129 if (!imported.hasLibraryName()) { | |
130 compiler.withCurrentElement(library, () { | |
131 compiler.reportError(tag, | |
132 'no #library tag found in ${imported.script.uri}'); | |
133 }); | |
134 } | |
135 if (tag !== null && tag.prefix !== null) { | |
136 SourceString prefix = | |
137 new SourceString(tag.prefix.dartString.slowToString()); | |
138 Element e = library.find(prefix); | |
139 if (e === null) { | |
140 e = new PrefixElement(prefix, library, tag.getBeginToken()); | |
141 library.define(e, compiler); | |
142 } | |
143 if (e.kind !== ElementKind.PREFIX) { | |
144 compiler.withCurrentElement(e, () { | |
145 compiler.reportWarning(new Identifier(e.position()), | |
146 'duplicated definition'); | |
147 }); | |
148 compiler.reportError(tag.prefix, 'duplicate defintion'); | |
149 } | |
150 imported.forEachExport((Element element) { | |
151 Element existing = e.imported.putIfAbsent(element.name, () => element); | |
152 if (existing !== element) { | |
153 compiler.withCurrentElement(existing, () { | |
154 compiler.reportWarning(new Identifier(existing.position()), | |
155 'duplicated import'); | |
156 }); | |
157 compiler.withCurrentElement(element, () { | |
158 compiler.reportError(new Identifier(element.position()), | |
159 'duplicated import'); | |
160 }); | |
161 } | |
162 }); | |
163 } else { | |
164 imported.forEachExport((Element element) { | |
165 compiler.withCurrentElement(element, () { | |
166 library.define(element, compiler); | |
167 }); | |
168 }); | |
169 } | |
170 } | |
171 } | |
172 | |
173 class DietParserTask extends CompilerTask { | |
174 DietParserTask(Compiler compiler) : super(compiler); | |
175 final String name = 'Diet Parser'; | |
176 | |
177 dietParse(CompilationUnitElement compilationUnit, Token tokens) { | |
178 measure(() { | |
179 ElementListener listener = new ElementListener(compiler, compilationUnit); | |
180 PartialParser parser = new PartialParser(listener); | |
181 parser.parseUnit(tokens); | |
182 }); | |
183 } | |
184 } | |
185 | |
186 /** | |
187 * The fields of this class models a state machine for checking script | |
188 * tags come in the correct order. | |
189 */ | |
190 class TagState { | |
191 static final int NO_TAG_SEEN = 0; | |
192 static final int LIBRARY = 1; | |
193 static final int IMPORT = 2; | |
194 static final int SOURCE = 3; | |
195 static final int RESOURCE = 4; | |
196 | |
197 /** Next state. */ | |
198 static final List<int> NEXT = | |
199 const <int>[NO_TAG_SEEN, | |
200 IMPORT, // Only one library tag is allowed. | |
201 IMPORT, | |
202 SOURCE, | |
203 RESOURCE]; | |
204 } | |
OLD | NEW |