author | Chengsong |
Sat, 06 Jul 2019 19:48:20 +0100 | |
changeset 59 | 8ff7b7508824 |
parent 58 | f0360e17080e |
child 60 | c737a0259194 |
permissions | -rw-r--r-- |
45 | 1 |
\documentclass[a4paper,UKenglish]{lipics} |
30 | 2 |
\usepackage{graphic} |
3 |
\usepackage{data} |
|
4 |
\usepackage{tikz-cd} |
|
35 | 5 |
\usepackage{algorithm} |
6 |
\usepackage{amsmath} |
|
7 |
\usepackage[noend]{algpseudocode} |
|
42 | 8 |
\usepackage{enumitem} |
30 | 9 |
% \documentclass{article} |
10 |
%\usepackage[utf8]{inputenc} |
|
11 |
%\usepackage[english]{babel} |
|
12 |
%\usepackage{listings} |
|
13 |
% \usepackage{amsthm} |
|
14 |
% \usepackage{hyperref} |
|
15 |
% \usepackage[margin=0.5in]{geometry} |
|
16 |
%\usepackage{pmboxdraw} |
|
17 |
||
18 |
\title{POSIX Regular Expression Matching and Lexing} |
|
19 |
\author{Chengsong Tan} |
|
20 |
\affil{King's College London\\ |
|
21 |
London, UK\\ |
|
22 |
\texttt{chengsong.tan@kcl.ac.uk}} |
|
23 |
\authorrunning{Chengsong Tan} |
|
24 |
\Copyright{Chengsong Tan} |
|
25 |
||
26 |
\newcommand{\dn}{\stackrel{\mbox{\scriptsize def}}{=}}% |
|
27 |
\newcommand{\ZERO}{\mbox{\bf 0}} |
|
28 |
\newcommand{\ONE}{\mbox{\bf 1}} |
|
29 |
\def\lexer{\mathit{lexer}} |
|
30 |
\def\mkeps{\mathit{mkeps}} |
|
31 |
\def\inj{\mathit{inj}} |
|
32 |
\def\Empty{\mathit{Empty}} |
|
33 |
\def\Left{\mathit{Left}} |
|
34 |
\def\Right{\mathit{Right}} |
|
35 |
\def\Stars{\mathit{Stars}} |
|
36 |
\def\Char{\mathit{Char}} |
|
37 |
\def\Seq{\mathit{Seq}} |
|
38 |
\def\Der{\mathit{Der}} |
|
39 |
\def\nullable{\mathit{nullable}} |
|
40 |
\def\Z{\mathit{Z}} |
|
41 |
\def\S{\mathit{S}} |
|
42 |
||
43 |
%\theoremstyle{theorem} |
|
44 |
%\newtheorem{theorem}{Theorem} |
|
45 |
%\theoremstyle{lemma} |
|
46 |
%\newtheorem{lemma}{Lemma} |
|
47 |
%\newcommand{\lemmaautorefname}{Lemma} |
|
48 |
%\theoremstyle{definition} |
|
49 |
%\newtheorem{definition}{Definition} |
|
35 | 50 |
\algnewcommand\algorithmicswitch{\textbf{switch}} |
51 |
\algnewcommand\algorithmiccase{\textbf{case}} |
|
52 |
\algnewcommand\algorithmicassert{\texttt{assert}} |
|
53 |
\algnewcommand\Assert[1]{\State \algorithmicassert(#1)}% |
|
54 |
% New "environments" |
|
55 |
\algdef{SE}[SWITCH]{Switch}{EndSwitch}[1]{\algorithmicswitch\ #1\ \algorithmicdo}{\algorithmicend\ \algorithmicswitch}% |
|
56 |
\algdef{SE}[CASE]{Case}{EndCase}[1]{\algorithmiccase\ #1}{\algorithmicend\ \algorithmiccase}% |
|
57 |
\algtext*{EndSwitch}% |
|
58 |
\algtext*{EndCase}% |
|
30 | 59 |
|
60 |
||
61 |
\begin{document} |
|
62 |
||
63 |
\maketitle |
|
64 |
||
65 |
\begin{abstract} |
|
66 |
Brzozowski introduced in 1964 a beautifully simple algorithm for |
|
67 |
regular expression matching based on the notion of derivatives of |
|
68 |
regular expressions. In 2014, Sulzmann and Lu extended this |
|
40 | 69 |
algorithm to not just give a YES/NO answer for whether or not a |
70 |
regular expression matches a string, but in case it matches also |
|
58 | 71 |
answers with \emph{how} it matches the string. This is important for |
40 | 72 |
applications such as lexing (tokenising a string). The problem is to |
73 |
make the algorithm by Sulzmann and Lu fast on all inputs without |
|
74 |
breaking its correctness. We have already developed some |
|
59 | 75 |
simplification rules for this, but have not yet proved that they |
40 | 76 |
preserve the correctness of the algorithm. We also have not yet |
77 |
looked at extended regular expressions, such as bounded repetitions, |
|
78 |
negation and back-references. |
|
30 | 79 |
\end{abstract} |
80 |
||
81 |
\section{Introduction} |
|
82 |
||
83 |
This PhD-project is about regular expression matching and |
|
84 |
lexing. Given the maturity of this topic, the reader might wonder: |
|
85 |
Surely, regular expressions must have already been studied to death? |
|
86 |
What could possibly be \emph{not} known in this area? And surely all |
|
87 |
implemented algorithms for regular expression matching are blindingly |
|
88 |
fast? |
|
89 |
||
90 |
Unfortunately these preconceptions are not supported by evidence: Take |
|
91 |
for example the regular expression $(a^*)^*\,b$ and ask whether |
|
92 |
strings of the form $aa..a$ match this regular |
|
93 |
expression. Obviously they do not match---the expected $b$ in the last |
|
94 |
position is missing. One would expect that modern regular expression |
|
95 |
matching engines can find this out very quickly. Alas, if one tries |
|
96 |
this example in JavaScript, Python or Java 8 with strings like 28 |
|
97 |
$a$'s, one discovers that this decision takes around 30 seconds and |
|
98 |
takes considerably longer when adding a few more $a$'s, as the graphs |
|
99 |
below show: |
|
100 |
||
101 |
\begin{center} |
|
102 |
\begin{tabular}{@{}c@{\hspace{0mm}}c@{\hspace{0mm}}c@{}} |
|
103 |
\begin{tikzpicture} |
|
104 |
\begin{axis}[ |
|
105 |
xlabel={$n$}, |
|
106 |
x label style={at={(1.05,-0.05)}}, |
|
107 |
ylabel={time in secs}, |
|
108 |
enlargelimits=false, |
|
109 |
xtick={0,5,...,30}, |
|
110 |
xmax=33, |
|
111 |
ymax=35, |
|
112 |
ytick={0,5,...,30}, |
|
113 |
scaled ticks=false, |
|
114 |
axis lines=left, |
|
115 |
width=5cm, |
|
116 |
height=4cm, |
|
117 |
legend entries={JavaScript}, |
|
118 |
legend pos=north west, |
|
119 |
legend cell align=left] |
|
120 |
\addplot[red,mark=*, mark options={fill=white}] table {re-js.data}; |
|
121 |
\end{axis} |
|
122 |
\end{tikzpicture} |
|
123 |
& |
|
124 |
\begin{tikzpicture} |
|
125 |
\begin{axis}[ |
|
126 |
xlabel={$n$}, |
|
127 |
x label style={at={(1.05,-0.05)}}, |
|
128 |
%ylabel={time in secs}, |
|
129 |
enlargelimits=false, |
|
130 |
xtick={0,5,...,30}, |
|
131 |
xmax=33, |
|
132 |
ymax=35, |
|
133 |
ytick={0,5,...,30}, |
|
134 |
scaled ticks=false, |
|
135 |
axis lines=left, |
|
136 |
width=5cm, |
|
137 |
height=4cm, |
|
138 |
legend entries={Python}, |
|
139 |
legend pos=north west, |
|
140 |
legend cell align=left] |
|
141 |
\addplot[blue,mark=*, mark options={fill=white}] table {re-python2.data}; |
|
142 |
\end{axis} |
|
143 |
\end{tikzpicture} |
|
144 |
& |
|
145 |
\begin{tikzpicture} |
|
146 |
\begin{axis}[ |
|
147 |
xlabel={$n$}, |
|
148 |
x label style={at={(1.05,-0.05)}}, |
|
149 |
%ylabel={time in secs}, |
|
150 |
enlargelimits=false, |
|
151 |
xtick={0,5,...,30}, |
|
152 |
xmax=33, |
|
153 |
ymax=35, |
|
154 |
ytick={0,5,...,30}, |
|
155 |
scaled ticks=false, |
|
156 |
axis lines=left, |
|
157 |
width=5cm, |
|
158 |
height=4cm, |
|
159 |
legend entries={Java 8}, |
|
160 |
legend pos=north west, |
|
161 |
legend cell align=left] |
|
162 |
\addplot[cyan,mark=*, mark options={fill=white}] table {re-java.data}; |
|
163 |
\end{axis} |
|
164 |
\end{tikzpicture}\\ |
|
165 |
\multicolumn{3}{c}{Graphs: Runtime for matching $(a^*)^*\,b$ with strings |
|
166 |
of the form $\underbrace{aa..a}_{n}$.} |
|
167 |
\end{tabular} |
|
168 |
\end{center} |
|
169 |
||
58 | 170 |
\noindent These are clearly abysmal and possibly surprising results. One |
171 |
would expect these systems doing much better than that---after all, |
|
172 |
given a DFA and a string, deciding whether a string is matched by this |
|
173 |
DFA should be linear. |
|
30 | 174 |
|
175 |
Admittedly, the regular expression $(a^*)^*\,b$ is carefully chosen to |
|
176 |
exhibit this ``exponential behaviour''. Unfortunately, such regular |
|
177 |
expressions are not just a few ``outliers'', but actually they are |
|
178 |
frequent enough that a separate name has been created for |
|
179 |
them---\emph{evil regular expressions}. In empiric work, Davis et al |
|
180 |
report that they have found thousands of such evil regular expressions |
|
181 |
in the JavaScript and Python ecosystems \cite{Davis18}. |
|
182 |
||
58 | 183 |
This exponential blowup in matching algorithms sometimes causes |
59 | 184 |
considerable disturbance in real life: for example on 20 July 2016 one evil |
58 | 185 |
regular expression brought the webpage |
186 |
\href{http://stackexchange.com}{Stack Exchange} to its |
|
46 | 187 |
knees.\footnote{https://stackstatus.net/post/147710624694/outage-postmortem-july-20-2016} |
30 | 188 |
In this instance, a regular expression intended to just trim white |
189 |
spaces from the beginning and the end of a line actually consumed |
|
190 |
massive amounts of CPU-resources and because of this the web servers |
|
58 | 191 |
ground to a halt. This happened when a post with 20,000 white spaces was |
192 |
submitted, but importantly the white spaces were neither at the |
|
30 | 193 |
beginning nor at the end. As a result, the regular expression matching |
58 | 194 |
engine needed to backtrack over many choices. The underlying problem is |
195 |
that many ``real life'' regular expression matching engines do not use |
|
196 |
DFAs for matching. This is because they support regular expressions that |
|
197 |
are not covered by the classical automata theory, and in this more |
|
198 |
general setting there are quite a few research questions still |
|
199 |
unanswered and fast algorithms still need to be developed (for example |
|
200 |
how to include bounded repetitions, negation and back-references). |
|
30 | 201 |
|
202 |
There is also another under-researched problem to do with regular |
|
203 |
expressions and lexing, i.e.~the process of breaking up strings into |
|
204 |
sequences of tokens according to some regular expressions. In this |
|
205 |
setting one is not just interested in whether or not a regular |
|
206 |
expression matches a string, but if it matches also in \emph{how} it |
|
207 |
matches the string. Consider for example a regular expression |
|
208 |
$r_{key}$ for recognising keywords such as \textit{if}, \textit{then} |
|
209 |
and so on; and a regular expression $r_{id}$ for recognising |
|
210 |
identifiers (say, a single character followed by characters or |
|
211 |
numbers). One can then form the compound regular expression |
|
212 |
$(r_{key} + r_{id})^*$ and use it to tokenise strings. But then how |
|
213 |
should the string \textit{iffoo} be tokenised? It could be tokenised |
|
214 |
as a keyword followed by an identifier, or the entire string as a |
|
215 |
single identifier. Similarly, how should the string \textit{if} be |
|
216 |
tokenised? Both regular expressions, $r_{key}$ and $r_{id}$, would |
|
217 |
``fire''---so is it an identifier or a keyword? While in applications |
|
218 |
there is a well-known strategy to decide these questions, called POSIX |
|
219 |
matching, only relatively recently precise definitions of what POSIX |
|
220 |
matching actually means have been formalised |
|
46 | 221 |
\cite{AusafDyckhoffUrban2016,OkuiSuzuki2010,Vansummeren2006}. |
222 |
Such a definition has also been given by Sulzmann and Lu \cite{Sulzmann2014}, but the |
|
223 |
corresponding correctness proof turned out to be faulty \cite{AusafDyckhoffUrban2016}. |
|
224 |
Roughly, POSIX matching means matching the longest initial substring. |
|
37 | 225 |
In the case of a tie, the initial submatch is chosen according to some priorities attached to the |
30 | 226 |
regular expressions (e.g.~keywords have a higher priority than |
227 |
identifiers). This sounds rather simple, but according to Grathwohl et |
|
228 |
al \cite[Page 36]{CrashCourse2014} this is not the case. They wrote: |
|
229 |
||
230 |
\begin{quote} |
|
231 |
\it{}``The POSIX strategy is more complicated than the greedy because of |
|
232 |
the dependence on information about the length of matched strings in the |
|
233 |
various subexpressions.'' |
|
234 |
\end{quote} |
|
235 |
||
236 |
\noindent |
|
237 |
This is also supported by evidence collected by Kuklewicz |
|
238 |
\cite{Kuklewicz} who noticed that a number of POSIX regular expression |
|
239 |
matchers calculate incorrect results. |
|
240 |
||
241 |
Our focus is on an algorithm introduced by Sulzmann and Lu in 2014 for |
|
242 |
regular expression matching according to the POSIX strategy |
|
243 |
\cite{Sulzmann2014}. Their algorithm is based on an older algorithm by |
|
244 |
Brzozowski from 1964 where he introduced the notion of derivatives of |
|
245 |
regular expressions \cite{Brzozowski1964}. We shall briefly explain |
|
58 | 246 |
this algorithm next. |
30 | 247 |
|
46 | 248 |
\section{The Algorithm by Brzozowski based on Derivatives of Regular |
249 |
Expressions} |
|
30 | 250 |
|
40 | 251 |
Suppose (basic) regular expressions are given by the following grammar: |
38 | 252 |
\[ r ::= \ZERO \mid \ONE |
253 |
\mid c |
|
254 |
\mid r_1 \cdot r_2 |
|
255 |
\mid r_1 + r_2 |
|
256 |
\mid r^* |
|
257 |
\] |
|
30 | 258 |
|
259 |
\noindent |
|
46 | 260 |
The intended meaning of the constructors is as follows: $\ZERO$ |
30 | 261 |
cannot match any string, $\ONE$ can match the empty string, the |
262 |
character regular expression $c$ can match the character $c$, and so |
|
40 | 263 |
on. |
264 |
||
58 | 265 |
The ingenious contribution by Brzozowski is the notion of |
30 | 266 |
\emph{derivatives} of regular expressions. The idea behind this |
267 |
notion is as follows: suppose a regular expression $r$ can match a |
|
268 |
string of the form $c\!::\! s$ (that is a list of characters starting |
|
269 |
with $c$), what does the regular expression look like that can match |
|
40 | 270 |
just $s$? Brzozowski gave a neat answer to this question. He started |
271 |
with the definition of $nullable$: |
|
36 | 272 |
\begin{center} |
273 |
\begin{tabular}{lcl} |
|
274 |
$\nullable(\ZERO)$ & $\dn$ & $\mathit{false}$ \\ |
|
275 |
$\nullable(\ONE)$ & $\dn$ & $\mathit{true}$ \\ |
|
276 |
$\nullable(c)$ & $\dn$ & $\mathit{false}$ \\ |
|
277 |
$\nullable(r_1 + r_2)$ & $\dn$ & $\nullable(r_1) \vee \nullable(r_2)$ \\ |
|
278 |
$\nullable(r_1\cdot r_2)$ & $\dn$ & $\nullable(r_1) \wedge \nullable(r_2)$ \\ |
|
279 |
$\nullable(r^*)$ & $\dn$ & $\mathit{true}$ \\ |
|
280 |
\end{tabular} |
|
281 |
\end{center} |
|
38 | 282 |
This function simply tests whether the empty string is in $L(r)$. |
36 | 283 |
He then defined |
30 | 284 |
the following operation on regular expressions, written |
285 |
$r\backslash c$ (the derivative of $r$ w.r.t.~the character $c$): |
|
286 |
||
287 |
\begin{center} |
|
288 |
\begin{tabular}{lcl} |
|
289 |
$\ZERO \backslash c$ & $\dn$ & $\ZERO$\\ |
|
290 |
$\ONE \backslash c$ & $\dn$ & $\ZERO$\\ |
|
291 |
$d \backslash c$ & $\dn$ & |
|
292 |
$\mathit{if} \;c = d\;\mathit{then}\;\ONE\;\mathit{else}\;\ZERO$\\ |
|
293 |
$(r_1 + r_2)\backslash c$ & $\dn$ & $r_1 \backslash c \,+\, r_2 \backslash c$\\ |
|
36 | 294 |
$(r_1 \cdot r_2)\backslash c$ & $\dn$ & $\mathit{if} \, nullable(r_1)$\\ |
30 | 295 |
& & $\mathit{then}\;(r_1\backslash c) \cdot r_2 \,+\, r_2\backslash c$\\ |
296 |
& & $\mathit{else}\;(r_1\backslash c) \cdot r_2$\\ |
|
297 |
$(r^*)\backslash c$ & $\dn$ & $(r\backslash c) \cdot r^*$\\ |
|
298 |
\end{tabular} |
|
299 |
\end{center} |
|
300 |
||
46 | 301 |
%Assuming the classic notion of a |
302 |
%\emph{language} of a regular expression, written $L(\_)$, t |
|
30 | 303 |
|
40 | 304 |
\noindent |
305 |
The main property of the derivative operation is that |
|
30 | 306 |
|
307 |
\begin{center} |
|
308 |
$c\!::\!s \in L(r)$ holds |
|
309 |
if and only if $s \in L(r\backslash c)$. |
|
310 |
\end{center} |
|
311 |
||
312 |
\noindent |
|
46 | 313 |
For us the main advantage is that derivatives can be |
38 | 314 |
straightforwardly implemented in any functional programming language, |
315 |
and are easily definable and reasoned about in theorem provers---the |
|
316 |
definitions just consist of inductive datatypes and simple recursive |
|
317 |
functions. Moreover, the notion of derivatives can be easily |
|
318 |
generalised to cover extended regular expression constructors such as |
|
319 |
the not-regular expression, written $\neg\,r$, or bounded repetitions |
|
320 |
(for example $r^{\{n\}}$ and $r^{\{n..m\}}$), which cannot be so |
|
321 |
straightforwardly realised within the classic automata approach. |
|
322 |
For the moment however, we focus only on the usual basic regular expressions. |
|
323 |
||
324 |
||
40 | 325 |
Now if we want to find out whether a string $s$ matches with a regular |
326 |
expression $r$, build the derivatives of $r$ w.r.t.\ (in succession) |
|
327 |
all the characters of the string $s$. Finally, test whether the |
|
328 |
resulting regular expression can match the empty string. If yes, then |
|
329 |
$r$ matches $s$, and no in the negative case. To implement this idea |
|
330 |
we can generalise the derivative operation to strings like this: |
|
46 | 331 |
|
30 | 332 |
\begin{center} |
333 |
\begin{tabular}{lcl} |
|
334 |
$r \backslash (c\!::\!s) $ & $\dn$ & $(r \backslash c) \backslash s$ \\ |
|
40 | 335 |
$r \backslash [\,] $ & $\dn$ & $r$ |
30 | 336 |
\end{tabular} |
337 |
\end{center} |
|
40 | 338 |
|
37 | 339 |
\noindent |
46 | 340 |
and then define as regular-expression matching algorithm: |
30 | 341 |
\[ |
342 |
match\;s\;r \;\dn\; nullable(r\backslash s) |
|
343 |
\] |
|
40 | 344 |
|
345 |
\noindent |
|
58 | 346 |
This algorithm can be illustrated graphically as follows |
46 | 347 |
\begin{equation}\label{graph:*} |
348 |
\begin{tikzcd} |
|
349 |
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} |
|
38 | 350 |
\end{tikzcd} |
46 | 351 |
\end{equation} |
40 | 352 |
|
353 |
\noindent |
|
46 | 354 |
where we start with a regular expression $r_0$, build successive |
355 |
derivatives until we exhaust the string and then use \textit{nullable} |
|
356 |
to test whether the result can match the empty string. It can be |
|
357 |
relatively easily shown that this matcher is correct (that is given |
|
58 | 358 |
an $s$ and a $r$, it generates YES if and only if $s \in L(r)$). |
46 | 359 |
|
360 |
||
361 |
\section{Values and the Algorithm by Sulzmann and Lu} |
|
38 | 362 |
|
30 | 363 |
One limitation, however, of Brzozowski's algorithm is that it only |
364 |
produces a YES/NO answer for whether a string is being matched by a |
|
365 |
regular expression. Sulzmann and Lu~\cite{Sulzmann2014} extended this |
|
366 |
algorithm to allow generation of an actual matching, called a |
|
46 | 367 |
\emph{value}. Values and regular expressions correspond to each |
368 |
other as illustrated in the following table: |
|
369 |
||
30 | 370 |
|
371 |
\begin{center} |
|
372 |
\begin{tabular}{c@{\hspace{20mm}}c} |
|
373 |
\begin{tabular}{@{}rrl@{}} |
|
374 |
\multicolumn{3}{@{}l}{\textbf{Regular Expressions}}\medskip\\ |
|
375 |
$r$ & $::=$ & $\ZERO$\\ |
|
376 |
& $\mid$ & $\ONE$ \\ |
|
377 |
& $\mid$ & $c$ \\ |
|
378 |
& $\mid$ & $r_1 \cdot r_2$\\ |
|
379 |
& $\mid$ & $r_1 + r_2$ \\ |
|
380 |
\\ |
|
381 |
& $\mid$ & $r^*$ \\ |
|
382 |
\end{tabular} |
|
383 |
& |
|
384 |
\begin{tabular}{@{\hspace{0mm}}rrl@{}} |
|
385 |
\multicolumn{3}{@{}l}{\textbf{Values}}\medskip\\ |
|
386 |
$v$ & $::=$ & \\ |
|
387 |
& & $\Empty$ \\ |
|
388 |
& $\mid$ & $\Char(c)$ \\ |
|
389 |
& $\mid$ & $\Seq\,v_1\, v_2$\\ |
|
390 |
& $\mid$ & $\Left(v)$ \\ |
|
391 |
& $\mid$ & $\Right(v)$ \\ |
|
392 |
& $\mid$ & $\Stars\,[v_1,\ldots\,v_n]$ \\ |
|
393 |
\end{tabular} |
|
394 |
\end{tabular} |
|
395 |
\end{center} |
|
396 |
||
397 |
\noindent |
|
58 | 398 |
There is no value corresponding to $\ZERO$; $\Empty$ corresponds to |
399 |
$\ONE$; $\Seq$ to the sequence regular expression and so on. The idea of |
|
400 |
values is to encode parse trees. To see this, suppose a \emph{flatten} |
|
46 | 401 |
operation, written $|v|$, which we can use to extract the underlying |
58 | 402 |
string of a value $v$. For example, $|\mathit{Seq} \, (\textit{Char x}) \, |
46 | 403 |
(\textit{Char y})|$ is the string $xy$. We omit the straightforward |
58 | 404 |
definition of flatten. Using flatten, we can describe how values encode |
405 |
parse trees: $\Seq\,v_1\, v_2$ encodes how the string $|v_1| @ |v_2|$ |
|
406 |
matches the regex $r_1 \cdot r_2$: $r_1$ matches the substring $|v_1|$ and, |
|
407 |
respectively, $r_2$ matches the substring $|v_2|$. Exactly how these two are matched |
|
408 |
is contained in the sub-structure of $v_1$ and $v_2$. |
|
30 | 409 |
|
59 | 410 |
To give a concrete example of how value works, consider the string $xy$ |
46 | 411 |
and the regular expression $(x + (y + xy))^*$. We can view this regular |
30 | 412 |
expression as a tree and if the string $xy$ is matched by two Star |
46 | 413 |
``iterations'', then the $x$ is matched by the left-most alternative in |
414 |
this tree and the $y$ by the right-left alternative. This suggests to |
|
415 |
record this matching as |
|
30 | 416 |
|
417 |
\begin{center} |
|
418 |
$\Stars\,[\Left\,(\Char\,x), \Right(\Left(\Char\,y))]$ |
|
419 |
\end{center} |
|
420 |
||
421 |
\noindent |
|
422 |
where $\Stars$ records how many |
|
423 |
iterations were used; and $\Left$, respectively $\Right$, which |
|
424 |
alternative is used. The value for |
|
425 |
matching $xy$ in a single ``iteration'', i.e.~the POSIX value, |
|
426 |
would look as follows |
|
427 |
||
428 |
\begin{center} |
|
429 |
$\Stars\,[\Seq\,(\Char\,x)\,(\Char\,y)]$ |
|
430 |
\end{center} |
|
431 |
||
432 |
\noindent |
|
433 |
where $\Stars$ has only a single-element list for the single iteration |
|
434 |
and $\Seq$ indicates that $xy$ is matched by a sequence regular |
|
435 |
expression. |
|
436 |
||
437 |
The contribution of Sulzmann and Lu is an extension of Brzozowski's |
|
438 |
algorithm by a second phase (the first phase being building successive |
|
46 | 439 |
derivatives---see \eqref{graph:*}). In this second phase, a POSIX value |
440 |
is generated assuming the regular expression matches the string. |
|
54 | 441 |
Pictorially, the algorithm is as follows: |
46 | 442 |
|
59 | 443 |
\begin{equation}\label{graph:2} |
30 | 444 |
\begin{tikzcd} |
36 | 445 |
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] \\ |
30 | 446 |
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] |
447 |
\end{tikzcd} |
|
59 | 448 |
\end{equation} |
37 | 449 |
|
46 | 450 |
\noindent |
59 | 451 |
For convenience, we shall employ the following notations: the regular expression we |
58 | 452 |
start with is $r_0$, and the given string $s$ is composed of characters $c_0 c_1 |
453 |
\ldots c_n$. First, we build the derivatives $r_1$, $r_2$, \ldots according to |
|
46 | 454 |
the characters $c_0$, $c_1$,\ldots until we exhaust the string and |
58 | 455 |
obtain at the derivative $r_n$. We test whether this derivative is |
46 | 456 |
$\textit{nullable}$ or not. If not, we know the string does not match |
457 |
$r$ and no value needs to be generated. If yes, we start building the |
|
458 |
parse tree incrementally by \emph{injecting} back the characters into |
|
58 | 459 |
the values $v_n, \ldots, v_0$. For this we first call the function |
46 | 460 |
$\textit{mkeps}$, which builds the parse tree for how the empty string |
58 | 461 |
has matched the (nullable) regular expression $r_n$. This function is defined |
46 | 462 |
as |
30 | 463 |
|
51
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
464 |
\begin{center} |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
465 |
\begin{tabular}{lcl} |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
466 |
$\mkeps(\ONE)$ & $\dn$ & $\Empty$ \\ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
467 |
$\mkeps(r_{1}+r_{2})$ & $\dn$ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
468 |
& \textit{if} $\nullable(r_{1})$\\ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
469 |
& & \textit{then} $\Left(\mkeps(r_{1}))$\\ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
470 |
& & \textit{else} $\Right(\mkeps(r_{2}))$\\ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
471 |
$\mkeps(r_1\cdot r_2)$ & $\dn$ & $\Seq\,(\mkeps\,r_1)\,(\mkeps\,r_2)$\\ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
472 |
$mkeps(r^*)$ & $\dn$ & $\Stars\,[]$ |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
473 |
\end{tabular} |
5df7faf69238
added mkeps and pder, still have not proof read it
Chengsong
parents:
50
diff
changeset
|
474 |
\end{center} |
41 | 475 |
|
59 | 476 |
|
477 |
\noindent There are no cases for $\ZERO$ and $c$, since |
|
478 |
these regular expression cannot match the empty string. Note |
|
479 |
also that in case of alternatives we give preference to the |
|
480 |
regular expression on the left-hand side. This will become |
|
481 |
important later on. |
|
482 |
||
483 |
After this, we inject back the characters one by one in order to build |
|
46 | 484 |
the parse tree $v_i$ for how the regex $r_i$ matches the string |
55 | 485 |
$s_i$ ($s_i = c_i \ldots c_n$ ) from the previous parse tree $v_{i+1}$. After injecting back $n$ characters, we |
486 |
get the parse tree for how $r_0$ matches $s$, exactly as we wanted. |
|
59 | 487 |
|
488 |
We need a function that |
|
489 |
reverses the "chopping off" for values which we did in the |
|
490 |
first phase for derivatives. The corresponding function is |
|
491 |
called $\textit{inj}$ (short for injection). This function takes three |
|
492 |
arguments: the first one is a regular expression $\textit{r_{i-1}}$ |
|
493 |
before the character is chopped off, the second is $\textit{c_{i-1}}$, |
|
494 |
the character |
|
495 |
we |
|
496 |
want to inject and the third argument is the value $textit{v_i}$, where we |
|
497 |
will inject the character(it corresponds to the regular expression |
|
498 |
after the character |
|
499 |
has been chopped off). The result of this function is a |
|
500 |
new value. The definition of $\textit{inj}$ is as follows: |
|
501 |
||
502 |
\begin{center} |
|
503 |
\begin{tabular}{l@{\hspace{1mm}}c@{\hspace{1mm}}l} |
|
504 |
$\textit{inj}\,(c)\,c\,Empty$ & $\dn$ & $Char\,c$\\ |
|
505 |
$\textit{inj}\,(r_1 + r_2)\,c\,\Left(v)$ & $\dn$ & $\Left(\textit{inj}\,r_1\,c\,v)$\\ |
|
506 |
$\textit{inj}\,(r_1 + r_2)\,c\,Right(v)$ & $\dn$ & $Right(\textit{inj}\,r_2\,c\,v)$\\ |
|
507 |
$\textit{inj}\,(r_1 \cdot r_2)\,c\,Seq(v_1,v_2)$ & $\dn$ & $Seq(\textit{inj}\,r_1\,c\,v_1,v_2)$\\ |
|
508 |
$\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)$\\ |
|
509 |
$\textit{inj}\,(r_1 \cdot r_2)\,c\,Right(v)$ & $\dn$ & $Seq(\textit{mkeps}(r_1),\textit{inj}\,r_2\,c\,v)$\\ |
|
510 |
$\textit{inj}\,(r^*)\,c\,Seq(v,Stars\,vs)$ & $\dn$ & $Stars((\textit{inj}\,r\,c\,v)\,::\,vs)$\\ |
|
511 |
\end{tabular} |
|
512 |
\end{center} |
|
513 |
||
514 |
\noindent This definition is by recursion on the regular |
|
515 |
expressions' and values' shapes. |
|
516 |
To understands this definition in a more vivid way, the reader |
|
517 |
might imagine this: when we do the derivative on regular expression |
|
518 |
$r_{i-1}$, we chop off a character from $r_{i-1}$ to form $r_i$. |
|
519 |
This leaves a "hole" on $r_i$. In order to calculate a value for |
|
520 |
$r_{i-1}$, we need to locate where that hole is. We can find this location |
|
521 |
informtaion by comparing $r_{i-1}$ and $v_i$. |
|
522 |
For instance, if $r_{i-1}$ is of shape $r_a \cdot r_b$, and $v_i$ |
|
523 |
is of shape $\textit{Left(Seq(v_a, v_b))}$, we know immediately that |
|
524 |
\[ (r_a \cdot r_b)\backslash c = (r_a\backslash c) \cdot r_b \,+\, r_b\backslash c\], |
|
525 |
otherwise if $r_a$ is not nullable, |
|
526 |
\[ (r_a \cdot r_b)\backslash c = (r_a\backslash c) \cdot r_b\], |
|
527 |
the value $v_i$ should be of shape $Seq(\ldots)$, contradicting the fact that |
|
528 |
$v_i$ is actually $Left(\ldots)$. Furthermore, since $v_i$ is of shape |
|
529 |
$Left(\ldots)$ instead of $Right(\ldots)$, we know that the left |
|
530 |
branch is taken instead of the right one. We have therefore found out |
|
531 |
that the hole will be on $r_a$. So we recursively call $\textit{inj} |
|
532 |
r_a c v_a$ to fill that hole in $v_a$. After injection, the value |
|
533 |
$v_i$ for $r_i = r_a \cdot \rb$ should be $\textit{Seq}(\textit{inj} |
|
534 |
r_a c v_a, v_b)$. |
|
535 |
||
536 |
To understand what is going on, it might be best to do some |
|
537 |
example calculations and compare them with Figure~\eqref{graph:2}. |
|
538 |
||
539 |
||
540 |
||
55 | 541 |
A correctness proof using induction can be routinely established. |
41 | 542 |
|
58 | 543 |
It is instructive to see how this algorithm works by a little example. |
544 |
Suppose we have a regular expression $(a+b+ab+c+abc)^*$ and we want to |
|
545 |
match it against the string $abc$(when $abc$ is written as a regular |
|
546 |
expression, the most standard way of expressing it should be $a \cdot (b |
|
547 |
\cdot c)$. We omit the parenthesis and dots here for readability). By |
|
548 |
POSIX rules the lexer should go for the longest matching, i.e. it should |
|
549 |
match the string $abc$ in one star iteration, using the longest string |
|
550 |
$abc$ in the sub-expression $a+b+ab+c+abc$(we use $r$ to denote this |
|
551 |
sub-expression for conciseness). Here is how the lexer achieves a parse |
|
552 |
tree for this matching. First, we build successive derivatives until we |
|
553 |
exhaust the string, as illustrated here( we simplified some regular |
|
554 |
expressions like $0 \cdot b$ to $0$ for conciseness. Similarly, we allow |
|
555 |
$\textit{ALT}$ to take a list of regular expressions as an argument |
|
556 |
instead of just 2 operands to reduce the nested depth of |
|
557 |
$\textit{ALT}$): |
|
41 | 558 |
|
559 |
\[ r^* \xrightarrow{\backslash a} r_1 = (1+0+1 \cdot b + 0 + 1 \cdot b \cdot c) \cdot r* \xrightarrow{\backslash b}\] |
|
560 |
\[r_2 = (0+0+1 \cdot 1 + 0 + 1 \cdot 1 \cdot c) \cdot r^* +(0+1+0 + 0 + 0) \cdot r* \xrightarrow{\backslash c}\] |
|
561 |
\[r_3 = ((0+0+0 + 0 + 1 \cdot 1 \cdot 1) \cdot r^* + (0+0+0 + 1 + 0) \cdot r*) +((0+1+0 + 0 + 0) \cdot r*+(0+0+0 + 1 + 0) \cdot r* ) |
|
562 |
\] |
|
563 |
||
564 |
Now instead of using $nullable$ to give a $yes$, we call $mkeps$ to construct a parse tree for how $r_3$ matched the string $abc$. $mkeps$ gives the following value $v_3$: \\$Left(Left(Seq(Right(Right(Right(Seq(Empty, Seq(Empty, Empty)))))), Stars []))$\\ |
|
56 | 565 |
This corresponds to the leftmost term |
566 |
$((0+0+0 + 0 + 1 \cdot 1 \cdot 1) \cdot r^* $\\ |
|
567 |
in $r_3$. Note that its leftmost location allows $mkeps$ to choose it as the first candidate that meets the requirement of being $nullable$. This location is naturally generated by the splitting clause\\ $(r_1 \cdot r_2)\backslash c (when \; r_1 \; nullable)) \, = (r_1\backslash c) \cdot r_2 \,+\, r_2\backslash c. \\$ By this clause, we put |
|
568 |
$r_1 \backslash c \cdot r_2 $ at the $\textit{front}$ and $r_2 \backslash c$ at the $\textit{back}$. This allows $mkeps$ to always pick up among two matches the one with a longer prefix. The value \\ |
|
41 | 569 |
$Left(Left(Seq(Right(Right(Right(Seq(Empty, Seq(Empty, Empty)))))), Stars []))$\\ |
56 | 570 |
tells us how about the empty string matches the final regular expression after doing all the derivatives: among the regular expressions in \\$(0+0+0 + 0 + 1 \cdot 1 \cdot 1) \cdot r^* + (0+0+0 + 1 + 0) \cdot r*) +((0+1+0 + 0 + 0) \cdot r*+(0+0+0 + 1 + 0) \cdot r* )$, \\ |
57 | 571 |
we choose the left most nullable one, which is composed of a sequence of an alternative and a star. In that alternative $0+0+0 + 0 + 1 \cdot 1 \cdot 1$ we take the rightmost choice. |
41 | 572 |
|
57 | 573 |
Using the value $v_3$, the character c, and the regular expression $r_2$, we can recover how $r_2$ matched the string $[c]$ : we inject $c$ back to $v_3$, and get \\ $v_2 = Left(Seq(Right(Right(Right(Seq(Empty, Seq(Empty, c)))))), Stars [])$, \\ |
574 |
which tells us how $r_2$ matched $c$. After this we inject back the character $b$, and get\\ $v_1 = Seq(Right(Right(Right(Seq(Empty, Seq(b, c)))))), Stars [])$ for how $r_1= (1+0+1 \cdot b + 0 + 1 \cdot b \cdot c) \cdot r*$ matched the string $bc$ before it split into 2 pieces. Finally, after injecting character a back to $v_1$, we get the parse tree $v_0= Stars [Right(Right(Right(Seq(a, Seq(b, c)))))]$ for how r matched $abc$. |
|
42 | 575 |
We omit the details of injection function, which is provided by Sulzmann and Lu's paper \cite{Sulzmann2014}. |
576 |
Readers might have noticed that the parse tree information as actually already available when doing derivatives. For example, immediately after the operation $\backslash a$ we know that if we want to match a string that starts with a, we can either take the initial match to be |
|
577 |
\begin{enumerate} |
|
578 |
\item[1)] just $a$ or |
|
579 |
\item[2)] string $ab$ or |
|
580 |
\item[3)] string $abc$. |
|
581 |
\end{enumerate} |
|
45 | 582 |
In order to differentiate between these choices, we just need to remember their positions--$a$ is on the left, $ab$ is in the middle , and $abc$ is on the right. Which one of these alternatives is chosen later does not affect their relative position because our algorithm does not change this order. There is no need to traverse this information twice. This leads to a new approach of lexing-- if we store the information for parse trees in the corresponding regular expression pieces, update this information when we do derivative operation on them, and collect the information when finished with derivatives and calling $mkeps$ for deciding which branch is POSIX, we can generate the parse tree in one pass, instead of doing an n-step backward transformation.This leads to Sulzmann and Lu's novel idea of using bit-codes on derivatives. |
42 | 583 |
|
584 |
In the next section, we shall focus on the bit-coded algorithm and the natural |
|
585 |
process of simplification of regular expressions using bit-codes, which is needed in |
|
30 | 586 |
order to obtain \emph{fast} versions of the Brzozowski's, and Sulzmann |
587 |
and Lu's algorithms. This is where the PhD-project hopes to advance |
|
588 |
the state-of-the-art. |
|
589 |
||
590 |
||
591 |
\section{Simplification of Regular Expressions} |
|
42 | 592 |
Using bit-codes to guide parsing is not a new idea. |
45 | 593 |
It was applied to context free grammars and then adapted by Henglein and Nielson for efficient regular expression parsing \cite{nielson11bcre}. Sulzmann and Lu took a step further by integrating bitcodes into derivatives. |
30 | 594 |
|
43 | 595 |
The argument for complicating the data structures from basic regular expressions to those with bitcodes |
596 |
is that we can introduce simplification without making the algorithm crash or impossible to reason about. |
|
597 |
The reason why we need simplification is due to the shortcoming of a naive algorithm using Brzozowski's definition only. |
|
30 | 598 |
The main drawback of building successive derivatives according to |
599 |
Brzozowski's definition is that they can grow very quickly in size. |
|
600 |
This is mainly due to the fact that the derivative operation generates |
|
601 |
often ``useless'' $\ZERO$s and $\ONE$s in derivatives. As a result, |
|
602 |
if implemented naively both algorithms by Brzozowski and by Sulzmann |
|
603 |
and Lu are excruciatingly slow. For example when starting with the |
|
604 |
regular expression $(a + aa)^*$ and building 12 successive derivatives |
|
605 |
w.r.t.~the character $a$, one obtains a derivative regular expression |
|
606 |
with more than 8000 nodes (when viewed as a tree). Operations like |
|
607 |
derivative and $\nullable$ need to traverse such trees and |
|
608 |
consequently the bigger the size of the derivative the slower the |
|
609 |
algorithm. Fortunately, one can simplify regular expressions after |
|
610 |
each derivative step. Various simplifications of regular expressions |
|
611 |
are possible, such as the simplifications of $\ZERO + r$, |
|
612 |
$r + \ZERO$, $\ONE\cdot r$, $r \cdot \ONE$, and $r + r$ to just |
|
613 |
$r$. These simplifications do not affect the answer for whether a |
|
614 |
regular expression matches a string or not, but fortunately also do |
|
615 |
not affect the POSIX strategy of how regular expressions match |
|
616 |
strings---although the latter is much harder to establish. Some |
|
617 |
initial results in this regard have been obtained in |
|
618 |
\cite{AusafDyckhoffUrban2016}. However, what has not been achieved yet |
|
619 |
is a very tight bound for the size. Such a tight bound is suggested by |
|
620 |
work of Antimirov who proved that (partial) derivatives can be bound |
|
621 |
by the number of characters contained in the initial regular |
|
35 | 622 |
expression \cite{Antimirov95}. |
623 |
||
624 |
Antimirov defined the "partial derivatives" of regular expressions to be this: |
|
625 |
%TODO definition of partial derivatives |
|
52
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
626 |
\begin{center} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
627 |
\begin{tabular}{lcl} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
628 |
$\textit{pder} \; c \; 0$ & $\dn$ & $\emptyset$\\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
629 |
$\textit{pder} \; c \; 1$ & $\dn$ & $\emptyset$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
630 |
$\textit{pder} \; c \; d$ & $\dn$ & $\textit{if} \; c \,=\, d \; \{ 1 \} \; \textit{else} \; \emptyset$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
631 |
$\textit{pder} \; c \; r_1+r_2$ & $\dn$ & $pder \; c \; r_1 \cup pder \; c \; r_2$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
632 |
$\textit{pder} \; c \; r_1 \cdot r_2$ & $\dn$ & $\textit{if} \; nullable \; r_1 \; \{ r \cdot r_2 \mid r \in pder \; c \; r_1 \} \cup pder \; c \; r_2 \; \textit{else} \; \{ r \cdot r_2 \mid r \in pder \; c \; r_1 \} $ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
633 |
$\textit{pder} \; c \; r^*$ & $\dn$ & $ \{ r' \cdot r^* \mid r' \in pder \; c \; r \} $ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
634 |
\end{tabular} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
635 |
\end{center} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
636 |
|
35 | 637 |
|
638 |
it is essentially a set of regular expressions that come from the sub-structure of the original regular expression. |
|
639 |
Antimirov has proved a nice size bound of the size of partial derivatives. Roughly speaking the size will not exceed the fourth power of the number of nodes in that regular expression. Interestingly, we observed from experiment that after the simplification step, our regular expression has the same size or is smaller than the partial derivatives. This allows us to prove a tight bound on the size of regular expression during the running time of the algorithm if we can establish the connection between our simplification rules and partial derivatives. |
|
640 |
||
641 |
%We believe, and have generated test |
|
642 |
%data, that a similar bound can be obtained for the derivatives in |
|
643 |
%Sulzmann and Lu's algorithm. Let us give some details about this next. |
|
30 | 644 |
|
43 | 645 |
Bit-codes look like this: |
646 |
\[ b ::= S \mid Z \; \;\; |
|
647 |
bs ::= [] \mid b:bs |
|
648 |
\] |
|
649 |
They are just a string of bits, the names "S" and "Z" here are kind of arbitrary, we can use 0 and 1 or binary symbol to substitute them. They are a compact form of parse trees. |
|
650 |
Here is how values and bit-codes are related: |
|
651 |
Bitcodes are essentially incomplete values. |
|
30 | 652 |
This can be straightforwardly seen in the following transformation: |
653 |
\begin{center} |
|
654 |
\begin{tabular}{lcl} |
|
655 |
$\textit{code}(\Empty)$ & $\dn$ & $[]$\\ |
|
656 |
$\textit{code}(\Char\,c)$ & $\dn$ & $[]$\\ |
|
657 |
$\textit{code}(\Left\,v)$ & $\dn$ & $\Z :: code(v)$\\ |
|
658 |
$\textit{code}(\Right\,v)$ & $\dn$ & $\S :: code(v)$\\ |
|
659 |
$\textit{code}(\Seq\,v_1\,v_2)$ & $\dn$ & $code(v_1) \,@\, code(v_2)$\\ |
|
660 |
$\textit{code}(\Stars\,[])$ & $\dn$ & $[\S]$\\ |
|
661 |
$\textit{code}(\Stars\,(v\!::\!vs))$ & $\dn$ & $\Z :: code(v) \;@\; |
|
662 |
code(\Stars\,vs)$ |
|
663 |
\end{tabular} |
|
664 |
\end{center} |
|
665 |
where $\Z$ and $\S$ are arbitrary names for the bits in the |
|
666 |
bitsequences. |
|
667 |
Here code encodes a value into a bitsequence by converting Left into $\Z$, Right into $\S$, the start point of a non-empty star iteration into $\S$, and the border where a local star terminates into $\Z$. This conversion is apparently lossy, as it throws away the character information, and does not decode the boundary between the two operands of the sequence constructor. Moreover, with only the bitcode we cannot even tell whether the $\S$s and $\Z$s are for $Left/Right$ or $Stars$. The reason for choosing this compact way of storing information is that the relatively small size of bits can be easily moved around during the lexing process. In order to recover the bitcode back into values, we will need the regular expression as the extra information and decode them back into value:\\ |
|
37 | 668 |
%\begin{definition}[Bitdecoding of Values]\mbox{} |
36 | 669 |
\begin{center} |
670 |
\begin{tabular}{@{}l@{\hspace{1mm}}c@{\hspace{1mm}}l@{}} |
|
671 |
$\textit{decode}'\,bs\,(\ONE)$ & $\dn$ & $(\Empty, bs)$\\ |
|
672 |
$\textit{decode}'\,bs\,(c)$ & $\dn$ & $(\Char\,c, bs)$\\ |
|
673 |
$\textit{decode}'\,(\Z\!::\!bs)\;(r_1 + r_2)$ & $\dn$ & |
|
674 |
$\textit{let}\,(v, bs_1) = \textit{decode}'\,bs\,r_1\;\textit{in}\; |
|
675 |
(\Left\,v, bs_1)$\\ |
|
676 |
$\textit{decode}'\,(\S\!::\!bs)\;(r_1 + r_2)$ & $\dn$ & |
|
677 |
$\textit{let}\,(v, bs_1) = \textit{decode}'\,bs\,r_2\;\textit{in}\; |
|
678 |
(\Right\,v, bs_1)$\\ |
|
679 |
$\textit{decode}'\,bs\;(r_1\cdot r_2)$ & $\dn$ & |
|
680 |
$\textit{let}\,(v_1, bs_1) = \textit{decode}'\,bs\,r_1\;\textit{in}$\\ |
|
681 |
& & $\textit{let}\,(v_2, bs_2) = \textit{decode}'\,bs_1\,r_2$\\ |
|
682 |
& & \hspace{35mm}$\textit{in}\;(\Seq\,v_1\,v_2, bs_2)$\\ |
|
683 |
$\textit{decode}'\,(\Z\!::\!bs)\,(r^*)$ & $\dn$ & $(\Stars\,[], bs)$\\ |
|
684 |
$\textit{decode}'\,(\S\!::\!bs)\,(r^*)$ & $\dn$ & |
|
685 |
$\textit{let}\,(v, bs_1) = \textit{decode}'\,bs\,r\;\textit{in}$\\ |
|
686 |
& & $\textit{let}\,(\Stars\,vs, bs_2) = \textit{decode}'\,bs_1\,r^*$\\ |
|
687 |
& & \hspace{35mm}$\textit{in}\;(\Stars\,v\!::\!vs, bs_2)$\bigskip\\ |
|
688 |
||
689 |
$\textit{decode}\,bs\,r$ & $\dn$ & |
|
690 |
$\textit{let}\,(v, bs') = \textit{decode}'\,bs\,r\;\textit{in}$\\ |
|
691 |
& & $\textit{if}\;bs' = []\;\textit{then}\;\textit{Some}\,v\; |
|
692 |
\textit{else}\;\textit{None}$ |
|
693 |
\end{tabular} |
|
694 |
\end{center} |
|
37 | 695 |
%\end{definition} |
30 | 696 |
|
43 | 697 |
|
53 | 698 |
Sulzmann and Lu's integrated the bitcodes into annotated regular expressions by attaching them to the head of every substructure of a regular expression\cite{Sulzmann2014}. They are |
43 | 699 |
defined by the following grammar: |
700 |
||
701 |
\begin{center} |
|
702 |
\begin{tabular}{lcl} |
|
703 |
$\textit{a}$ & $::=$ & $\textit{ZERO}$\\ |
|
704 |
& $\mid$ & $\textit{ONE}\;\;bs$\\ |
|
705 |
& $\mid$ & $\textit{CHAR}\;\;bs\,c$\\ |
|
706 |
& $\mid$ & $\textit{ALTS}\;\;bs\,as$\\ |
|
707 |
& $\mid$ & $\textit{SEQ}\;\;bs\,a_1\,a_2$\\ |
|
708 |
& $\mid$ & $\textit{STAR}\;\;bs\,a$ |
|
709 |
\end{tabular} |
|
710 |
\end{center} |
|
711 |
||
712 |
\noindent |
|
713 |
where $bs$ stands for bitsequences, and $as$ (in \textit{ALTS}) for a |
|
714 |
list of annotated regular expressions. These bitsequences encode |
|
715 |
information about the (POSIX) value that should be generated by the |
|
716 |
Sulzmann and Lu algorithm. |
|
717 |
||
30 | 718 |
To do lexing using annotated regular expressions, we shall first transform the |
719 |
usual (un-annotated) regular expressions into annotated regular |
|
720 |
expressions:\\ |
|
37 | 721 |
%\begin{definition} |
36 | 722 |
\begin{center} |
723 |
\begin{tabular}{lcl} |
|
724 |
$(\ZERO)^\uparrow$ & $\dn$ & $\textit{ZERO}$\\ |
|
725 |
$(\ONE)^\uparrow$ & $\dn$ & $\textit{ONE}\,[]$\\ |
|
726 |
$(c)^\uparrow$ & $\dn$ & $\textit{CHAR}\,[]\,c$\\ |
|
727 |
$(r_1 + r_2)^\uparrow$ & $\dn$ & |
|
728 |
$\textit{ALT}\;[]\,(\textit{fuse}\,[\Z]\,r_1^\uparrow)\, |
|
729 |
(\textit{fuse}\,[\S]\,r_2^\uparrow)$\\ |
|
730 |
$(r_1\cdot r_2)^\uparrow$ & $\dn$ & |
|
731 |
$\textit{SEQ}\;[]\,r_1^\uparrow\,r_2^\uparrow$\\ |
|
732 |
$(r^*)^\uparrow$ & $\dn$ & |
|
733 |
$\textit{STAR}\;[]\,r^\uparrow$\\ |
|
734 |
\end{tabular} |
|
735 |
\end{center} |
|
37 | 736 |
%\end{definition} |
44
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
737 |
|
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
738 |
Here $fuse$ is an auxiliary function that helps to attach bits to the front of an annotated regular expression. Its definition goes as follows: |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
739 |
\begin{center} |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
740 |
\begin{tabular}{lcl} |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
741 |
$\textit{fuse}\,bs\,(\textit{ZERO})$ & $\dn$ & $\textit{ZERO}$\\ |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
742 |
$\textit{fuse}\,bs\,(\textit{ONE}\,bs')$ & $\dn$ & |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
743 |
$\textit{ONE}\,(bs\,@\,bs')$\\ |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
744 |
$\textit{fuse}\,bs\,(\textit{CHAR}\,bs'\,c)$ & $\dn$ & |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
745 |
$\textit{CHAR}\,(bs\,@\,bs')\,c$\\ |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
746 |
$\textit{fuse}\,bs\,(\textit{ALT}\,bs'\,a_1\,a_2)$ & $\dn$ & |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
747 |
$\textit{ALT}\,(bs\,@\,bs')\,a_1\,a_2$\\ |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
748 |
$\textit{fuse}\,bs\,(\textit{SEQ}\,bs'\,a_1\,a_2)$ & $\dn$ & |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
749 |
$\textit{SEQ}\,(bs\,@\,bs')\,a_1\,a_2$\\ |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
750 |
$\textit{fuse}\,bs\,(\textit{STAR}\,bs'\,a)$ & $\dn$ & |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
751 |
$\textit{STAR}\,(bs\,@\,bs')\,a$ |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
752 |
\end{tabular} |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
753 |
\end{center} |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
754 |
|
43 | 755 |
After internalise we do successive derivative operations on the annotated regular expression. |
44
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
756 |
This derivative operation is the same as what we previously have for the simple regular expressions, except that we take special care of the bits :\\ |
37 | 757 |
%\begin{definition}{bder} |
36 | 758 |
\begin{center} |
759 |
\begin{tabular}{@{}lcl@{}} |
|
760 |
$(\textit{ZERO})\backslash c$ & $\dn$ & $\textit{ZERO}$\\ |
|
761 |
$(\textit{ONE}\;bs)\backslash c$ & $\dn$ & $\textit{ZERO}$\\ |
|
762 |
$(\textit{CHAR}\;bs\,d)\backslash c$ & $\dn$ & |
|
763 |
$\textit{if}\;c=d\; \;\textit{then}\; |
|
764 |
\textit{ONE}\;bs\;\textit{else}\;\textit{ZERO}$\\ |
|
765 |
$(\textit{ALT}\;bs\,a_1\,a_2)\backslash c$ & $\dn$ & |
|
766 |
$\textit{ALT}\,bs\,(a_1\backslash c)\,(a_2\backslash c)$\\ |
|
767 |
$(\textit{SEQ}\;bs\,a_1\,a_2)\backslash c$ & $\dn$ & |
|
768 |
$\textit{if}\;\textit{bnullable}\,a_1$\\ |
|
769 |
& &$\textit{then}\;\textit{ALT}\,bs\,(\textit{SEQ}\,[]\,(a_1\backslash c)\,a_2)$\\ |
|
770 |
& &$\phantom{\textit{then}\;\textit{ALT}\,bs\,}(\textit{fuse}\,(\textit{bmkeps}\,a_1)\,(a_2\backslash c))$\\ |
|
771 |
& &$\textit{else}\;\textit{SEQ}\,bs\,(a_1\backslash c)\,a_2$\\ |
|
772 |
$(\textit{STAR}\,bs\,a)\backslash c$ & $\dn$ & |
|
773 |
$\textit{SEQ}\;bs\,(\textit{fuse}\, [\Z] (r\backslash c))\, |
|
774 |
(\textit{STAR}\,[]\,r)$ |
|
775 |
\end{tabular} |
|
776 |
\end{center} |
|
37 | 777 |
%\end{definition} |
44
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
778 |
For instance, when we unfold $STAR \; bs \; a$ into a sequence, we attach an additional bit Z to the front of $r \backslash c$ to indicate that there is one more star iteration. |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
779 |
The other example, the $SEQ$ clause is more subtle-- when $a_1$ is $bnullable$(here bnullable is exactly the same as nullable, except that it is for annotated regular expressions, therefore we omit the definition). |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
780 |
Assume that $bmkeps$ correctly extracts the bitcode for how $a_1$ matches the string prior to character c(more on this later), then the right branch of $ALTS$, which is $fuse \; bmkeps \; a_1 (a_2 \backslash c)$ will collapse the regular expression $a_1$(as it has already been fully matched) and store the parsing information at the head of the regular expression $a_2 \backslash c$ by fusing to it. The bitsequence $bs$, which was initially attached to the head of $SEQ$, has now been elevated to the top-level of ALT, |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
781 |
as this information will be needed whichever way the $SEQ$ is matched--no matter whether c belongs to $a_1$ or $ a_2$. |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
782 |
After carefully doing these derivatives and maintaining all the parsing information, we complete the parsing by collecting the bits using a special $mkeps$ function for annotated regular expressions--$bmkeps$: |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
783 |
|
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
784 |
|
37 | 785 |
%\begin{definition}[\textit{bmkeps}]\mbox{} |
36 | 786 |
\begin{center} |
787 |
\begin{tabular}{lcl} |
|
788 |
$\textit{bmkeps}\,(\textit{ONE}\,bs)$ & $\dn$ & $bs$\\ |
|
789 |
$\textit{bmkeps}\,(\textit{ALT}\,bs\,a_1\,a_2)$ & $\dn$ & |
|
790 |
$\textit{if}\;\textit{bnullable}\,a_1$\\ |
|
791 |
& &$\textit{then}\;bs\,@\,\textit{bmkeps}\,a_1$\\ |
|
792 |
& &$\textit{else}\;bs\,@\,\textit{bmkeps}\,a_2$\\ |
|
793 |
$\textit{bmkeps}\,(\textit{SEQ}\,bs\,a_1\,a_2)$ & $\dn$ & |
|
794 |
$bs \,@\,\textit{bmkeps}\,a_1\,@\, \textit{bmkeps}\,a_2$\\ |
|
795 |
$\textit{bmkeps}\,(\textit{STAR}\,bs\,a)$ & $\dn$ & |
|
796 |
$bs \,@\, [\S]$ |
|
797 |
\end{tabular} |
|
798 |
\end{center} |
|
37 | 799 |
%\end{definition} |
44
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
800 |
This function completes the parse tree information by |
45 | 801 |
travelling along the path on the regular expression that corresponds to a POSIX value snd collect all the bits, and |
44
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
802 |
using S to indicate the end of star iterations. If we take the bitsproduced by $bmkeps$ and decode it, |
4d674a971852
another changes. have written more. but havent typed them. tomorrow will continue.
Chengsong
parents:
43
diff
changeset
|
803 |
we get the parse tree we need, the working flow looks like this:\\ |
37 | 804 |
\begin{center} |
805 |
\begin{tabular}{lcl} |
|
806 |
$\textit{blexer}\;r\,s$ & $\dn$ & |
|
807 |
$\textit{let}\;a = (r^\uparrow)\backslash s\;\textit{in}$\\ |
|
808 |
& & $\;\;\textit{if}\; \textit{bnullable}(a)$\\ |
|
809 |
& & $\;\;\textit{then}\;\textit{decode}\,(\textit{bmkeps}\,a)\,r$\\ |
|
810 |
& & $\;\;\textit{else}\;\textit{None}$ |
|
811 |
\end{tabular} |
|
812 |
\end{center} |
|
53 | 813 |
Here $(r^\uparrow)\backslash s$ is similar to what we have previously defined for |
814 |
$r\backslash s$. |
|
30 | 815 |
|
816 |
The main point of the bitsequences and annotated regular expressions |
|
817 |
is that we can apply rather aggressive (in terms of size) |
|
818 |
simplification rules in order to keep derivatives small. |
|
819 |
||
820 |
We have |
|
821 |
developed such ``aggressive'' simplification rules and generated test |
|
822 |
data that show that the expected bound can be achieved. Obviously we |
|
823 |
could only partially cover the search space as there are infinitely |
|
824 |
many regular expressions and strings. One modification we introduced |
|
825 |
is to allow a list of annotated regular expressions in the |
|
826 |
\textit{ALTS} constructor. This allows us to not just delete |
|
827 |
unnecessary $\ZERO$s and $\ONE$s from regular expressions, but also |
|
828 |
unnecessary ``copies'' of regular expressions (very similar to |
|
829 |
simplifying $r + r$ to just $r$, but in a more general |
|
35 | 830 |
setting). |
49 | 831 |
Another modification is that we use simplification rules |
832 |
inspired by Antimirov's work on partial derivatives. They maintain the |
|
833 |
idea that only the first ``copy'' of a regular expression in an |
|
834 |
alternative contributes to the calculation of a POSIX value. All |
|
835 |
subsequent copies can be pruned from the regular expression. |
|
836 |
||
52
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
837 |
A recursive definition of simplification function that looks similar to scala code is given below:\\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
838 |
\begin{center} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
839 |
\begin{tabular}{@{}lcl@{}} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
840 |
$\textit{simp} \; a$ & $\dn$ & $\textit{a} \; \textit{if} \; a = (\textit{ONE} \; bs) \; or\; (\textit{CHAR} \, bs \; c) \; or\; (\textit{STAR}\; bs\; a_1)$\\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
841 |
|
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
842 |
$\textit{simp} \; \textit{SEQ}\;bs\,a_1\,a_2$ & $\dn$ & $ (\textit{simp} \; a_1, \textit{simp} \; a_2) \; \textit{match} $ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
843 |
&&$\textit{case} \; (0, \_) \Rightarrow 0$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
844 |
&&$ \textit{case} \; (\_, 0) \Rightarrow 0$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
845 |
&&$ \textit{case} \; (1, a_2') \Rightarrow \textit{fuse} \; bs \; a_2'$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
846 |
&&$ \textit{case} \; (a_1', 1) \Rightarrow \textit{fuse} \; bs \; a_1'$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
847 |
&&$ \textit{case} \; (a_1', a_2') \Rightarrow \textit{SEQ} \; bs \; a_1' \; a_2'$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
848 |
|
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
849 |
$\textit{simp} \; \textit{ALT}\;bs\,as$ & $\dn$ & $\textit{ distinct}( \textit{flatten} ( \textit{map simp as})) \; \textit{match} $ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
850 |
&&$\textit{case} \; [] \Rightarrow 0$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
851 |
&&$ \textit{case} \; a :: [] \Rightarrow \textit{fuse bs a}$ \\ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
852 |
&&$ \textit{case} \; as' \Rightarrow \textit{ALT bs as'}$ |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
853 |
\end{tabular} |
25bbbb8b0e90
just in case of some accidents from erasing my work
Chengsong
parents:
51
diff
changeset
|
854 |
\end{center} |
47 | 855 |
|
856 |
The simplification does a pattern matching on the regular expression. When it detected that |
|
857 |
the regular expression is an alternative or sequence, it will try to simplify its children regular expressions |
|
858 |
recursively and then see if one of the children turn into 0 or 1, which might trigger further simplification |
|
859 |
at the current level. The most involved part is the ALTS clause, where we use two auxiliary functions |
|
860 |
flatten and distinct to open up nested ALT and reduce as many duplicates as possible. |
|
48 | 861 |
Function distinct keeps the first occurring copy only and remove all later ones when detected duplicates. |
862 |
Function flatten opens up nested ALT. Its recursive definition is given below: |
|
53 | 863 |
\begin{center} |
864 |
\begin{tabular}{@{}lcl@{}} |
|
865 |
$\textit{flatten} \; (\textit{ALT}\;bs\,as) :: as'$ & $\dn$ & $(\textit{ map fuse}( \textit{bs, \_} ) \textit{ as}) \; +\!+ \; \textit{flatten} \; as' $ \\ |
|
866 |
$\textit{flatten} \; \textit{ZERO} :: as'$ & $\dn$ & $ \textit{flatten} \; as' $ \\ |
|
867 |
$\textit{flatten} \; a :: as'$ & $\dn$ & $a :: \textit{flatten} \; as' $ |
|
868 |
\end{tabular} |
|
869 |
\end{center} |
|
870 |
||
48 | 871 |
Here flatten behaves like the traditional functional programming flatten function, |
872 |
what it does is basically removing parentheses like changing $a+(b+c)$ into $a+b+c$. |
|
47 | 873 |
|
53 | 874 |
Suppose we apply simplification after each derivative step, |
875 |
and view these two operations as an atomic one: $a \backslash_{simp} c \dn \textit{simp}(a \backslash c)$. |
|
876 |
Then we can use the previous natural extension from derivative w.r.t character to derivative w.r.t string: |
|
877 |
||
878 |
\begin{center} |
|
879 |
\begin{tabular}{lcl} |
|
880 |
$r \backslash_{simp} (c\!::\!s) $ & $\dn$ & $(r \backslash_{simp} c) \backslash_{simp} s$ \\ |
|
881 |
$r \backslash [\,] $ & $\dn$ & $r$ |
|
882 |
\end{tabular} |
|
883 |
\end{center} |
|
884 |
||
885 |
we get an optimized version of the algorithm: |
|
886 |
\begin{center} |
|
887 |
\begin{tabular}{lcl} |
|
888 |
$\textit{blexer\_simp}\;r\,s$ & $\dn$ & |
|
889 |
$\textit{let}\;a = (r^\uparrow)\backslash_{simp} s\;\textit{in}$\\ |
|
890 |
& & $\;\;\textit{if}\; \textit{bnullable}(a)$\\ |
|
891 |
& & $\;\;\textit{then}\;\textit{decode}\,(\textit{bmkeps}\,a)\,r$\\ |
|
892 |
& & $\;\;\textit{else}\;\textit{None}$ |
|
893 |
\end{tabular} |
|
894 |
\end{center} |
|
48 | 895 |
|
896 |
This algorithm effectively keeps the regular expression size small, for example, |
|
897 |
with this simplification our previous $(a + aa)^*$ example's 8000 nodes will be reduced to only 6 and stay constant, however long the input string is. |
|
35 | 898 |
|
30 | 899 |
|
35 | 900 |
|
49 | 901 |
We are currently engaged in 2 tasks related to this algorithm. |
902 |
The first one is proving that our simplification rules |
|
30 | 903 |
actually do not affect the POSIX value that should be generated by the |
49 | 904 |
algorithm according to the specification of a POSIX value |
905 |
and furthermore obtain a much |
|
906 |
tighter bound on the sizes of derivatives. The result is that our |
|
907 |
algorithm should be correct and faster on all inputs. The original |
|
908 |
blow-up, as observed in JavaScript, Python and Java, would be excluded |
|
909 |
from happening in our algorithm.For |
|
30 | 910 |
this proof we use the theorem prover Isabelle. Once completed, this |
911 |
result will advance the state-of-the-art: Sulzmann and Lu wrote in |
|
912 |
their paper \cite{Sulzmann2014} about the bitcoded ``incremental |
|
913 |
parsing method'' (that is the matching algorithm outlined in this |
|
914 |
section): |
|
915 |
||
916 |
\begin{quote}\it |
|
917 |
``Correctness Claim: We further claim that the incremental parsing |
|
918 |
method in Figure~5 in combination with the simplification steps in |
|
919 |
Figure 6 yields POSIX parse trees. We have tested this claim |
|
920 |
extensively by using the method in Figure~3 as a reference but yet |
|
921 |
have to work out all proof details.'' |
|
922 |
\end{quote} |
|
923 |
||
924 |
\noindent |
|
49 | 925 |
We would settle the correctness claim. |
926 |
It is relatively straightforward to establish that after 1 simplification step, the part of derivative that corresponds to a POSIX value remains intact and can still be collected, in other words, |
|
927 |
bmkeps r = bmkeps simp r |
|
58 | 928 |
as this basically comes down to proving actions like removing the additional $r$ in $r+r$ does not delete important POSIX information in a regular expression. |
49 | 929 |
The hardcore of this problem is to prove that |
930 |
bmkeps bders r = bmkeps bders simp r |
|
58 | 931 |
That is, if we do derivative on regular expression r and the simplified version for, they can still prove the same POSIX value if there is one . This is not as straghtforward as the previous proposition, as the two regular expression r and simp r might become very different regular expressions after repeated application ofd simp and derivative. |
49 | 932 |
The crucial point is to find the "gene" of a regular expression and how it is kept intact during simplification. |
933 |
To aid this, we are utilizing the helping function retrieve described by Sulzmann and Lu: |
|
934 |
\\definition of retrieve\\ |
|
58 | 935 |
This function assembled the bitcode that corresponds to a parse tree for how the current derivative matches the suffix of the string(the characters that have not yet appeared, but is stored in the value). |
49 | 936 |
Sulzmann and Lu used this to connect the bit-coded algorithm to the older algorithm by the following equation:\\ |
53 | 937 |
$inj \;a\; c \; v = \textit{decode} \; (\textit{retrieve}\; ((\textit{internalise}\; r)\backslash_{simp} c) v)$\\ |
938 |
A little fact that needs to be stated to help comprehension:\\ |
|
939 |
$r^\uparrow = a$($a$ stands for $annotated$).\\ |
|
58 | 940 |
Fahad and Christian also used this fact to prove the correctness of bit-coded algorithm without simplification. |
50 | 941 |
Our purpose of using this, however, is try to establish \\ |
53 | 942 |
$ \textit{retrieve} \; a \; v \;=\; \textit{retrieve} \; \textit{simp}(a) \; v'.$\\ |
943 |
The idea is that using $v'$, |
|
58 | 944 |
a simplified version of $v$ that possibly had gone through the same simplification step as $\textit{simp}(a)$ we are still able to extract the bitsequence that gives the same parsing information as the unsimplified one. |
53 | 945 |
After establishing this, we might be able to finally bridge the gap of proving\\ |
946 |
$\textit{retrieve} \; r \backslash s \; v = \;\textit{retrieve} \; \textit{simp}(r) \backslash s \; v'$\\ |
|
947 |
and subsequently\\ |
|
948 |
$\textit{retrieve} \; r \backslash s \; v\; = \; \textit{retrieve} \; r \backslash_{simp} s \; v'$.\\ |
|
58 | 949 |
This proves that our simplified version of regular expression still contains all the bitcodes needed. |
49 | 950 |
|
58 | 951 |
The second task is to speed up the more aggressive simplification. Currently it is slower than a naive simplification(the naive version as implemented in ADU of course can explode in some cases). |
49 | 952 |
So it needs to be explored how to make it faster. Our possibility would be to explore again the connection to DFAs. This is very much work in progress. |
30 | 953 |
|
954 |
\section{Conclusion} |
|
955 |
||
956 |
In this PhD-project we are interested in fast algorithms for regular |
|
957 |
expression matching. While this seems to be a ``settled'' area, in |
|
958 |
fact interesting research questions are popping up as soon as one steps |
|
959 |
outside the classic automata theory (for example in terms of what kind |
|
960 |
of regular expressions are supported). The reason why it is |
|
961 |
interesting for us to look at the derivative approach introduced by |
|
962 |
Brzozowski for regular expression matching, and then much further |
|
963 |
developed by Sulzmann and Lu, is that derivatives can elegantly deal |
|
964 |
with some of the regular expressions that are of interest in ``real |
|
965 |
life''. This includes the not-regular expression, written $\neg\,r$ |
|
966 |
(that is all strings that are not recognised by $r$), but also bounded |
|
967 |
regular expressions such as $r^{\{n\}}$ and $r^{\{n..m\}}$). There is |
|
968 |
also hope that the derivatives can provide another angle for how to |
|
969 |
deal more efficiently with back-references, which are one of the |
|
970 |
reasons why regular expression engines in JavaScript, Python and Java |
|
971 |
choose to not implement the classic automata approach of transforming |
|
972 |
regular expressions into NFAs and then DFAs---because we simply do not |
|
973 |
know how such back-references can be represented by DFAs. |
|
974 |
||
975 |
||
976 |
\bibliographystyle{plain} |
|
977 |
\bibliography{root} |
|
978 |
||
979 |
||
980 |
\end{document} |