2 * Copyright (C) 2019 Richard W.M. Jones
3 * Copyright (C) 2019 Red Hat Inc.
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
26 exception SyntaxError of string
29 let pos = lexbuf.lex_curr_p in
31 { pos with pos_bol = pos.pos_cnum; pos_lnum = pos.pos_lnum + 1 }
34 let white = [' ' '\t']+
35 let newline = '\r' | '\n' | "\r\n"
36 let comment = '#' (_#'\n')*
37 let id = ['a'-'z' 'A'-'Z' '_'] ['a'-'z' 'A'-'Z' '0'-'9' '_' '-']*
42 | comment { read lexbuf }
43 | newline { new_line lexbuf; read lexbuf }
52 | '"' { read_string (Ast.Substs.create ()) lexbuf }
53 | "{" { read_code false (Ast.Substs.create ()) (ref 1) lexbuf }
54 | "@{" { read_code true (Ast.Substs.create ()) (ref 1) lexbuf }
74 | "*" id { (* NB: The initial '*' is part of the name. *)
75 TACTIC (Lexing.lexeme lexbuf) }
76 | id { ID (Lexing.lexeme lexbuf) }
77 | _ { raise (SyntaxError ("unexpected character: " ^
78 Lexing.lexeme lexbuf)) }
81 (* Parse "STRING" literal with %-substitutions. *)
85 { Ast.Substs.add_char buf '"'; read_string buf lexbuf }
87 { Ast.Substs.add_char buf '\007'; read_string buf lexbuf }
89 { Ast.Substs.add_char buf '\008'; read_string buf lexbuf }
91 { Ast.Substs.add_char buf '\009'; read_string buf lexbuf }
93 { Ast.Substs.add_char buf '\010'; read_string buf lexbuf }
95 { Ast.Substs.add_char buf '\011'; read_string buf lexbuf }
97 { Ast.Substs.add_char buf '\012'; read_string buf lexbuf }
99 { Ast.Substs.add_char buf '\013'; read_string buf lexbuf }
101 { Ast.Substs.add_char buf '\\'; read_string buf lexbuf }
102 | '"' { STRING (Ast.Substs.get buf) }
103 | newline { Ast.Substs.add_char buf '\n';
104 new_line lexbuf; read_string buf lexbuf }
105 | '%' '%' { Ast.Substs.add_char buf '%'; read_string buf lexbuf }
106 | '%' id { let id = Lexing.lexeme lexbuf in
107 let len = String.length id in
108 Ast.Substs.add_var buf (String.sub id 1 (len-1));
109 read_string buf lexbuf }
110 | '%' _ { raise (SyntaxError ("illegal character in %-substitution: " ^
111 Lexing.lexeme lexbuf)) }
112 | [^ '"' '\\' '\r' '\n' '%' ]+
113 { Ast.Substs.add_string buf (Lexing.lexeme lexbuf);
114 read_string buf lexbuf }
115 | _ { raise (SyntaxError ("illegal character in string: " ^
116 Lexing.lexeme lexbuf)) }
117 | eof { raise (SyntaxError ("unterminated string")) }
119 (* Parse { CODE } literal with %-substitutions.
121 * Note the range of %-substitutions possible is larger than
124 and read_code quiet buf level =
126 | '{' { Ast.Substs.add_char buf '{';
127 incr level; read_code quiet buf level lexbuf }
129 if !level = 0 then CODE (Ast.Substs.get buf, quiet)
131 Ast.Substs.add_char buf '}';
132 read_code quiet buf level lexbuf
134 | newline { Ast.Substs.add_char buf '\n';
135 new_line lexbuf; read_code quiet buf level lexbuf }
136 | '%' '%' { Ast.Substs.add_char buf '%'; read_code quiet buf level lexbuf }
137 | '%' '@' { Ast.Substs.add_var buf "@"; read_code quiet buf level lexbuf }
138 | '%' '<' { Ast.Substs.add_var buf "<"; read_code quiet buf level lexbuf }
139 | '%' '^' { Ast.Substs.add_var buf "^"; read_code quiet buf level lexbuf }
140 | '%' id { let id = Lexing.lexeme lexbuf in
141 let len = String.length id in
142 Ast.Substs.add_var buf (String.sub id 1 (len-1));
143 read_code quiet buf level lexbuf }
144 | '%' _ { raise (SyntaxError ("illegal character in %-substitution: " ^
145 Lexing.lexeme lexbuf)) }
146 | [^ '{' '}' '\r' '\n' '%' ]+
147 { Ast.Substs.add_string buf (Lexing.lexeme lexbuf);
148 read_code quiet buf level lexbuf }
149 | _ { raise (SyntaxError ("illegal character in code section: " ^
150 Lexing.lexeme lexbuf)) }
151 | eof { raise (SyntaxError ("unterminated code section")) }