Unreviewed, rolling out r237084, r237088, r237098, and
[WebKit-https.git] / Source / JavaScriptCore / offlineasm / generate_offset_extractor.rb
1 #!/usr/bin/env ruby
2
3 # Copyright (C) 2011 Apple Inc. All rights reserved.
4 #
5 # Redistribution and use in source and binary forms, with or without
6 # modification, are permitted provided that the following conditions
7 # are met:
8 # 1. Redistributions of source code must retain the above copyright
9 #    notice, this list of conditions and the following disclaimer.
10 # 2. Redistributions in binary form must reproduce the above copyright
11 #    notice, this list of conditions and the following disclaimer in the
12 #    documentation and/or other materials provided with the distribution.
13 #
14 # THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
15 # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
16 # THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17 # PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
18 # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
19 # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
20 # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
21 # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
22 # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
23 # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
24 # THE POSSIBILITY OF SUCH DAMAGE.
25
26 $: << File.dirname(__FILE__)
27
28 require "config"
29 require "backends"
30 require "digest/sha1"
31 require "offsets"
32 require "parser"
33 require "self_hash"
34 require "settings"
35 require "transform"
36
37 IncludeFile.processIncludeOptions()
38
39 inputFlnm = ARGV.shift
40 outputFlnm = ARGV.shift
41
42 validBackends = canonicalizeBackendNames(ARGV.shift.split(/[,\s]+/))
43 includeOnlyBackends(validBackends)
44
45 def emitMagicNumber
46     OFFSET_MAGIC_NUMBERS.each {
47         | number |
48         $output.puts "unsigned(#{number}),"
49     }
50 end
51
52 inputHash = "// offlineasm input hash: #{parseHash(inputFlnm)} #{selfHash}"
53
54 if FileTest.exist? outputFlnm
55     File.open(outputFlnm, "r") {
56         | inp |
57         firstLine = inp.gets
58         if firstLine and firstLine.chomp == inputHash
59             $stderr.puts "offlineasm: Nothing changed."
60             exit 0
61         end
62     }
63 end
64
65 originalAST = parse(inputFlnm)
66
67 #
68 # Optimize the AST to make configuration extraction faster. This reduces the AST to a form
69 # that only contains the things that matter for our purposes: offsets, sizes, and if
70 # statements.
71 #
72
73 class Node
74     def offsetsPruneTo(sequence)
75         children.each {
76             | child |
77             child.offsetsPruneTo(sequence)
78         }
79     end
80     
81     def offsetsPrune
82         result = Sequence.new(codeOrigin, [])
83         offsetsPruneTo(result)
84         result
85     end
86 end
87
88 class IfThenElse
89     def offsetsPruneTo(sequence)
90         ifThenElse = IfThenElse.new(codeOrigin, predicate, thenCase.offsetsPrune)
91         ifThenElse.elseCase = elseCase.offsetsPrune
92         sequence.list << ifThenElse
93     end
94 end
95
96 class StructOffset
97     def offsetsPruneTo(sequence)
98         sequence.list << self
99     end
100 end
101
102 class Sizeof
103     def offsetsPruneTo(sequence)
104         sequence.list << self
105     end
106 end
107
108 class ConstExpr
109     def offsetsPruneTo(sequence)
110         sequence.list << self
111     end
112 end
113
114 prunedAST = originalAST.offsetsPrune
115
116 File.open(outputFlnm, "w") {
117     | outp |
118     $output = outp
119     outp.puts inputHash
120     length = 0
121
122     emitCodeInAllConfigurations(prunedAST) {
123         | settings, ast, backend, index |
124         constsList = ast.filter(ConstExpr).uniq.sort
125
126         constsList.each_with_index {
127             | const, index |
128             outp.puts "constexpr int64_t constValue#{index} = static_cast<int64_t>(#{const.value});"
129         }
130     }
131
132     emitCodeInAllConfigurations(prunedAST) {
133         | settings, ast, backend, index |
134         offsetsList = ast.filter(StructOffset).uniq.sort
135         sizesList = ast.filter(Sizeof).uniq.sort
136         constsList = ast.filter(ConstExpr).uniq.sort
137         length += OFFSET_HEADER_MAGIC_NUMBERS.size + (OFFSET_MAGIC_NUMBERS.size + 1) * (1 + offsetsList.size + sizesList.size + constsList.size)
138     }
139     outp.puts "static const int64_t extractorTable[#{length}] = {"
140     emitCodeInAllConfigurations(prunedAST) {
141         | settings, ast, backend, index |
142         OFFSET_HEADER_MAGIC_NUMBERS.each {
143             | number |
144             $output.puts "unsigned(#{number}),"
145         }
146
147         offsetsList = ast.filter(StructOffset).uniq.sort
148         sizesList = ast.filter(Sizeof).uniq.sort
149         constsList = ast.filter(ConstExpr).uniq.sort
150
151         emitMagicNumber
152         outp.puts "#{index},"
153         offsetsList.each {
154             | offset |
155             emitMagicNumber
156             outp.puts "OFFLINE_ASM_OFFSETOF(#{offset.struct}, #{offset.field}),"
157         }
158         sizesList.each {
159             | sizeof |
160             emitMagicNumber
161             outp.puts "sizeof(#{sizeof.struct}),"
162         }
163         constsList.each_index {
164             | index |
165             emitMagicNumber
166             outp.puts "constValue#{index},"
167         }
168     }
169     outp.puts "};"
170
171 }