]>
Commit | Line | Data |
---|---|---|
1 | # Copyright (C) 2011 Apple Inc. All rights reserved. | |
2 | # | |
3 | # Redistribution and use in source and binary forms, with or without | |
4 | # modification, are permitted provided that the following conditions | |
5 | # are met: | |
6 | # 1. Redistributions of source code must retain the above copyright | |
7 | # notice, this list of conditions and the following disclaimer. | |
8 | # 2. Redistributions in binary form must reproduce the above copyright | |
9 | # notice, this list of conditions and the following disclaimer in the | |
10 | # documentation and/or other materials provided with the distribution. | |
11 | # | |
12 | # THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' | |
13 | # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, | |
14 | # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | |
15 | # PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS | |
16 | # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | |
17 | # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | |
18 | # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | |
19 | # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | |
20 | # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | |
21 | # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF | |
22 | # THE POSSIBILITY OF SUCH DAMAGE. | |
23 | ||
24 | require "config" | |
25 | require "ast" | |
26 | ||
27 | # | |
28 | # node.resolveSettings(settings) | |
29 | # | |
30 | # Construct a new AST that does not have any IfThenElse nodes by | |
31 | # substituting concrete boolean values for each Setting. | |
32 | # | |
33 | ||
34 | class Node | |
35 | def resolveSettings(settings) | |
36 | mapChildren { | |
37 | | child | | |
38 | child.resolveSettings(settings) | |
39 | } | |
40 | end | |
41 | end | |
42 | ||
43 | class True | |
44 | def resolveSettings(settings) | |
45 | self | |
46 | end | |
47 | end | |
48 | ||
49 | class False | |
50 | def resolveSettings(settings) | |
51 | self | |
52 | end | |
53 | end | |
54 | ||
55 | class Setting | |
56 | def resolveSettings(settings) | |
57 | settings[@name].asNode | |
58 | end | |
59 | end | |
60 | ||
61 | class And | |
62 | def resolveSettings(settings) | |
63 | (@left.resolveSettings(settings).value and @right.resolveSettings(settings).value).asNode | |
64 | end | |
65 | end | |
66 | ||
67 | class Or | |
68 | def resolveSettings(settings) | |
69 | (@left.resolveSettings(settings).value or @right.resolveSettings(settings).value).asNode | |
70 | end | |
71 | end | |
72 | ||
73 | class Not | |
74 | def resolveSettings(settings) | |
75 | (not @child.resolveSettings(settings).value).asNode | |
76 | end | |
77 | end | |
78 | ||
79 | class IfThenElse | |
80 | def resolveSettings(settings) | |
81 | if @predicate.resolveSettings(settings).value | |
82 | @thenCase.resolveSettings(settings) | |
83 | else | |
84 | @elseCase.resolveSettings(settings) | |
85 | end | |
86 | end | |
87 | end | |
88 | ||
89 | class Sequence | |
90 | def resolveSettings(settings) | |
91 | newList = [] | |
92 | @list.each { | |
93 | | item | | |
94 | item = item.resolveSettings(settings) | |
95 | if item.is_a? Sequence | |
96 | newList += item.list | |
97 | else | |
98 | newList << item | |
99 | end | |
100 | } | |
101 | Sequence.new(codeOrigin, newList) | |
102 | end | |
103 | end | |
104 | ||
105 | # | |
106 | # node.demacroify(macros) | |
107 | # node.substitute(mapping) | |
108 | # | |
109 | # demacroify() constructs a new AST that does not have any Macro | |
110 | # nodes, while substitute() replaces Variable nodes with the given | |
111 | # nodes in the mapping. | |
112 | # | |
113 | ||
114 | class Node | |
115 | def demacroify(macros) | |
116 | mapChildren { | |
117 | | child | | |
118 | child.demacroify(macros) | |
119 | } | |
120 | end | |
121 | ||
122 | def substitute(mapping) | |
123 | mapChildren { | |
124 | | child | | |
125 | child.substitute(mapping) | |
126 | } | |
127 | end | |
128 | ||
129 | def substituteLabels(mapping) | |
130 | mapChildren { | |
131 | | child | | |
132 | child.substituteLabels(mapping) | |
133 | } | |
134 | end | |
135 | end | |
136 | ||
137 | class Macro | |
138 | def substitute(mapping) | |
139 | myMapping = {} | |
140 | mapping.each_pair { | |
141 | | key, value | | |
142 | unless @variables.include? key | |
143 | myMapping[key] = value | |
144 | end | |
145 | } | |
146 | mapChildren { | |
147 | | child | | |
148 | child.substitute(myMapping) | |
149 | } | |
150 | end | |
151 | end | |
152 | ||
153 | class Variable | |
154 | def substitute(mapping) | |
155 | if mapping[self] | |
156 | mapping[self] | |
157 | else | |
158 | self | |
159 | end | |
160 | end | |
161 | end | |
162 | ||
163 | class LocalLabel | |
164 | def substituteLabels(mapping) | |
165 | if mapping[self] | |
166 | mapping[self] | |
167 | else | |
168 | self | |
169 | end | |
170 | end | |
171 | end | |
172 | ||
173 | class Sequence | |
174 | def substitute(constants) | |
175 | newList = [] | |
176 | myConstants = constants.dup | |
177 | @list.each { | |
178 | | item | | |
179 | if item.is_a? ConstDecl | |
180 | myConstants[item.variable] = item.value.substitute(myConstants) | |
181 | else | |
182 | newList << item.substitute(myConstants) | |
183 | end | |
184 | } | |
185 | Sequence.new(codeOrigin, newList) | |
186 | end | |
187 | ||
188 | def renameLabels(comment) | |
189 | mapping = {} | |
190 | ||
191 | @list.each { | |
192 | | item | | |
193 | if item.is_a? LocalLabel | |
194 | mapping[item] = LocalLabel.unique(if comment then comment + "_" else "" end + item.cleanName) | |
195 | end | |
196 | } | |
197 | ||
198 | substituteLabels(mapping) | |
199 | end | |
200 | ||
201 | def demacroify(macros) | |
202 | myMacros = macros.dup | |
203 | @list.each { | |
204 | | item | | |
205 | if item.is_a? Macro | |
206 | myMacros[item.name] = item | |
207 | end | |
208 | } | |
209 | newList = [] | |
210 | @list.each { | |
211 | | item | | |
212 | if item.is_a? Macro | |
213 | # Ignore. | |
214 | elsif item.is_a? MacroCall | |
215 | mapping = {} | |
216 | myMyMacros = myMacros.dup | |
217 | raise "Could not find macro #{item.name} at #{item.codeOriginString}" unless myMacros[item.name] | |
218 | raise "Argument count mismatch for call to #{item.name} at #{item.codeOriginString}" unless item.operands.size == myMacros[item.name].variables.size | |
219 | item.operands.size.times { | |
220 | | idx | | |
221 | if item.operands[idx].is_a? Variable and myMacros[item.operands[idx].name] | |
222 | myMyMacros[myMacros[item.name].variables[idx].name] = myMacros[item.operands[idx].name] | |
223 | mapping[myMacros[item.name].variables[idx].name] = nil | |
224 | elsif item.operands[idx].is_a? Macro | |
225 | myMyMacros[myMacros[item.name].variables[idx].name] = item.operands[idx] | |
226 | mapping[myMacros[item.name].variables[idx].name] = nil | |
227 | else | |
228 | myMyMacros[myMacros[item.name].variables[idx]] = nil | |
229 | mapping[myMacros[item.name].variables[idx]] = item.operands[idx] | |
230 | end | |
231 | } | |
232 | if item.annotation | |
233 | newList << Instruction.new(item.codeOrigin, "localAnnotation", [], item.annotation) | |
234 | end | |
235 | newList += myMacros[item.name].body.substitute(mapping).demacroify(myMyMacros).renameLabels(item.name).list | |
236 | else | |
237 | newList << item.demacroify(myMacros) | |
238 | end | |
239 | } | |
240 | Sequence.new(codeOrigin, newList).substitute({}) | |
241 | end | |
242 | end | |
243 | ||
244 | # | |
245 | # node.resolveOffsets(offsets, sizes) | |
246 | # | |
247 | # Construct a new AST that has offset values instead of symbolic | |
248 | # offsets. | |
249 | # | |
250 | ||
251 | class Node | |
252 | def resolveOffsets(offsets, sizes) | |
253 | mapChildren { | |
254 | | child | | |
255 | child.resolveOffsets(offsets, sizes) | |
256 | } | |
257 | end | |
258 | end | |
259 | ||
260 | class StructOffset | |
261 | def resolveOffsets(offsets, sizes) | |
262 | if offsets[self] | |
263 | Immediate.new(codeOrigin, offsets[self]) | |
264 | else | |
265 | self | |
266 | end | |
267 | end | |
268 | end | |
269 | ||
270 | class Sizeof | |
271 | def resolveOffsets(offsets, sizes) | |
272 | if sizes[self] | |
273 | Immediate.new(codeOrigin, sizes[self]) | |
274 | else | |
275 | puts "Could not find #{self.inspect} in #{sizes.keys.inspect}" | |
276 | puts "sizes = #{sizes.inspect}" | |
277 | self | |
278 | end | |
279 | end | |
280 | end | |
281 | ||
282 | # | |
283 | # node.fold | |
284 | # | |
285 | # Resolve constant references and compute arithmetic expressions. | |
286 | # | |
287 | ||
288 | class Node | |
289 | def fold | |
290 | mapChildren { | |
291 | | child | | |
292 | child.fold | |
293 | } | |
294 | end | |
295 | end | |
296 | ||
297 | class AddImmediates | |
298 | def fold | |
299 | @left = @left.fold | |
300 | @right = @right.fold | |
301 | return self unless @left.is_a? Immediate | |
302 | return self unless @right.is_a? Immediate | |
303 | Immediate.new(codeOrigin, @left.value + @right.value) | |
304 | end | |
305 | end | |
306 | ||
307 | class SubImmediates | |
308 | def fold | |
309 | @left = @left.fold | |
310 | @right = @right.fold | |
311 | return self unless @left.is_a? Immediate | |
312 | return self unless @right.is_a? Immediate | |
313 | Immediate.new(codeOrigin, @left.value - @right.value) | |
314 | end | |
315 | end | |
316 | ||
317 | class MulImmediates | |
318 | def fold | |
319 | @left = @left.fold | |
320 | @right = @right.fold | |
321 | return self unless @left.is_a? Immediate | |
322 | return self unless @right.is_a? Immediate | |
323 | Immediate.new(codeOrigin, @left.value * @right.value) | |
324 | end | |
325 | end | |
326 | ||
327 | class NegImmediate | |
328 | def fold | |
329 | @child = @child.fold | |
330 | return self unless @child.is_a? Immediate | |
331 | Immediate.new(codeOrigin, -@child.value) | |
332 | end | |
333 | end | |
334 | ||
335 | class OrImmediates | |
336 | def fold | |
337 | @left = @left.fold | |
338 | @right = @right.fold | |
339 | return self unless @left.is_a? Immediate | |
340 | return self unless @right.is_a? Immediate | |
341 | Immediate.new(codeOrigin, @left.value | @right.value) | |
342 | end | |
343 | end | |
344 | ||
345 | class AndImmediates | |
346 | def fold | |
347 | @left = @left.fold | |
348 | @right = @right.fold | |
349 | return self unless @left.is_a? Immediate | |
350 | return self unless @right.is_a? Immediate | |
351 | Immediate.new(codeOrigin, @left.value & @right.value) | |
352 | end | |
353 | end | |
354 | ||
355 | class XorImmediates | |
356 | def fold | |
357 | @left = @left.fold | |
358 | @right = @right.fold | |
359 | return self unless @left.is_a? Immediate | |
360 | return self unless @right.is_a? Immediate | |
361 | Immediate.new(codeOrigin, @left.value ^ @right.value) | |
362 | end | |
363 | end | |
364 | ||
365 | class BitnotImmediate | |
366 | def fold | |
367 | @child = @child.fold | |
368 | return self unless @child.is_a? Immediate | |
369 | Immediate.new(codeOrigin, ~@child.value) | |
370 | end | |
371 | end | |
372 | ||
373 | # | |
374 | # node.resolveAfterSettings(offsets, sizes) | |
375 | # | |
376 | # Compile assembly against a set of offsets. | |
377 | # | |
378 | ||
379 | class Node | |
380 | def resolve(offsets, sizes) | |
381 | demacroify({}).resolveOffsets(offsets, sizes).fold | |
382 | end | |
383 | end | |
384 | ||
385 | # | |
386 | # node.validate | |
387 | # | |
388 | # Checks that the node is ready for backend compilation. | |
389 | # | |
390 | ||
391 | class Node | |
392 | def validate | |
393 | raise "Unresolved #{dump} at #{codeOriginString}" | |
394 | end | |
395 | ||
396 | def validateChildren | |
397 | children.each { | |
398 | | node | | |
399 | node.validate | |
400 | } | |
401 | end | |
402 | end | |
403 | ||
404 | class Sequence | |
405 | def validate | |
406 | validateChildren | |
407 | ||
408 | # Further verify that this list contains only instructions, labels, and skips. | |
409 | @list.each { | |
410 | | node | | |
411 | unless node.is_a? Instruction or | |
412 | node.is_a? Label or | |
413 | node.is_a? LocalLabel or | |
414 | node.is_a? Skip | |
415 | raise "Unexpected #{node.inspect} at #{node.codeOrigin}" | |
416 | end | |
417 | } | |
418 | end | |
419 | end | |
420 | ||
421 | class Immediate | |
422 | def validate | |
423 | end | |
424 | end | |
425 | ||
426 | class StringLiteral | |
427 | def validate | |
428 | end | |
429 | end | |
430 | ||
431 | class RegisterID | |
432 | def validate | |
433 | end | |
434 | end | |
435 | ||
436 | class FPRegisterID | |
437 | def validate | |
438 | end | |
439 | end | |
440 | ||
441 | class Address | |
442 | def validate | |
443 | validateChildren | |
444 | end | |
445 | end | |
446 | ||
447 | class BaseIndex | |
448 | def validate | |
449 | validateChildren | |
450 | end | |
451 | end | |
452 | ||
453 | class AbsoluteAddress | |
454 | def validate | |
455 | validateChildren | |
456 | end | |
457 | end | |
458 | ||
459 | class Instruction | |
460 | def validate | |
461 | validateChildren | |
462 | end | |
463 | end | |
464 | ||
465 | class SubImmediates | |
466 | def validate | |
467 | raise "Invalid operand #{left.dump} to immediate subtraction" unless left.immediateOperand? | |
468 | raise "Invalid operand #{right.dump} to immediate subtraction" unless right.immediateOperand? | |
469 | end | |
470 | end | |
471 | ||
472 | class Error | |
473 | def validate | |
474 | end | |
475 | end | |
476 | ||
477 | class Label | |
478 | def validate | |
479 | end | |
480 | end | |
481 | ||
482 | class LocalLabel | |
483 | def validate | |
484 | end | |
485 | end | |
486 | ||
487 | class LabelReference | |
488 | def validate | |
489 | end | |
490 | end | |
491 | ||
492 | class LocalLabelReference | |
493 | def validate | |
494 | end | |
495 | end | |
496 | ||
497 | class Skip | |
498 | def validate | |
499 | end | |
500 | end | |
501 |