racc 1.4.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (109) hide show
  1. data/.gitattributes +2 -0
  2. data/.gitignore +7 -0
  3. data/COPYING +515 -0
  4. data/ChangeLog +846 -0
  5. data/DEPENDS +4 -0
  6. data/README.en.rdoc +86 -0
  7. data/README.ja.rdoc +96 -0
  8. data/Rakefile +15 -0
  9. data/TODO +5 -0
  10. data/bin/racc +308 -0
  11. data/bin/racc2y +195 -0
  12. data/bin/y2racc +339 -0
  13. data/doc/en/NEWS.en.rdoc +282 -0
  14. data/doc/en/command.en.html +78 -0
  15. data/doc/en/debug.en.rdoc +20 -0
  16. data/doc/en/grammar.en.rdoc +230 -0
  17. data/doc/en/index.en.html +10 -0
  18. data/doc/en/parser.en.rdoc +74 -0
  19. data/doc/en/usage.en.html +92 -0
  20. data/doc/ja/NEWS.ja.rdoc +307 -0
  21. data/doc/ja/command.ja.html +94 -0
  22. data/doc/ja/debug.ja.rdoc +36 -0
  23. data/doc/ja/grammar.ja.rdoc +348 -0
  24. data/doc/ja/index.ja.html +10 -0
  25. data/doc/ja/parser.ja.rdoc +125 -0
  26. data/doc/ja/usage.ja.html +414 -0
  27. data/ext/racc/cparse/MANIFEST +4 -0
  28. data/ext/racc/cparse/cparse.c +824 -0
  29. data/ext/racc/cparse/depend +1 -0
  30. data/ext/racc/cparse/extconf.rb +7 -0
  31. data/fastcache/extconf.rb +2 -0
  32. data/fastcache/fastcache.c +185 -0
  33. data/lib/racc.rb +6 -0
  34. data/lib/racc/compat.rb +40 -0
  35. data/lib/racc/debugflags.rb +59 -0
  36. data/lib/racc/exception.rb +15 -0
  37. data/lib/racc/grammar.rb +1115 -0
  38. data/lib/racc/grammarfileparser.rb +559 -0
  39. data/lib/racc/info.rb +16 -0
  40. data/lib/racc/iset.rb +91 -0
  41. data/lib/racc/logfilegenerator.rb +214 -0
  42. data/lib/racc/parser.rb +439 -0
  43. data/lib/racc/parserfilegenerator.rb +511 -0
  44. data/lib/racc/pre-setup +13 -0
  45. data/lib/racc/sourcetext.rb +34 -0
  46. data/lib/racc/state.rb +971 -0
  47. data/lib/racc/statetransitiontable.rb +316 -0
  48. data/lib/racc/static.rb +5 -0
  49. data/misc/dist.sh +31 -0
  50. data/sample/array.y +67 -0
  51. data/sample/array2.y +59 -0
  52. data/sample/calc-ja.y +66 -0
  53. data/sample/calc.y +65 -0
  54. data/sample/conflict.y +15 -0
  55. data/sample/hash.y +60 -0
  56. data/sample/lalr.y +17 -0
  57. data/sample/lists.y +57 -0
  58. data/sample/syntax.y +46 -0
  59. data/sample/yyerr.y +46 -0
  60. data/setup.rb +1587 -0
  61. data/tasks/doc.rb +12 -0
  62. data/tasks/email.rb +55 -0
  63. data/tasks/file.rb +37 -0
  64. data/tasks/gem.rb +37 -0
  65. data/tasks/test.rb +16 -0
  66. data/test/assets/chk.y +126 -0
  67. data/test/assets/conf.y +16 -0
  68. data/test/assets/digraph.y +29 -0
  69. data/test/assets/echk.y +118 -0
  70. data/test/assets/err.y +60 -0
  71. data/test/assets/expect.y +7 -0
  72. data/test/assets/firstline.y +4 -0
  73. data/test/assets/ichk.y +102 -0
  74. data/test/assets/intp.y +546 -0
  75. data/test/assets/mailp.y +437 -0
  76. data/test/assets/newsyn.y +25 -0
  77. data/test/assets/noend.y +4 -0
  78. data/test/assets/nonass.y +41 -0
  79. data/test/assets/normal.y +27 -0
  80. data/test/assets/norule.y +4 -0
  81. data/test/assets/nullbug1.y +25 -0
  82. data/test/assets/nullbug2.y +15 -0
  83. data/test/assets/opt.y +123 -0
  84. data/test/assets/percent.y +35 -0
  85. data/test/assets/recv.y +97 -0
  86. data/test/assets/rrconf.y +14 -0
  87. data/test/assets/scan.y +72 -0
  88. data/test/assets/syntax.y +50 -0
  89. data/test/assets/unterm.y +5 -0
  90. data/test/assets/useless.y +12 -0
  91. data/test/assets/yyerr.y +46 -0
  92. data/test/bench.y +36 -0
  93. data/test/helper.rb +88 -0
  94. data/test/infini.y +8 -0
  95. data/test/scandata/brace +7 -0
  96. data/test/scandata/gvar +1 -0
  97. data/test/scandata/normal +4 -0
  98. data/test/scandata/percent +18 -0
  99. data/test/scandata/slash +10 -0
  100. data/test/src.intp +34 -0
  101. data/test/start.y +20 -0
  102. data/test/test_chk_y.rb +51 -0
  103. data/test/test_grammar_file_parser.rb +15 -0
  104. data/test/test_racc_command.rb +155 -0
  105. data/test/test_scan_y.rb +51 -0
  106. data/test/testscanner.rb +51 -0
  107. data/web/racc.en.rhtml +42 -0
  108. data/web/racc.ja.rhtml +51 -0
  109. metadata +166 -0
@@ -0,0 +1,195 @@
1
+ #!/usr/local/bin/ruby
2
+ #
3
+ # $Id$
4
+ #
5
+ # Copyright (c) 1999-2006 Minero Aoki
6
+ #
7
+ # This program is feee software.
8
+ # You can distribute/modify this program under the terms of
9
+ # the GNU LGPL, Lesser General Public License version 2.1.
10
+ # For details of the LGPL, see the file "COPYING".
11
+ #
12
+
13
+ require 'racc/grammarfileparser'
14
+ require 'racc/info'
15
+ require 'optparse'
16
+
17
+ def main
18
+ @with_action = true
19
+ with_header = false
20
+ with_inner = false
21
+ with_footer = false
22
+ output = nil
23
+ parser = OptionParser.new
24
+ parser.banner = "Usage: #{File.basename($0)} [-AHIF] [-oFILENAME] GRAMMARFILE"
25
+ parser.on('-o', '--output=FILENAME', 'output file name [<input>.yacc]') {|name|
26
+ output = name
27
+ }
28
+ parser.on('-A', '--without-action', 'Does not include actions.') {
29
+ @with_action = false
30
+ }
31
+ parser.on('-H', '--with-header', 'Includes header part.') {
32
+ with_header = true
33
+ }
34
+ parser.on('-I', '--with-inner', 'Includes inner part.') {
35
+ with_inner = true
36
+ }
37
+ parser.on('-F', '--with-footer', 'Includes footer part.') {
38
+ with_footer = true
39
+ }
40
+ parser.on('--version', 'Prints version and quit.') {
41
+ puts "racc2y version #{Racc::Version}"
42
+ exit 0
43
+ }
44
+ parser.on('--copyright', 'Prints copyright and quit.') {
45
+ puts Racc::Copyright
46
+ exit 0
47
+ }
48
+ parser.on('--help', 'Prints this message and quit.') {
49
+ puts parser.help
50
+ exit 1
51
+ }
52
+ begin
53
+ parser.parse!
54
+ rescue OptionParser::ParseError => err
55
+ $stderr.puts err.message
56
+ $stderr.puts parser.help
57
+ exit 1
58
+ end
59
+ if ARGV.empty?
60
+ $stderr.puts "no input file"
61
+ exit 1
62
+ end
63
+ unless ARGV.size == 1
64
+ $stderr.puts "too many inputs"
65
+ exit 1
66
+ end
67
+ input = ARGV[0]
68
+
69
+ begin
70
+ result = Racc::GrammarFileParser.parse_file(input)
71
+ result.grammar.init
72
+ File.open(output || "#{input}.yacc", 'w') {|f|
73
+ f.puts "/* generated from #{input} */"
74
+ if with_header
75
+ f.puts
76
+ f.puts '%{'
77
+ print_user_codes f, result.params.header
78
+ f.puts '%}'
79
+ end
80
+ f.puts
81
+ print_terminals f, result.grammar
82
+ f.puts
83
+ print_precedence_table f, precedence_table(result.grammar)
84
+ f.puts
85
+ f.puts '%%'
86
+ print_grammar f, result.grammar
87
+ f.puts '%%'
88
+ if with_inner
89
+ f.puts '/*---- inner ----*/'
90
+ print_user_codes f, result.params.inner
91
+ end
92
+ if with_footer
93
+ f.puts '/*---- footer ----*/'
94
+ print_user_codes f, result.params.footer
95
+ end
96
+ }
97
+ rescue SystemCallError => err
98
+ $stderr.puts err.message
99
+ exit 1
100
+ end
101
+ end
102
+
103
+ def print_terminals(f, grammar)
104
+ init_indent = '%token'.size
105
+ f.print '%token'
106
+ columns = init_indent
107
+ grammar.symboltable.each_terminal do |t|
108
+ next unless t.terminal?
109
+ next if t.dummy?
110
+ next if t == grammar.symboltable.anchor
111
+ next if t == grammar.symboltable.error
112
+ unless t.value.kind_of?(String)
113
+ if columns > 60
114
+ f.puts
115
+ f.print ' ' * init_indent
116
+ columns = init_indent
117
+ end
118
+ columns += f.write(" #{yacc_symbol(t)}")
119
+ end
120
+ end
121
+ f.puts
122
+ end
123
+
124
+ def precedence_table(grammar)
125
+ table = []
126
+ grammar.symboltable.select {|sym| sym.precedence }.each do |sym|
127
+ (table[sym.prec] ||= [sym.assoc]).push sym
128
+ end
129
+ table.compact
130
+ end
131
+
132
+ def print_precedence_table(f, table)
133
+ return if table.empty?
134
+ f.puts '/* precedance table */'
135
+ table.each do |syms|
136
+ assoc = syms.shift
137
+ f.printf '%%%-8s ', assoc.to_s.downcase
138
+ f.puts syms.map {|s| yacc_symbol(s) }.join(' ')
139
+ end
140
+ f.puts
141
+ end
142
+
143
+ def print_grammar(f, grammar)
144
+ prev_target = nil
145
+ indent = 10
146
+ embactions = []
147
+ grammar.each do |rule|
148
+ if rule.target.dummy?
149
+ embactions.push rule.action unless rule.action.empty?
150
+ next
151
+ end
152
+ if rule.target == prev_target
153
+ f.print ' ' * indent, '|'
154
+ else
155
+ prev_target = rule.target
156
+ f.printf "\n%-10s:", yacc_symbol(prev_target)
157
+ end
158
+ rule.symbols.each do |s|
159
+ if s.dummy? # target of dummy rule for embedded action
160
+ f.puts
161
+ print_action f, embactions.shift, indent
162
+ f.print ' ' * (indent + 1)
163
+ else
164
+ f.print ' ', yacc_symbol(s)
165
+ end
166
+ end
167
+ if rule.specified_prec
168
+ f.print ' %prec ', yacc_symbol(rule.specified_prec)
169
+ end
170
+ f.puts
171
+ unless rule.action.empty?
172
+ print_action f, rule.action, indent
173
+ end
174
+ end
175
+ end
176
+
177
+ def print_action(f, action, indent)
178
+ return unless @with_action
179
+ f.print ' ' * (indent + 4), "{\n"
180
+ f.print ' ' * (indent + 6), action.source.text.strip, "\n"
181
+ f.print ' ' * (indent + 4) , "}\n"
182
+ end
183
+
184
+ def print_user_codes(f, srcs)
185
+ return if srcs.empty?
186
+ srcs.each do |src|
187
+ f.puts src.text
188
+ end
189
+ end
190
+
191
+ def yacc_symbol(s)
192
+ s.to_s.gsub('"', "'")
193
+ end
194
+
195
+ main
@@ -0,0 +1,339 @@
1
+ #!/usr/local/bin/ruby
2
+ #
3
+ # $Id$
4
+ #
5
+ # Copyright (c) 1999-2006 Minero Aoki
6
+ #
7
+ # This program is free software.
8
+ # You can distribute/modify this program under the terms of
9
+ # the GNU LGPL, Lesser General Public Lisence version 2.1.
10
+ # For details of the GNU LGPL, see the file "COPYING".
11
+ #
12
+
13
+ require 'racc/info'
14
+ require 'strscan'
15
+ require 'forwardable'
16
+ require 'optparse'
17
+
18
+ def main
19
+ @with_action = true
20
+ @with_header = false
21
+ @with_usercode = false
22
+ cname = 'MyParser'
23
+ input = nil
24
+ output = nil
25
+ parser = OptionParser.new
26
+ parser.banner = "Usage: #{File.basename($0)} [-Ahu] [-c <classname>] [-o <filename>] <input>"
27
+ parser.on('-o', '--output=FILENAME', 'output file name [<input>.racc]') {|name|
28
+ output = name
29
+ }
30
+ parser.on('-c', '--classname=NAME', "Name of the parser class. [#{cname}]") {|name|
31
+ cname = name
32
+ }
33
+ parser.on('-A', '--without-action', 'Does not include actions.') {
34
+ @with_action = false
35
+ }
36
+ parser.on('-h', '--with-header', 'Includes header (%{...%}).') {
37
+ @with_header = true
38
+ }
39
+ parser.on('-u', '--with-user-code', 'Includes user code.') {
40
+ @with_usercode = true
41
+ }
42
+ parser.on('--version', 'Prints version and quit.') {
43
+ puts "y2racc version #{Racc::Version}"
44
+ exit 0
45
+ }
46
+ parser.on('--copyright', 'Prints copyright and quit.') {
47
+ puts Racc::Copyright
48
+ exit 0
49
+ }
50
+ parser.on('--help', 'Prints this message and quit.') {
51
+ puts parser.help
52
+ exit 1
53
+ }
54
+ begin
55
+ parser.parse!
56
+ rescue OptionParser::ParseError => err
57
+ $stderr.puts err.message
58
+ $stderr.puts parser.help
59
+ exit 1
60
+ end
61
+ if ARGV.empty?
62
+ $stderr.puts 'no input'
63
+ exit 1
64
+ end
65
+ if ARGV.size > 1
66
+ $stderr.puts 'too many input'
67
+ exit 1
68
+ end
69
+ input = ARGV[0]
70
+
71
+ begin
72
+ result = YaccFileParser.parse_file(input)
73
+ File.open(output || "#{input}.racc", 'w') {|f|
74
+ convert cname, result, f
75
+ }
76
+ rescue SystemCallError => err
77
+ $stderr.puts err.message
78
+ exit 1
79
+ end
80
+ end
81
+
82
+ def convert(classname, result, f)
83
+ init_indent = 'token'.size
84
+ f.puts %<# Converted from "#{result.filename}" by y2racc version #{Racc::Version}>
85
+ f.puts
86
+ f.puts "class #{classname}"
87
+ unless result.terminals.empty?
88
+ f.puts
89
+ f.print 'token'
90
+ columns = init_indent
91
+ result.terminals.each do |t|
92
+ if columns > 60
93
+ f.puts
94
+ f.print ' ' * init_indent
95
+ columns = init_indent
96
+ end
97
+ columns += f.write(" #{t}")
98
+ end
99
+ f.puts
100
+ end
101
+ unless result.precedence_table.empty?
102
+ f.puts
103
+ f.puts 'preclow'
104
+ result.precedence_table.each do |assoc, toks|
105
+ f.printf " %-8s %s\n", assoc, toks.join(' ') unless toks.empty?
106
+ end
107
+ f.puts 'prechigh'
108
+ end
109
+ if result.start
110
+ f.puts
111
+ f.puts "start #{@start}"
112
+ end
113
+
114
+ f.puts
115
+ f.puts 'rule'
116
+ texts = @with_action ? result.grammar : result.grammar_without_actions
117
+ texts.each do |text|
118
+ f.print text
119
+ end
120
+
121
+ if @with_header and result.header
122
+ f.puts
123
+ f.puts '---- header'
124
+ f.puts result.header
125
+ end
126
+ if @with_usercode and result.usercode
127
+ f.puts
128
+ f.puts '---- footer'
129
+ f.puts result.usercode
130
+ end
131
+ end
132
+
133
+ class ParseError < StandardError; end
134
+
135
+ class StringScanner_withlineno
136
+ def initialize(src)
137
+ @s = StringScanner.new(src)
138
+ @lineno = 1
139
+ end
140
+
141
+ extend Forwardable
142
+ def_delegator "@s", :eos?
143
+ def_delegator "@s", :rest
144
+
145
+ attr_reader :lineno
146
+
147
+ def scan(re)
148
+ advance_lineno(@s.scan(re))
149
+ end
150
+
151
+ def scan_until(re)
152
+ advance_lineno(@s.scan_until(re))
153
+ end
154
+
155
+ def skip(re)
156
+ str = advance_lineno(@s.scan(re))
157
+ str ? str.size : nil
158
+ end
159
+
160
+ def getch
161
+ advance_lineno(@s.getch)
162
+ end
163
+
164
+ private
165
+
166
+ def advance_lineno(str)
167
+ @lineno += str.count("\n") if str
168
+ str
169
+ end
170
+ end
171
+
172
+ class YaccFileParser
173
+
174
+ Result = Struct.new(:terminals, :precedence_table, :start,
175
+ :header, :grammar, :usercode, :filename)
176
+ class Result # reopen
177
+ def initialize
178
+ super
179
+ self.terminals = []
180
+ self.precedence_table = []
181
+ self.start = nil
182
+ self.grammar = []
183
+ self.header = nil
184
+ self.usercode = nil
185
+ self.filename = nil
186
+ end
187
+
188
+ def grammar_without_actions
189
+ grammar().map {|text| text[0,1] == '{' ? '{}' : text }
190
+ end
191
+ end
192
+
193
+ def YaccFileParser.parse_file(filename)
194
+ new().parse(File.read(filename), filename)
195
+ end
196
+
197
+ def parse(src, filename = '-')
198
+ @result = Result.new
199
+ @filename = filename
200
+ @result.filename = filename
201
+ s = StringScanner_withlineno.new(src)
202
+ parse_header s
203
+ parse_grammar s
204
+ @result
205
+ end
206
+
207
+ private
208
+
209
+ COMMENT = %r</\*[^*]*\*+(?:[^/*][^*]*\*+)*/>
210
+ CHAR = /'((?:[^'\\]+|\\.)*)'/
211
+ STRING = /"((?:[^"\\]+|\\.)*)"/
212
+
213
+ def parse_header(s)
214
+ skip_until_percent s
215
+ until s.eos?
216
+ case
217
+ when t = s.scan(/left/)
218
+ @result.precedence_table.push ['left', scan_symbols(s)]
219
+ when t = s.scan(/right/)
220
+ @result.precedence_table.push ['right', scan_symbols(s)]
221
+ when t = s.scan(/nonassoc/)
222
+ @result.precedence_table.push ['nonassoc', scan_symbols(s)]
223
+ when t = s.scan(/token/)
224
+ list = scan_symbols(s)
225
+ list.shift if /\A<(.*)>\z/ =~ list[0]
226
+ @result.terminals.concat list
227
+ when t = s.scan(/start/)
228
+ @result.start = scan_symbols(s)[0]
229
+ when s.skip(%r<(?:
230
+ type | union | expect | thong | binary |
231
+ semantic_parser | pure_parser | no_lines |
232
+ raw | token_table
233
+ )\b>x)
234
+ skip_until_percent s
235
+ when s.skip(/\{/) # header (%{...%})
236
+ str = s.scan_until(/\%\}/)
237
+ str.chop!
238
+ str.chop!
239
+ @result.header = str
240
+ skip_until_percent s
241
+ when s.skip(/\%/) # grammar (%%...)
242
+ return
243
+ else
244
+ raise ParseError, "#{@filename}:#{s.lineno}: scan error"
245
+ end
246
+ end
247
+ end
248
+
249
+ def skip_until_percent(s)
250
+ until s.eos?
251
+ s.skip /[^\%\/]+/
252
+ next if s.skip(COMMENT)
253
+ return if s.getch == '%'
254
+ end
255
+ end
256
+
257
+ def scan_symbols(s)
258
+ list = []
259
+ until s.eos?
260
+ s.skip /\s+/
261
+ if s.skip(COMMENT)
262
+ ;
263
+ elsif t = s.scan(CHAR)
264
+ list.push t
265
+ elsif t = s.scan(STRING)
266
+ list.push t
267
+ elsif s.skip(/\%/)
268
+ break
269
+ elsif t = s.scan(/\S+/)
270
+ list.push t
271
+ else
272
+ raise ParseError, "#{@filename}:#{@lineno}: scan error"
273
+ end
274
+ end
275
+ list
276
+ end
277
+
278
+ def parse_grammar(s)
279
+ buf = []
280
+ until s.eos?
281
+ if t = s.scan(/[^%'"{\/]+/)
282
+ buf.push t
283
+ break if s.eos?
284
+ end
285
+ if s.skip(/\{/)
286
+ buf.push scan_action(s)
287
+ elsif t = s.scan(/'(?:[^'\\]+|\\.)*'/) then buf.push t
288
+ elsif t = s.scan(/"(?:[^"\\]+|\\.)*"/) then buf.push t
289
+ elsif t = s.scan(COMMENT) then buf.push t
290
+ elsif s.skip(/%prec\b/) then buf.push '='
291
+ elsif s.skip(/%%/)
292
+ @result.usercode = s.rest
293
+ break
294
+ else
295
+ buf.push s.getch
296
+ end
297
+ end
298
+ @result.grammar = buf
299
+ end
300
+
301
+ def scan_action(s)
302
+ buf = '{'
303
+ nest = 1
304
+ until s.eos?
305
+ if t = s.scan(%r<[^/{}'"]+>)
306
+ buf << t
307
+ break if s.eos?
308
+ elsif t = s.scan(COMMENT)
309
+ buf << t
310
+ elsif t = s.scan(CHAR)
311
+ buf << t
312
+ elsif t = s.scan(STRING)
313
+ buf << t
314
+ else
315
+ c = s.getch
316
+ buf << c
317
+ case c
318
+ when '{'
319
+ nest += 1
320
+ when '}'
321
+ nest -= 1
322
+ return buf if nest == 0
323
+ end
324
+ end
325
+ end
326
+ $stderr.puts "warning: unterminated action in #{@filename}"
327
+ buf
328
+ end
329
+
330
+ end
331
+
332
+ unless Object.method_defined?(:funcall)
333
+ class Object
334
+ alias funcall __send__
335
+ end
336
+ end
337
+
338
+
339
+ main