-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathparser.lisp
347 lines (316 loc) · 13.8 KB
/
parser.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
;;; Copyright 2012-2020 Google LLC
;;;
;;; Use of this source code is governed by an MIT-style
;;; license that can be found in the LICENSE file or at
;;; https://opensource.org/licenses/MIT.
(in-package #:cl-protobufs.implementation)
;;; Text parsing utilities
(defun-inline proto-whitespace-char-p (ch)
(declare #.*optimize-fast-unsafe*)
(and ch (member ch '(#\space #\tab #\return #\newline))))
(defun-inline proto-hash-char-p (ch)
(declare #.*optimize-fast-unsafe*)
(and ch (eq ch #\#)))
(defun-inline proto-eol-char-p (ch)
(declare #.*optimize-fast-unsafe*)
(and ch (member ch '(#\return #\newline))))
(defun-inline proto-token-char-p (ch)
(declare #.*optimize-fast-unsafe*)
(and ch (or (alpha-char-p ch)
(digit-char-p ch)
(member ch '(#\_ #\.)))))
(defun skip-whitespace-comments-and-chars (stream &key chars)
"Skip all whitespace characters, text-format comments and elements of CHARS
are coming up in the STREAM."
(loop for ch = (peek-char nil stream nil)
until (or (null ch)
(and (not (proto-whitespace-char-p ch))
(not (proto-hash-char-p ch))
(not (if (listp chars)
(member ch chars)
(eql ch chars)))))
do
(if (proto-hash-char-p ch)
(read-line stream nil)
(read-char stream nil))))
(defun skip-whitespace (stream)
"Skip all the whitespace characters that are coming up in the stream."
(loop for ch = (peek-char nil stream nil)
until (or (null ch) (not (proto-whitespace-char-p ch)))
do
(read-char stream nil)))
(defun report-error-with-line (stream error-message)
"It determines the position of an error in the STREAM
and reports it along with the ERROR-MESSAGE, line number, the content of the line and a
caret string that visually marks the error position in the line."
(let ((error-pos (file-position stream))
error-line
error-line-number
error-line-start-pos)
(file-position stream 0)
(loop
:for start-pos = (file-position stream)
:for line = (read-line stream nil)
:for line-number from 0
:until (or (null line) (> start-pos error-pos))
:do (setf
error-line line
error-line-number line-number
error-line-start-pos start-pos))
(let* ((error-column (- error-pos error-line-start-pos))
(indent (length (format nil "Line ~D: " error-line-number)))
(padding-string (format nil "~A^" (make-string (+ indent error-column)
:initial-element #\Space))))
(protobuf-error (format nil "~A~%Line ~D: ~A~%~A"
error-message error-line-number error-line padding-string)))))
(defun expect-matching-end (stream start-char)
"Expect that the starting block element START-CHAR matches the next element
in the STREAM which should end the block, signal an error if there's no match.
The return value is the character that was eaten."
(let ((end-char (peek-char nil stream nil)))
(unless (or (and (eq start-char #\{)
(eq end-char #\}))
(and (eq start-char #\<)
(eq end-char #\>)))
(report-error-with-line stream (format nil
"Opening character ~S doesn't have a matching closing character, found ~S instead."
start-char end-char)))
(read-char stream)))
(defun expect-char (stream char &optional chars within)
"Expect to see 'char' as the next character in the stream; signal an error if it's not there.
Then skip all of the following whitespace.
The return value is the character that was eaten."
(let (ch)
(if (if (listp char)
(member (peek-char nil stream nil) char)
(eql (peek-char nil stream nil) char))
(setq ch (read-char stream))
(protobuf-error "No ~S found~@[ within '~A'~] at position ~D"
char within (file-position stream)))
(maybe-skip-chars stream chars)
ch))
(defun expect-token-or-string (stream string)
"Expect to see STRING as the next string in STREAM, as parsed by PARSE-TOKEN-OR-STRING.
Signal an error if not present, and return the parsed string."
(let ((str (parse-token-or-string stream)))
(skip-whitespace stream)
(if (string= str string)
str
(error "No ~S found at position ~D" string (file-position stream)))))
(defun maybe-skip-chars (stream chars)
"Skip some optional characters in the stream,
then skip all of the following whitespace."
(skip-whitespace-comments-and-chars stream)
(when chars
(loop
(let ((ch (peek-char nil stream nil)))
(when (or (null ch) (not (member ch chars)))
(skip-whitespace-comments-and-chars stream)
(return-from maybe-skip-chars)))
(read-char stream))))
;;--- Collect the comment so we can attach it to its associated object
(defun maybe-skip-comments (stream)
"If what appears next in the stream is a comment, skip it and any following comments,
then skip any following whitespace."
(loop
(let ((ch (peek-char nil stream nil)))
(unless (eql ch #\/)
(return-from maybe-skip-comments))
(read-char stream)
(case (peek-char nil stream nil)
((#\/)
(skip-line-comment stream))
((#\*)
(skip-block-comment stream))
((nil)
(skip-whitespace stream)
(return-from maybe-skip-comments))
(otherwise
(protobuf-error "Found '/' at position ~D to start a comment, but no following '/' or '*'"
(file-position stream)))))))
(defun skip-line-comment (stream)
"Skip to the end of a line comment, that is, to the end of the line.
Then skip any following whitespace."
(loop for ch = (read-char stream nil)
until (or (null ch) (proto-eol-char-p ch)))
(skip-whitespace stream))
(defun skip-block-comment (stream)
"Skip to the end of a block comment, that is, until a '*/' is seen.
Then skip any following whitespace."
(loop for ch = (read-char stream nil)
do (cond ((null ch)
(protobuf-error "Premature end of file while skipping block comment"))
((and (eql ch #\*)
(eql (peek-char nil stream nil) #\/))
(read-char stream nil)
(return))))
(skip-whitespace stream))
(defun parse-token (stream &optional additional-chars)
"Parse the next token in the stream, then skip following whitespace/comments.
The returned value is the token."
(maybe-skip-comments stream)
(when (let ((ch (peek-char nil stream nil)))
(or (proto-token-char-p ch) (member ch additional-chars)))
(loop for ch = (read-char stream nil)
for ch1 = (peek-char nil stream nil)
collect ch into token
until (or (null ch1)
(and (not (proto-token-char-p ch1))
(not (member ch1 additional-chars))))
finally (progn
(skip-whitespace stream)
(maybe-skip-comments stream)
(return (coerce token 'string))))))
(defun parse-parenthesized-token (stream)
"Parse the next token in the stream, then skip the following whitespace.
The token might be surrounded by parentheses.
The returned value is the token."
(let ((left (peek-char nil stream nil)))
(when (eql left #\()
(read-char stream))
(when (proto-token-char-p (peek-char nil stream nil))
(loop for ch = (read-char stream nil)
for ch1 = (peek-char nil stream nil)
collect ch into token
until (or (null ch1) (not (proto-token-char-p ch1)))
finally (progn
(skip-whitespace stream)
(when (eql left #\()
(expect-char stream #\)))
(return (coerce token 'string)))))))
(defun parse-token-or-string (stream)
(if (eql (peek-char nil stream nil) #\")
(values (parse-string stream) 'string)
(values (parse-token stream) 'symbol)))
(defun parse-string (stream)
"Parse the next quoted string in the stream, then skip the following whitespace.
The returned value is the string, without the quotation marks."
(let ((ch0 (read-char stream nil)))
(unless (member ch0 '(#\' #\"))
(protobuf-error "Starting string character ~c should be \' or \"." ch0))
(loop for ch = (read-char stream nil)
until (or (null ch) (char= ch ch0))
when (eql ch #\\)
do (setq ch (unescape-char stream))
collect ch into string
finally (progn
(skip-whitespace-comments-and-chars stream)
(if (eql (peek-char nil stream nil) ch0)
;; If the next character is a quote character, that means
;; we should go parse another string and concatenate it
(return (strcat (coerce string 'string) (parse-string stream)))
(return (coerce string 'string)))))))
(defun unescape-char (stream)
"Parse the next \"escaped\" character from the stream."
(let ((ch (read-char stream nil)))
(assert (not (null ch)) ()
"End of stream reached while reading escaped character")
(case ch
((#\x)
;; Two hex digits
(let* ((d1 (digit-char-p (read-char stream) 16))
(d2 (digit-char-p (read-char stream) 16)))
(code-char (+ (* d1 16) d2))))
((#\0 #\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9)
(if (not (digit-char-p (peek-char nil stream nil)))
#\null
;; Three octal digits
(let* ((d1 (digit-char-p ch 8))
(d2 (digit-char-p (read-char stream) 8))
(d3 (digit-char-p (read-char stream) 8)))
(code-char (+ (* d1 64) (* d2 8) d3)))))
((#\t) #\tab)
((#\n) #\newline)
((#\r) #\return)
((#\f) #\page)
((#\b) #\backspace)
((#\a) #\bell)
((#\e) #\esc)
(otherwise ch))))
(defun escape-char (ch)
"The inverse of 'unescape-char', for printing."
(if (and (standard-char-p ch) (graphic-char-p ch))
ch
(case ch
((#\null) "\\0")
((#\tab) "\\t")
((#\newline) "\\n")
((#\return) "\\r")
((#\page) "\\f")
((#\backspace) "\\b")
((#\bell) "\\a")
((#\esc) "\\e")
(otherwise
(format nil "\\x~2,'0X" (char-code ch))))))
(defun parse-signed-int (stream)
"Parse the next token in the stream as an integer, then skip the following whitespace.
The returned value is the integer."
(let* ((sign (if (eql (peek-char nil stream nil) #\-)
(progn (read-char stream) -1)
1))
(int (parse-unsigned-int stream)))
(* int sign)))
(defun parse-unsigned-int (stream)
"Parse the next token in the stream as an integer, then skip the following whitespace.
The returned value is the integer."
(when (digit-char-p (peek-char nil stream nil))
(loop for ch = (read-char stream nil)
for ch1 = (peek-char nil stream nil)
collect ch into token
until (or (null ch1) (and (not (digit-char-p ch1)) (not (eql ch #\x))))
finally (progn
(skip-whitespace stream)
(let ((token (coerce token 'string)))
(if (starts-with token "0x")
(let ((*read-base* 16))
(return (parse-integer (subseq token 2))))
(return (parse-integer token))))))))
(defun parse-float (stream)
"Parse the next token in the STREAM as a float, then skip the following whitespace.
The returned value is the float."
(let ((number (parse-number stream :allow-inf-nan t)))
(when number
(case number
(:infinity float-features:single-float-positive-infinity)
(:-infinity float-features:single-float-negative-infinity)
(:nan float-features:single-float-nan)
(t (coerce number 'float))))))
(defun parse-double (stream &key append-d0)
"Parse the next token in the STREAM as a double, then skip the following whitespace.
If APPEND-D0 is true, then append 'd0' to the parsed number before attempting to convert
to a double. This is necessary in order to parse doubles from the stream which do not
already have the 'd0' suffix. The returned value is the double-float."
(let ((number (parse-number stream :append-d0 append-d0 :allow-inf-nan t)))
(when number
(case number
(:infinity float-features:double-float-positive-infinity)
(:-infinity float-features:double-float-negative-infinity)
(:nan float-features:DOUBLE-FLOAT-NAN )
(t (coerce number 'double-float))))))
(defun parse-number (stream &key append-d0 allow-inf-nan)
"Parse a number from STREAM. If APPEND-D0 is true, append \"d0\"
to the end of the parsed numerical string. If ALLOW-INF-NAN is
true, allow inifinty or nan values."
(let ((ch (peek-char nil stream nil)))
(when (or (digit-char-p ch)
(member ch '(#\- #\+ #\.))
(and allow-inf-nan
(member ch '(#\i #\n))))
(let ((token (parse-token stream '(#\- #\+ #\.))))
(when token
(skip-whitespace-comments-and-chars stream)
(if append-d0
(parse-numeric-string (concatenate 'string token "d0"))
(parse-numeric-string token)))))))
(defun parse-numeric-string (string)
(cond ((starts-with string "0x")
(parse-integer (subseq string 2) :radix 16))
((starts-with string "-0x")
(- (parse-integer (subseq string 3) :radix 16)))
((member string '("nan") :test #'string-equal) :nan)
((member string '("inf" "infinity") :test #'string-equal)
:infinity)
((member string '("-inf" "-infinity") :test #'string-equal)
:-infinity)
(t
(read-from-string string))))