532
|
1 |
% Chapter Template
|
|
2 |
|
|
3 |
\chapter{Regular Expressions and POSIX Lexing} % Main chapter title
|
|
4 |
|
|
5 |
\label{Inj} % In chapter 2 \ref{Chapter2} we will introduce the concepts
|
|
6 |
%and notations we
|
|
7 |
%use for describing the lexing algorithm by Sulzmann and Lu,
|
|
8 |
%and then give the algorithm and its variant, and discuss
|
|
9 |
%why more aggressive simplifications are needed.
|
|
10 |
|
|
11 |
|
|
12 |
\section{Basic Concepts and Notations for Strings, Languages, and Regular Expressions}
|
|
13 |
We have a primitive datatype char, denoting characters.
|
|
14 |
\[ char ::= a
|
|
15 |
\mid b
|
|
16 |
\mid c
|
|
17 |
\mid \ldots
|
|
18 |
\mid z
|
|
19 |
\]
|
|
20 |
(which one is better?)
|
|
21 |
\begin{center}
|
|
22 |
\begin{tabular}{lcl}
|
|
23 |
$\textit{char}$ & $\dn$ & $a | b | c | \ldots$\\
|
|
24 |
\end{tabular}
|
|
25 |
\end{center}
|
|
26 |
They can form strings by lists:
|
|
27 |
\begin{center}
|
|
28 |
\begin{tabular}{lcl}
|
|
29 |
$\textit{string}$ & $\dn$ & $[] | c :: cs$\\
|
|
30 |
& & $(c\; \text{has char type})$
|
|
31 |
\end{tabular}
|
|
32 |
\end{center}
|
|
33 |
And strings can be concatenated to form longer strings:
|
|
34 |
\begin{center}
|
|
35 |
\begin{tabular}{lcl}
|
|
36 |
$[] @ s_2$ & $\dn$ & $s_2$\\
|
|
37 |
$(c :: s_1) @ s_2$ & $\dn$ & $c :: (s_1 @ s_2)$
|
|
38 |
\end{tabular}
|
|
39 |
\end{center}
|
|
40 |
|
|
41 |
A set of strings can operate with another set of strings:
|
|
42 |
\begin{center}
|
|
43 |
\begin{tabular}{lcl}
|
|
44 |
$A @ B $ & $\dn$ & $\{s_A @ s_B \mid s_A \in A; s_B \in B \}$\\
|
|
45 |
\end{tabular}
|
|
46 |
\end{center}
|
|
47 |
We also call the above "language concatenation".
|
|
48 |
The power of a language is defined recursively, using the
|
|
49 |
concatenation operator $@$:
|
|
50 |
\begin{center}
|
|
51 |
\begin{tabular}{lcl}
|
|
52 |
$A^0 $ & $\dn$ & $\{ [] \}$\\
|
|
53 |
$A^{n+1}$ & $\dn$ & $A^n @ A$
|
|
54 |
\end{tabular}
|
|
55 |
\end{center}
|
|
56 |
The union of all the natural number powers of a language
|
|
57 |
is denoted by the Kleene star operator:
|
|
58 |
\begin{center}
|
|
59 |
\begin{tabular}{lcl}
|
536
|
60 |
$A*$ & $\dn$ & $\bigcup_{i \geq 0} A^i$ \\
|
532
|
61 |
\end{tabular}
|
|
62 |
\end{center}
|
|
63 |
|
536
|
64 |
To get an induction principle in Isabelle/HOL,
|
|
65 |
we instead define the Kleene star
|
532
|
66 |
as an inductive set:
|
|
67 |
\begin{center}
|
|
68 |
\begin{tabular}{lcl}
|
536
|
69 |
$[] \in A*$ & &\\
|
|
70 |
$s_1 \in A \land \; s_2 \in A* $ & $\implies$ & $s_1 @ s_2 \in A*$\\
|
532
|
71 |
\end{tabular}
|
|
72 |
\end{center}
|
|
73 |
\subsection{Language Derivatives}
|
|
74 |
We also define an operation of chopping off a character from
|
|
75 |
a language:
|
|
76 |
\begin{center}
|
|
77 |
\begin{tabular}{lcl}
|
|
78 |
$\textit{Der} \;c \;A$ & $\dn$ & $\{ s \mid c :: s \in A \}$\\
|
|
79 |
\end{tabular}
|
|
80 |
\end{center}
|
|
81 |
|
|
82 |
This can be generalised to chopping off a string from all strings within set $A$:
|
|
83 |
\begin{center}
|
|
84 |
\begin{tabular}{lcl}
|
|
85 |
$\textit{Ders} \;w \;A$ & $\dn$ & $\{ s \mid w@s \in A \}$\\
|
|
86 |
\end{tabular}
|
|
87 |
\end{center}
|
|
88 |
|
|
89 |
which is essentially the left quotient $A \backslash L'$ of $A$ against
|
|
90 |
the singleton language $L' = \{w\}$
|
|
91 |
in formal language theory.
|
536
|
92 |
For this dissertation the $\textit{Ders}$ definition with
|
|
93 |
a single string suffices.
|
532
|
94 |
|
|
95 |
With the sequencing, Kleene star, and $\textit{Der}$ operator on languages,
|
|
96 |
we have a few properties of how the language derivative can be defined using
|
|
97 |
sub-languages.
|
|
98 |
\begin{lemma}
|
536
|
99 |
\[
|
|
100 |
\Der \; c \; (A @ B) =
|
|
101 |
\begin{cases}
|
|
102 |
((\Der \; c \; A) @ B ) \cup \Der \; c\; B, & \text{if} \; [] \in A \\
|
|
103 |
(\Der \; c \; A) @ B, & \text{otherwise}
|
|
104 |
\end{cases}
|
|
105 |
\]
|
532
|
106 |
\end{lemma}
|
|
107 |
\noindent
|
|
108 |
This lemma states that if $A$ contains the empty string, $\Der$ can "pierce" through it
|
|
109 |
and get to $B$.
|
|
110 |
|
536
|
111 |
The language $A*$'s derivative can be described using the language derivative
|
532
|
112 |
of $A$:
|
|
113 |
\begin{lemma}
|
536
|
114 |
$\textit{Der} \;c \;A* = (\textit{Der}\; c A) @ (A*)$\\
|
532
|
115 |
\end{lemma}
|
|
116 |
\begin{proof}
|
|
117 |
\begin{itemize}
|
|
118 |
\item{$\subseteq$}
|
|
119 |
The set
|
536
|
120 |
\[ \{s \mid c :: s \in A*\} \]
|
532
|
121 |
is enclosed in the set
|
536
|
122 |
\[ \{s_1 @ s_2 \mid s_1 \, s_2. s_1 \in \{s \mid c :: s \in A\} \land s_2 \in A* \} \]
|
532
|
123 |
because whenever you have a string starting with a character
|
536
|
124 |
in the language of a Kleene star $A*$, then that character together with some sub-string
|
532
|
125 |
immediately after it will form the first iteration, and the rest of the string will
|
536
|
126 |
be still in $A*$.
|
532
|
127 |
\item{$\supseteq$}
|
|
128 |
Note that
|
536
|
129 |
\[ \Der \; c \; A* = \Der \; c \; (\{ [] \} \cup (A @ A*) ) \]
|
532
|
130 |
and
|
536
|
131 |
\[ \Der \; c \; (\{ [] \} \cup (A @ A*) ) = \Der\; c \; (A @ A*) \]
|
532
|
132 |
where the $\textit{RHS}$ of the above equatioin can be rewritten
|
536
|
133 |
as \[ (\Der \; c\; A) @ A* \cup A' \], $A'$ being a possibly empty set.
|
532
|
134 |
\end{itemize}
|
|
135 |
\end{proof}
|
|
136 |
Before we define the $\textit{Der}$ and $\textit{Ders}$ counterpart
|
|
137 |
for regular languages, we need to first give definitions for regular expressions.
|
|
138 |
|
536
|
139 |
\subsection{Regular Expressions and Their Meaning}
|
532
|
140 |
Suppose we have an alphabet $\Sigma$, the strings whose characters
|
|
141 |
are from $\Sigma$
|
|
142 |
can be expressed as $\Sigma^*$.
|
|
143 |
|
|
144 |
We use patterns to define a set of strings concisely. Regular expressions
|
|
145 |
are one of such patterns systems:
|
|
146 |
The basic regular expressions are defined inductively
|
|
147 |
by the following grammar:
|
|
148 |
\[ r ::= \ZERO \mid \ONE
|
|
149 |
\mid c
|
|
150 |
\mid r_1 \cdot r_2
|
|
151 |
\mid r_1 + r_2
|
|
152 |
\mid r^*
|
|
153 |
\]
|
|
154 |
|
|
155 |
The language or set of strings denoted
|
|
156 |
by regular expressions are defined as
|
|
157 |
%TODO: FILL in the other defs
|
|
158 |
\begin{center}
|
|
159 |
\begin{tabular}{lcl}
|
536
|
160 |
$L \; (\ZERO)$ & $\dn$ & $\phi$\\
|
|
161 |
$L \; (\ONE)$ & $\dn$ & $\{[]\}$\\
|
|
162 |
$L \; (c)$ & $\dn$ & $\{[c]\}$\\
|
532
|
163 |
$L \; (r_1 + r_2)$ & $\dn$ & $ L \; (r_1) \cup L \; ( r_2)$\\
|
|
164 |
$L \; (r_1 \cdot r_2)$ & $\dn$ & $ L \; (r_1) \cap L \; (r_2)$\\
|
536
|
165 |
$L \; (r^*)$ & $\dn$ & $ (L(r))^*$
|
532
|
166 |
\end{tabular}
|
|
167 |
\end{center}
|
536
|
168 |
\noindent
|
532
|
169 |
Which is also called the "language interpretation" of
|
|
170 |
a regular expression.
|
|
171 |
|
|
172 |
Now with semantic derivatives of a language and regular expressions and
|
|
173 |
their language interpretations in place, we are ready to define derivatives on regexes.
|
|
174 |
\subsection{Brzozowski Derivatives and a Regular Expression Matcher}
|
536
|
175 |
The language derivative acts on a string set and chops off a character from
|
532
|
176 |
all strings in that set, we want to define a derivative operation on regular expressions
|
|
177 |
so that after derivative $L(r\backslash c)$
|
|
178 |
will look as if it was obtained by doing a language derivative on $L(r)$:
|
|
179 |
\[
|
|
180 |
L(r \backslash c) = \Der \; c \; L(r)
|
|
181 |
\]
|
|
182 |
So we mimic the equalities we have for $\Der$ on language concatenation
|
|
183 |
|
|
184 |
\[
|
|
185 |
\Der \; c \; (A @ B) = \textit{if} \; [] \in A \; \textit{then} ((\Der \; c \; A) @ B ) \cup \Der \; c\; B \quad \textit{else}\; (\Der \; c \; A) @ B\\
|
|
186 |
\]
|
|
187 |
to get the derivative for sequence regular expressions:
|
|
188 |
\[
|
|
189 |
(r_1 \cdot r_2 ) \backslash c = \textit{if}\,([] \in L(r_1)) r_1 \backslash c \cdot r_2 + r_2 \backslash c \textit{else} (r_1 \backslash c) \cdot r_2
|
|
190 |
\]
|
|
191 |
|
|
192 |
\noindent
|
|
193 |
and language Kleene star:
|
|
194 |
\[
|
536
|
195 |
\textit{Der} \;c \;A* = (\textit{Der}\; c A) @ (A*)
|
532
|
196 |
\]
|
|
197 |
to get derivative of the Kleene star regular expression:
|
|
198 |
\[
|
|
199 |
r^* \backslash c = (r \backslash c)\cdot r^*
|
|
200 |
\]
|
|
201 |
Note that although we can formalise the boolean predicate
|
|
202 |
$[] \in L(r_1)$ without problems, if we want a function that works
|
|
203 |
computationally, then we would have to define a function that tests
|
|
204 |
whether an empty string is in the language of a regular expression.
|
|
205 |
We call such a function $\nullable$:
|
|
206 |
|
|
207 |
|
|
208 |
|
|
209 |
\begin{center}
|
|
210 |
\begin{tabular}{lcl}
|
|
211 |
$\ZERO \backslash c$ & $\dn$ & $\ZERO$\\
|
|
212 |
$\ONE \backslash c$ & $\dn$ & $\ZERO$\\
|
|
213 |
$d \backslash c$ & $\dn$ &
|
|
214 |
$\mathit{if} \;c = d\;\mathit{then}\;\ONE\;\mathit{else}\;\ZERO$\\
|
|
215 |
$(r_1 + r_2)\backslash c$ & $\dn$ & $r_1 \backslash c \,+\, r_2 \backslash c$\\
|
|
216 |
$(r_1 \cdot r_2)\backslash c$ & $\dn$ & $\mathit{if} \, nullable(r_1)$\\
|
|
217 |
& & $\mathit{then}\;(r_1\backslash c) \cdot r_2 \,+\, r_2\backslash c$\\
|
|
218 |
& & $\mathit{else}\;(r_1\backslash c) \cdot r_2$\\
|
|
219 |
$(r^*)\backslash c$ & $\dn$ & $(r\backslash c) \cdot r^*$\\
|
|
220 |
\end{tabular}
|
|
221 |
\end{center}
|
|
222 |
\noindent
|
|
223 |
The function derivative, written $r\backslash c$,
|
|
224 |
defines how a regular expression evolves into
|
|
225 |
a new regular expression after all the string it contains
|
|
226 |
is chopped off a certain head character $c$.
|
|
227 |
The most involved cases are the sequence
|
|
228 |
and star case.
|
|
229 |
The sequence case says that if the first regular expression
|
|
230 |
contains an empty string then the second component of the sequence
|
|
231 |
might be chosen as the target regular expression to be chopped
|
|
232 |
off its head character.
|
|
233 |
The star regular expression's derivative unwraps the iteration of
|
|
234 |
regular expression and attaches the star regular expression
|
|
235 |
to the sequence's second element to make sure a copy is retained
|
|
236 |
for possible more iterations in later phases of lexing.
|
|
237 |
|
|
238 |
|
|
239 |
The $\nullable$ function tests whether the empty string $""$
|
|
240 |
is in the language of $r$:
|
|
241 |
|
|
242 |
|
|
243 |
\begin{center}
|
|
244 |
\begin{tabular}{lcl}
|
|
245 |
$\nullable(\ZERO)$ & $\dn$ & $\mathit{false}$ \\
|
|
246 |
$\nullable(\ONE)$ & $\dn$ & $\mathit{true}$ \\
|
|
247 |
$\nullable(c)$ & $\dn$ & $\mathit{false}$ \\
|
|
248 |
$\nullable(r_1 + r_2)$ & $\dn$ & $\nullable(r_1) \vee \nullable(r_2)$ \\
|
|
249 |
$\nullable(r_1\cdot r_2)$ & $\dn$ & $\nullable(r_1) \wedge \nullable(r_2)$ \\
|
|
250 |
$\nullable(r^*)$ & $\dn$ & $\mathit{true}$ \\
|
|
251 |
\end{tabular}
|
|
252 |
\end{center}
|
|
253 |
\noindent
|
|
254 |
The empty set does not contain any string and
|
|
255 |
therefore not the empty string, the empty string
|
|
256 |
regular expression contains the empty string
|
|
257 |
by definition, the character regular expression
|
|
258 |
is the singleton that contains character only,
|
|
259 |
and therefore does not contain the empty string,
|
|
260 |
the alternative regular expression (or "or" expression)
|
|
261 |
might have one of its children regular expressions
|
|
262 |
being nullable and any one of its children being nullable
|
|
263 |
would suffice. The sequence regular expression
|
|
264 |
would require both children to have the empty string
|
|
265 |
to compose an empty string and the Kleene star
|
|
266 |
operation naturally introduced the empty string.
|
|
267 |
|
|
268 |
We have the following property where the derivative on regular
|
|
269 |
expressions coincides with the derivative on a set of strings:
|
|
270 |
|
|
271 |
\begin{lemma}
|
|
272 |
$\textit{Der} \; c \; L(r) = L (r\backslash c)$
|
|
273 |
\end{lemma}
|
|
274 |
|
|
275 |
\noindent
|
|
276 |
|
|
277 |
|
|
278 |
The main property of the derivative operation
|
|
279 |
that enables us to reason about the correctness of
|
|
280 |
an algorithm using derivatives is
|
|
281 |
|
|
282 |
\begin{center}
|
|
283 |
$c\!::\!s \in L(r)$ holds
|
|
284 |
if and only if $s \in L(r\backslash c)$.
|
|
285 |
\end{center}
|
|
286 |
|
|
287 |
\noindent
|
|
288 |
We can generalise the derivative operation shown above for single characters
|
|
289 |
to strings as follows:
|
|
290 |
|
|
291 |
\begin{center}
|
|
292 |
\begin{tabular}{lcl}
|
|
293 |
$r \backslash_s (c\!::\!s) $ & $\dn$ & $(r \backslash c) \backslash_s s$ \\
|
|
294 |
$r \backslash [\,] $ & $\dn$ & $r$
|
|
295 |
\end{tabular}
|
|
296 |
\end{center}
|
|
297 |
|
|
298 |
\noindent
|
|
299 |
When there is no ambiguity we will use $\backslash$ to denote
|
|
300 |
string derivatives for brevity.
|
|
301 |
|
|
302 |
and then define Brzozowski's regular-expression matching algorithm as:
|
|
303 |
|
|
304 |
\begin{definition}
|
|
305 |
$match\;s\;r \;\dn\; nullable(r\backslash s)$
|
|
306 |
\end{definition}
|
|
307 |
|
|
308 |
\noindent
|
|
309 |
Assuming the string is given as a sequence of characters, say $c_0c_1..c_n$,
|
|
310 |
this algorithm presented graphically is as follows:
|
|
311 |
|
|
312 |
\begin{equation}\label{graph:successive_ders}
|
|
313 |
\begin{tikzcd}
|
|
314 |
r_0 \arrow[r, "\backslash c_0"] & r_1 \arrow[r, "\backslash c_1"] & r_2 \arrow[r, dashed] & r_n \arrow[r,"\textit{nullable}?"] & \;\textrm{YES}/\textrm{NO}
|
|
315 |
\end{tikzcd}
|
|
316 |
\end{equation}
|
|
317 |
|
|
318 |
\noindent
|
|
319 |
where we start with a regular expression $r_0$, build successive
|
|
320 |
derivatives until we exhaust the string and then use \textit{nullable}
|
|
321 |
to test whether the result can match the empty string. It can be
|
|
322 |
relatively easily shown that this matcher is correct (that is given
|
|
323 |
an $s = c_0...c_{n-1}$ and an $r_0$, it generates YES if and only if $s \in L(r_0)$).
|
|
324 |
|
|
325 |
Beautiful and simple definition.
|
|
326 |
|
|
327 |
If we implement the above algorithm naively, however,
|
|
328 |
the algorithm can be excruciatingly slow.
|
|
329 |
|
|
330 |
|
|
331 |
\begin{figure}
|
|
332 |
\centering
|
|
333 |
\begin{tabular}{@{}c@{\hspace{0mm}}c@{\hspace{0mm}}c@{}}
|
|
334 |
\begin{tikzpicture}
|
|
335 |
\begin{axis}[
|
|
336 |
xlabel={$n$},
|
|
337 |
x label style={at={(1.05,-0.05)}},
|
|
338 |
ylabel={time in secs},
|
|
339 |
enlargelimits=false,
|
|
340 |
xtick={0,5,...,30},
|
|
341 |
xmax=33,
|
|
342 |
ymax=10000,
|
|
343 |
ytick={0,1000,...,10000},
|
|
344 |
scaled ticks=false,
|
|
345 |
axis lines=left,
|
|
346 |
width=5cm,
|
|
347 |
height=4cm,
|
|
348 |
legend entries={JavaScript},
|
|
349 |
legend pos=north west,
|
|
350 |
legend cell align=left]
|
|
351 |
\addplot[red,mark=*, mark options={fill=white}] table {EightThousandNodes.data};
|
|
352 |
\end{axis}
|
|
353 |
\end{tikzpicture}\\
|
|
354 |
\multicolumn{3}{c}{Graphs: Runtime for matching $(a^*)^*\,b$ with strings
|
|
355 |
of the form $\underbrace{aa..a}_{n}$.}
|
|
356 |
\end{tabular}
|
|
357 |
\caption{EightThousandNodes} \label{fig:EightThousandNodes}
|
|
358 |
\end{figure}
|
|
359 |
|
|
360 |
|
|
361 |
(8000 node data to be added here)
|
|
362 |
For example, when starting with the regular
|
|
363 |
expression $(a + aa)^*$ and building a few successive derivatives (around 10)
|
|
364 |
w.r.t.~the character $a$, one obtains a derivative regular expression
|
|
365 |
with more than 8000 nodes (when viewed as a tree)\ref{EightThousandNodes}.
|
|
366 |
The reason why $(a + aa) ^*$ explodes so drastically is that without
|
|
367 |
pruning, the algorithm will keep records of all possible ways of matching:
|
|
368 |
\begin{center}
|
|
369 |
$(a + aa) ^* \backslash [aa] = (\ZERO + \ONE \ONE)\cdot(a + aa)^* + (\ONE + \ONE a) \cdot (a + aa)^*$
|
|
370 |
\end{center}
|
|
371 |
|
|
372 |
\noindent
|
|
373 |
Each of the above alternative branches correspond to the match
|
|
374 |
$aa $, $a \quad a$ and $a \quad a \cdot (a)$(incomplete).
|
|
375 |
These different ways of matching will grow exponentially with the string length,
|
|
376 |
and without simplifications that throw away some of these very similar matchings,
|
|
377 |
it is no surprise that these expressions grow so quickly.
|
|
378 |
Operations like
|
|
379 |
$\backslash$ and $\nullable$ need to traverse such trees and
|
|
380 |
consequently the bigger the size of the derivative the slower the
|
|
381 |
algorithm.
|
|
382 |
|
|
383 |
Brzozowski was quick in finding that during this process a lot useless
|
|
384 |
$\ONE$s and $\ZERO$s are generated and therefore not optimal.
|
|
385 |
He also introduced some "similarity rules", such
|
|
386 |
as $P+(Q+R) = (P+Q)+R$ to merge syntactically
|
|
387 |
different but language-equivalent sub-regexes to further decrease the size
|
|
388 |
of the intermediate regexes.
|
|
389 |
|
|
390 |
More simplifications are possible, such as deleting duplicates
|
|
391 |
and opening up nested alternatives to trigger even more simplifications.
|
|
392 |
And suppose we apply simplification after each derivative step, and compose
|
|
393 |
these two operations together as an atomic one: $a \backslash_{simp}\,c \dn
|
|
394 |
\textit{simp}(a \backslash c)$. Then we can build
|
|
395 |
a matcher with simpler regular expressions.
|
|
396 |
|
|
397 |
If we want the size of derivatives in the algorithm to
|
|
398 |
stay even lower, we would need more aggressive simplifications.
|
|
399 |
Essentially we need to delete useless $\ZERO$s and $\ONE$s, as well as
|
|
400 |
delete duplicates whenever possible. For example, the parentheses in
|
|
401 |
$(a+b) \cdot c + b\cdot c$ can be opened up to get $a\cdot c + b \cdot c + b
|
|
402 |
\cdot c$, and then simplified to just $a \cdot c + b \cdot c$. Another
|
|
403 |
example is simplifying $(a^*+a) + (a^*+ \ONE) + (a +\ONE)$ to just
|
|
404 |
$a^*+a+\ONE$. These more aggressive simplification rules are for
|
|
405 |
a very tight size bound, possibly as low
|
|
406 |
as that of the \emph{partial derivatives}\parencite{Antimirov1995}.
|
|
407 |
|
|
408 |
Building derivatives and then simplifying them.
|
|
409 |
So far, so good. But what if we want to
|
|
410 |
do lexing instead of just getting a YES/NO answer?
|
|
411 |
This requires us to go back again to the world
|
|
412 |
without simplification first for a moment.
|
|
413 |
Sulzmann and Lu~\cite{Sulzmann2014} first came up with a nice and
|
|
414 |
elegant(arguably as beautiful as the original
|
|
415 |
derivatives definition) solution for this.
|
|
416 |
|
|
417 |
\subsection*{Values and the Lexing Algorithm by Sulzmann and Lu}
|
|
418 |
|
|
419 |
|
|
420 |
They first defined the datatypes for storing the
|
|
421 |
lexing information called a \emph{value} or
|
|
422 |
sometimes also \emph{lexical value}. These values and regular
|
|
423 |
expressions correspond to each other as illustrated in the following
|
|
424 |
table:
|
|
425 |
|
|
426 |
\begin{center}
|
|
427 |
\begin{tabular}{c@{\hspace{20mm}}c}
|
|
428 |
\begin{tabular}{@{}rrl@{}}
|
|
429 |
\multicolumn{3}{@{}l}{\textbf{Regular Expressions}}\medskip\\
|
|
430 |
$r$ & $::=$ & $\ZERO$\\
|
|
431 |
& $\mid$ & $\ONE$ \\
|
|
432 |
& $\mid$ & $c$ \\
|
|
433 |
& $\mid$ & $r_1 \cdot r_2$\\
|
|
434 |
& $\mid$ & $r_1 + r_2$ \\
|
|
435 |
\\
|
|
436 |
& $\mid$ & $r^*$ \\
|
|
437 |
\end{tabular}
|
|
438 |
&
|
|
439 |
\begin{tabular}{@{\hspace{0mm}}rrl@{}}
|
|
440 |
\multicolumn{3}{@{}l}{\textbf{Values}}\medskip\\
|
|
441 |
$v$ & $::=$ & \\
|
|
442 |
& & $\Empty$ \\
|
|
443 |
& $\mid$ & $\Char(c)$ \\
|
|
444 |
& $\mid$ & $\Seq\,v_1\, v_2$\\
|
|
445 |
& $\mid$ & $\Left(v)$ \\
|
|
446 |
& $\mid$ & $\Right(v)$ \\
|
|
447 |
& $\mid$ & $\Stars\,[v_1,\ldots\,v_n]$ \\
|
|
448 |
\end{tabular}
|
|
449 |
\end{tabular}
|
|
450 |
\end{center}
|
|
451 |
|
|
452 |
\noindent
|
|
453 |
|
|
454 |
Building on top of Sulzmann and Lu's attempt to formalise the
|
|
455 |
notion of POSIX lexing rules \parencite{Sulzmann2014},
|
|
456 |
Ausaf and Urban\parencite{AusafDyckhoffUrban2016} modelled
|
|
457 |
POSIX matching as a ternary relation recursively defined in a
|
|
458 |
natural deduction style.
|
|
459 |
With the formally-specified rules for what a POSIX matching is,
|
|
460 |
they proved in Isabelle/HOL that the algorithm gives correct results.
|
|
461 |
|
|
462 |
But having a correct result is still not enough,
|
|
463 |
we want at least some degree of $\mathbf{efficiency}$.
|
|
464 |
|
|
465 |
|
|
466 |
|
|
467 |
One regular expression can have multiple lexical values. For example
|
|
468 |
for the regular expression $(a+b)^*$, it has a infinite list of
|
|
469 |
values corresponding to it: $\Stars\,[]$, $\Stars\,[\Left(Char(a))]$,
|
|
470 |
$\Stars\,[\Right(Char(b))]$, $\Stars\,[\Left(Char(a),\,\Right(Char(b))]$,
|
|
471 |
$\ldots$, and vice versa.
|
|
472 |
Even for the regular expression matching a particular string, there could
|
|
473 |
be more than one value corresponding to it.
|
|
474 |
Take the example where $r= (a^*\cdot a^*)^*$ and the string
|
|
475 |
$s=\underbrace{aa\ldots a}_\text{n \textit{a}s}$.
|
|
476 |
If we do not allow any empty iterations in its lexical values,
|
|
477 |
there will be $n - 1$ "splitting points" on $s$ we can choose to
|
|
478 |
split or not so that each sub-string
|
|
479 |
segmented by those chosen splitting points will form different iterations:
|
|
480 |
\begin{center}
|
|
481 |
\begin{tabular}{lcr}
|
|
482 |
$a \mid aaa $ & $\rightarrow$ & $\Stars\, [v_{iteration \,a},\, v_{iteration \,aaa}]$\\
|
|
483 |
$aa \mid aa $ & $\rightarrow$ & $\Stars\, [v_{iteration \, aa},\, v_{iteration \, aa}]$\\
|
|
484 |
$a \mid aa\mid a $ & $\rightarrow$ & $\Stars\, [v_{iteration \, a},\, v_{iteration \, aa}, \, v_{iteration \, a}]$\\
|
|
485 |
& $\textit{etc}.$ &
|
|
486 |
\end{tabular}
|
|
487 |
\end{center}
|
|
488 |
|
|
489 |
And for each iteration, there are still multiple ways to split
|
|
490 |
between the two $a^*$s.
|
|
491 |
It is not surprising there are exponentially many lexical values
|
|
492 |
that are distinct for the regex and string pair $r= (a^*\cdot a^*)^*$ and
|
|
493 |
$s=\underbrace{aa\ldots a}_\text{n \textit{a}s}$.
|
|
494 |
|
|
495 |
A lexer to keep all the possible values will naturally
|
|
496 |
have an exponential runtime on ambiguous regular expressions.
|
|
497 |
Somehow one has to decide which lexical value to keep and
|
|
498 |
output in a lexing algorithm.
|
|
499 |
In practice, we are usually
|
|
500 |
interested in POSIX values, which by intuition always
|
|
501 |
\begin{itemize}
|
|
502 |
\item
|
|
503 |
match the leftmost regular expression when multiple options of matching
|
|
504 |
are available
|
|
505 |
\item
|
|
506 |
always match a subpart as much as possible before proceeding
|
|
507 |
to the next token.
|
|
508 |
\end{itemize}
|
|
509 |
The formal definition of a $\POSIX$ value $v$ for a regular expression
|
|
510 |
$r$ and string $s$, denoted as $(s, r) \rightarrow v$, can be specified
|
|
511 |
in the following set of rules:
|
|
512 |
(TODO: write the entire set of inference rules for POSIX )
|
|
513 |
\newcommand*{\inference}[3][t]{%
|
|
514 |
\begingroup
|
|
515 |
\def\and{\\}%
|
|
516 |
\begin{tabular}[#1]{@{\enspace}c@{\enspace}}
|
|
517 |
#2 \\
|
|
518 |
\hline
|
|
519 |
#3
|
|
520 |
\end{tabular}%
|
|
521 |
\endgroup
|
|
522 |
}
|
|
523 |
\begin{center}
|
|
524 |
\inference{$s_1 @ s_2 = s$ \and $(\nexists s_3 s_4 s_5. s_1 @ s_5 = s_3 \land s_5 \neq [] \land s_3 @ s_4 = s \land (s_3, r_1) \rightarrow v_3 \land (s_4, r_2) \rightarrow v_4)$ \and $(s_1, r_1) \rightarrow v_1$ \and $(s_2, r_2) \rightarrow v_2$ }{$(s, r_1 \cdot r_2) \rightarrow \Seq(v_1, v_2)$ }
|
|
525 |
\end{center}
|
|
526 |
|
|
527 |
The reason why we are interested in $\POSIX$ values is that they can
|
|
528 |
be practically used in the lexing phase of a compiler front end.
|
|
529 |
For instance, when lexing a code snippet
|
|
530 |
$\textit{iffoo} = 3$ with the regular expression $\textit{keyword} + \textit{identifier}$, we want $\textit{iffoo}$ to be recognized
|
|
531 |
as an identifier rather than a keyword.
|
|
532 |
|
|
533 |
For example, the above $r= (a^*\cdot a^*)^*$ and
|
|
534 |
$s=\underbrace{aa\ldots a}_\text{n \textit{a}s}$ example has the POSIX value
|
|
535 |
$ \Stars\,[\Seq(Stars\,[\underbrace{\Char(a),\ldots,\Char(a)}_\text{n iterations}], Stars\,[])]$.
|
|
536 |
The output of an algorithm we want would be a POSIX matching
|
|
537 |
encoded as a value.
|
|
538 |
|
|
539 |
|
|
540 |
|
|
541 |
|
|
542 |
The contribution of Sulzmann and Lu is an extension of Brzozowski's
|
|
543 |
algorithm by a second phase (the first phase being building successive
|
|
544 |
derivatives---see \eqref{graph:successive_ders}). In this second phase, a POSIX value
|
|
545 |
is generated if the regular expression matches the string.
|
|
546 |
How can we construct a value out of regular expressions and character
|
|
547 |
sequences only?
|
|
548 |
Two functions are involved: $\inj$ and $\mkeps$.
|
|
549 |
The function $\mkeps$ constructs a value from the last
|
|
550 |
one of all the successive derivatives:
|
|
551 |
\begin{ceqn}
|
|
552 |
\begin{equation}\label{graph:mkeps}
|
|
553 |
\begin{tikzcd}
|
|
554 |
r_0 \arrow[r, "\backslash c_0"] & r_1 \arrow[r, "\backslash c_1"] & r_2 \arrow[r, dashed] & r_n \arrow[d, "mkeps" description] \\
|
|
555 |
& & & v_n
|
|
556 |
\end{tikzcd}
|
|
557 |
\end{equation}
|
|
558 |
\end{ceqn}
|
|
559 |
|
|
560 |
It tells us how can an empty string be matched by a
|
|
561 |
regular expression, in a $\POSIX$ way:
|
|
562 |
|
|
563 |
\begin{center}
|
|
564 |
\begin{tabular}{lcl}
|
|
565 |
$\mkeps(\ONE)$ & $\dn$ & $\Empty$ \\
|
|
566 |
$\mkeps(r_{1}+r_{2})$ & $\dn$
|
|
567 |
& \textit{if} $\nullable(r_{1})$\\
|
|
568 |
& & \textit{then} $\Left(\mkeps(r_{1}))$\\
|
|
569 |
& & \textit{else} $\Right(\mkeps(r_{2}))$\\
|
|
570 |
$\mkeps(r_1\cdot r_2)$ & $\dn$ & $\Seq\,(\mkeps\,r_1)\,(\mkeps\,r_2)$\\
|
|
571 |
$mkeps(r^*)$ & $\dn$ & $\Stars\,[]$
|
|
572 |
\end{tabular}
|
|
573 |
\end{center}
|
|
574 |
|
|
575 |
|
|
576 |
\noindent
|
|
577 |
We favour the left to match an empty string if there is a choice.
|
|
578 |
When there is a star for us to match the empty string,
|
|
579 |
we give the $\Stars$ constructor an empty list, meaning
|
|
580 |
no iterations are taken.
|
|
581 |
|
|
582 |
|
|
583 |
After the $\mkeps$-call, we inject back the characters one by one in order to build
|
|
584 |
the lexical value $v_i$ for how the regex $r_i$ matches the string $s_i$
|
|
585 |
($s_i = c_i \ldots c_{n-1}$ ) from the previous lexical value $v_{i+1}$.
|
|
586 |
After injecting back $n$ characters, we get the lexical value for how $r_0$
|
|
587 |
matches $s$. The POSIX value is maintained throughout the process.
|
|
588 |
For this Sulzmann and Lu defined a function that reverses
|
|
589 |
the ``chopping off'' of characters during the derivative phase. The
|
|
590 |
corresponding function is called \emph{injection}, written
|
|
591 |
$\textit{inj}$; it takes three arguments: the first one is a regular
|
|
592 |
expression ${r_{i-1}}$, before the character is chopped off, the second
|
|
593 |
is a character ${c_{i-1}}$, the character we want to inject and the
|
|
594 |
third argument is the value ${v_i}$, into which one wants to inject the
|
|
595 |
character (it corresponds to the regular expression after the character
|
|
596 |
has been chopped off). The result of this function is a new value.
|
|
597 |
\begin{ceqn}
|
|
598 |
\begin{equation}\label{graph:inj}
|
|
599 |
\begin{tikzcd}
|
|
600 |
r_1 \arrow[r, dashed] \arrow[d]& r_i \arrow[r, "\backslash c_i"] \arrow[d] & r_{i+1} \arrow[r, dashed] \arrow[d] & r_n \arrow[d, "mkeps" description] \\
|
|
601 |
v_1 \arrow[u] & v_i \arrow[l, dashed] & v_{i+1} \arrow[l,"inj_{r_i} c_i"] & v_n \arrow[l, dashed]
|
|
602 |
\end{tikzcd}
|
|
603 |
\end{equation}
|
|
604 |
\end{ceqn}
|
|
605 |
|
|
606 |
|
|
607 |
\noindent
|
|
608 |
The
|
|
609 |
definition of $\textit{inj}$ is as follows:
|
|
610 |
|
|
611 |
\begin{center}
|
|
612 |
\begin{tabular}{l@{\hspace{1mm}}c@{\hspace{1mm}}l}
|
|
613 |
$\textit{inj}\,(c)\,c\,Empty$ & $\dn$ & $Char\,c$\\
|
|
614 |
$\textit{inj}\,(r_1 + r_2)\,c\,\Left(v)$ & $\dn$ & $\Left(\textit{inj}\,r_1\,c\,v)$\\
|
|
615 |
$\textit{inj}\,(r_1 + r_2)\,c\,Right(v)$ & $\dn$ & $Right(\textit{inj}\,r_2\,c\,v)$\\
|
|
616 |
$\textit{inj}\,(r_1 \cdot r_2)\,c\,Seq(v_1,v_2)$ & $\dn$ & $Seq(\textit{inj}\,r_1\,c\,v_1,v_2)$\\
|
|
617 |
$\textit{inj}\,(r_1 \cdot r_2)\,c\,\Left(Seq(v_1,v_2))$ & $\dn$ & $Seq(\textit{inj}\,r_1\,c\,v_1,v_2)$\\
|
|
618 |
$\textit{inj}\,(r_1 \cdot r_2)\,c\,Right(v)$ & $\dn$ & $Seq(\textit{mkeps}(r_1),\textit{inj}\,r_2\,c\,v)$\\
|
|
619 |
$\textit{inj}\,(r^*)\,c\,Seq(v,Stars\,vs)$ & $\dn$ & $Stars((\textit{inj}\,r\,c\,v)\,::\,vs)$\\
|
|
620 |
\end{tabular}
|
|
621 |
\end{center}
|
|
622 |
|
|
623 |
\noindent This definition is by recursion on the ``shape'' of regular
|
|
624 |
expressions and values.
|
|
625 |
The clauses do one thing--identifying the ``hole'' on a
|
|
626 |
value to inject the character back into.
|
|
627 |
For instance, in the last clause for injecting back to a value
|
|
628 |
that would turn into a new star value that corresponds to a star,
|
|
629 |
we know it must be a sequence value. And we know that the first
|
|
630 |
value of that sequence corresponds to the child regex of the star
|
|
631 |
with the first character being chopped off--an iteration of the star
|
|
632 |
that had just been unfolded. This value is followed by the already
|
|
633 |
matched star iterations we collected before. So we inject the character
|
|
634 |
back to the first value and form a new value with this latest iteration
|
|
635 |
being added to the previous list of iterations, all under the $\Stars$
|
|
636 |
top level.
|
|
637 |
|
|
638 |
Putting all the functions $\inj$, $\mkeps$, $\backslash$ together,
|
|
639 |
we have a lexer with the following recursive definition:
|
|
640 |
\begin{center}
|
|
641 |
\begin{tabular}{lcr}
|
|
642 |
$\lexer \; r \; [] $ & $=$ & $\mkeps \; r$\\
|
|
643 |
$\lexer \; r \;c::s$ & $=$ & $\inj \; r \; c (\lexer (r\backslash c) s)$
|
|
644 |
\end{tabular}
|
|
645 |
\end{center}
|
|
646 |
|
|
647 |
Pictorially, the algorithm is as follows:
|
|
648 |
|
|
649 |
\begin{ceqn}
|
|
650 |
\begin{equation}\label{graph:2}
|
|
651 |
\begin{tikzcd}
|
|
652 |
r_0 \arrow[r, "\backslash c_0"] \arrow[d] & r_1 \arrow[r, "\backslash c_1"] \arrow[d] & r_2 \arrow[r, dashed] \arrow[d] & r_n \arrow[d, "mkeps" description] \\
|
|
653 |
v_0 & v_1 \arrow[l,"inj_{r_0} c_0"] & v_2 \arrow[l, "inj_{r_1} c_1"] & v_n \arrow[l, dashed]
|
|
654 |
\end{tikzcd}
|
|
655 |
\end{equation}
|
|
656 |
\end{ceqn}
|
|
657 |
|
|
658 |
|
|
659 |
\noindent
|
|
660 |
For convenience, we shall employ the following notations: the regular
|
|
661 |
expression we start with is $r_0$, and the given string $s$ is composed
|
|
662 |
of characters $c_0 c_1 \ldots c_{n-1}$. In the first phase from the
|
|
663 |
left to right, we build the derivatives $r_1$, $r_2$, \ldots according
|
|
664 |
to the characters $c_0$, $c_1$ until we exhaust the string and obtain
|
|
665 |
the derivative $r_n$. We test whether this derivative is
|
|
666 |
$\textit{nullable}$ or not. If not, we know the string does not match
|
|
667 |
$r$, and no value needs to be generated. If yes, we start building the
|
|
668 |
values incrementally by \emph{injecting} back the characters into the
|
|
669 |
earlier values $v_n, \ldots, v_0$. This is the second phase of the
|
|
670 |
algorithm from right to left. For the first value $v_n$, we call the
|
|
671 |
function $\textit{mkeps}$, which builds a POSIX lexical value
|
|
672 |
for how the empty string has been matched by the (nullable) regular
|
|
673 |
expression $r_n$. This function is defined as
|
|
674 |
|
|
675 |
|
|
676 |
|
|
677 |
We have mentioned before that derivatives without simplification
|
|
678 |
can get clumsy, and this is true for values as well--they reflect
|
|
679 |
the size of the regular expression by definition.
|
|
680 |
|
|
681 |
One can introduce simplification on the regex and values but have to
|
|
682 |
be careful not to break the correctness, as the injection
|
|
683 |
function heavily relies on the structure of the regexes and values
|
|
684 |
being correct and matching each other.
|
|
685 |
It can be achieved by recording some extra rectification functions
|
|
686 |
during the derivatives step, and applying these rectifications in
|
|
687 |
each run during the injection phase.
|
|
688 |
And we can prove that the POSIX value of how
|
|
689 |
regular expressions match strings will not be affected---although it is much harder
|
|
690 |
to establish.
|
|
691 |
Some initial results in this regard have been
|
|
692 |
obtained in \cite{AusafDyckhoffUrban2016}.
|
|
693 |
|
|
694 |
|
|
695 |
|
|
696 |
%Brzozowski, after giving the derivatives and simplification,
|
|
697 |
%did not explore lexing with simplification, or he may well be
|
|
698 |
%stuck on an efficient simplification with proof.
|
|
699 |
%He went on to examine the use of derivatives together with
|
|
700 |
%automaton, and did not try lexing using products.
|
|
701 |
|
|
702 |
We want to get rid of the complex and fragile rectification of values.
|
|
703 |
Can we not create those intermediate values $v_1,\ldots v_n$,
|
|
704 |
and get the lexing information that should be already there while
|
|
705 |
doing derivatives in one pass, without a second injection phase?
|
|
706 |
In the meantime, can we make sure that simplifications
|
|
707 |
are easily handled without breaking the correctness of the algorithm?
|
|
708 |
|
|
709 |
Sulzmann and Lu solved this problem by
|
|
710 |
introducing additional information to the
|
|
711 |
regular expressions called \emph{bitcodes}.
|
|
712 |
|
|
713 |
|
|
714 |
|
|
715 |
|
|
716 |
|
|
717 |
|
|
718 |
|