532
|
1 |
% Chapter Template
|
|
2 |
|
|
3 |
\chapter{A Better Bound and Other Extensions} % Main chapter title
|
|
4 |
|
|
5 |
\label{Cubic} %In Chapter 5\ref{Chapter5} we discuss stronger simplifications to improve the finite bound
|
|
6 |
%in Chapter 4 to a polynomial one, and demonstrate how one can extend the
|
|
7 |
%algorithm to include constructs such as bounded repetitions and negations.
|
|
8 |
|
|
9 |
%----------------------------------------------------------------------------------------
|
|
10 |
% SECTION strongsimp
|
|
11 |
%----------------------------------------------------------------------------------------
|
|
12 |
\section{A Stronger Version of Simplification}
|
|
13 |
%TODO: search for isabelle proofs of algorithms that check equivalence
|
533
|
14 |
In our bit-coded lexing algorithm, with or without simplification,
|
|
15 |
two alternative (distinct) sub-matches for the (sub-)string and (sub-)regex pair
|
532
|
16 |
are always expressed in the "derivative regular expression" as two
|
|
17 |
disjoint alternative terms at the current (sub-)regex level. The fact that they
|
|
18 |
are parallel tells us when we retrieve the information from this (sub-)regex
|
|
19 |
there will always be a choice of which alternative term to take.
|
533
|
20 |
As an example, the regular expression $aa \cdot a^*+ a \cdot a^*$ (omitting bit-codes)
|
|
21 |
expresses two possibilities it will match future input.
|
|
22 |
It will either match 2 $a$'s, then 0 or more $a$'s, in other words, at least 2 more $a$'s
|
|
23 |
\begin{figure}\label{string_3parts1}
|
|
24 |
\begin{center}
|
|
25 |
\begin{tikzpicture}[every node/.append style={draw, rounded corners, inner sep=10pt}]
|
|
26 |
\node [rectangle split, rectangle split horizontal, rectangle split parts=3]
|
|
27 |
{Consumed Input
|
|
28 |
\nodepart{two} Expects $aa$
|
|
29 |
\nodepart{three} Then expects $a^*$};
|
|
30 |
|
|
31 |
\end{tikzpicture}
|
|
32 |
\end{center}
|
|
33 |
\end{figure}
|
|
34 |
\noindent
|
|
35 |
Or it will match at least 1 more $a$:
|
|
36 |
\begin{figure}\label{string_3parts2}
|
|
37 |
\begin{center}
|
|
38 |
\begin{tikzpicture}[every node/.append style={draw, rounded corners, inner sep=10pt}]
|
|
39 |
\node [rectangle split, rectangle split horizontal, rectangle split parts=3]
|
|
40 |
{Consumed
|
|
41 |
\nodepart{two} Expects $a$
|
|
42 |
\nodepart{three} Then expects $a^*$};
|
|
43 |
|
|
44 |
\end{tikzpicture}
|
|
45 |
\end{center}
|
|
46 |
\end{figure}
|
|
47 |
If these two possibilities are identical, we can eliminate
|
|
48 |
the second one as we know the second one corresponds to
|
|
49 |
a match that is not $\POSIX$.
|
|
50 |
|
|
51 |
|
|
52 |
If two identical regexes
|
|
53 |
happen to be grouped into different sequences, one cannot use
|
|
54 |
the $a + a \rightsquigarrow a$ rule to eliminate them, even if they
|
|
55 |
are "parallel" to each other:
|
|
56 |
\[
|
|
57 |
(a+b) \cdot r_1 + (a+c) \cdot r_2 \rightsquigarrow (a+b) \cdot r_1 + (c) \cdot r_2
|
|
58 |
\]
|
|
59 |
and that's because they are followed by possibly
|
|
60 |
different "suffixes" $r_1$ and $r_2$, and if
|
|
61 |
they do turn out to be different then doing
|
|
62 |
\[
|
|
63 |
(a+b) \cdot r_1 + (a+c) \cdot r_2 \rightsquigarrow (a+b) \cdot r_1 + (c) \cdot r_2
|
|
64 |
\]
|
|
65 |
might cause a possible match where the string is in $L(a \cdot r_2)$ to be lost.
|
532
|
66 |
|
|
67 |
Here is an example for this.
|
|
68 |
Assume we have the derivative regex
|
533
|
69 |
\[( r_1 + r_2 + r_3)\cdot r+
|
|
70 |
( r_1 + r_5 + r_6)\cdot( r_1 + r_2 + r_3)^*
|
532
|
71 |
\]
|
|
72 |
|
533
|
73 |
occurring in an intermediate step in our bit-coded lexing algorithm.
|
|
74 |
|
532
|
75 |
In this expression, there will be 6 "parallel" terms if we break up regexes
|
|
76 |
of shape $(a+b)\cdot c$ using the distributivity law (into $ac + bc$).
|
|
77 |
\begin{align}
|
|
78 |
(\nonumber \\
|
533
|
79 |
r_1 + & \label{term:1} \\
|
|
80 |
r_2 + & \label{term:2} \\
|
|
81 |
r_3 & \label{term:3} \\
|
|
82 |
& )\cdot r\; + \; (\nonumber \\
|
|
83 |
r_1 + & \label{term:4} \\
|
|
84 |
r_5 + & \label{term:5} \\
|
|
85 |
r_6 \label{term:6}\\
|
|
86 |
& )\cdot r\nonumber
|
532
|
87 |
\end{align}
|
|
88 |
|
|
89 |
They have been labelled, and each label denotes
|
|
90 |
one term, for example, \ref{term:1} denotes
|
|
91 |
\[
|
533
|
92 |
r_1 \cdot r
|
532
|
93 |
\]
|
|
94 |
\noindent
|
|
95 |
and \ref{term:3} denotes
|
|
96 |
\[
|
533
|
97 |
r_3\cdot r.
|
532
|
98 |
\]
|
533
|
99 |
From a lexer's point of view, \ref{term:4} will never
|
|
100 |
be picked up in later phases of matching because there
|
|
101 |
is a term \ref{term:1} giving identical matching information.
|
|
102 |
The first term \ref{term:1} will match a string in $L(r_1 \cdot r)$,
|
|
103 |
and so on for term \ref{term:2}, \ref{term:3}, etc.
|
|
104 |
|
532
|
105 |
\mybox{previous input $\ldots$}\mybox{$aaa$ }\mybox{rest of input $\ldots$}
|
533
|
106 |
\begin{center}\label{string_2parts}
|
|
107 |
|
|
108 |
\begin{tikzpicture}[every node/.append style={draw, rounded corners, inner sep=10pt}]
|
|
109 |
\node [rectangle split, rectangle split horizontal, rectangle split parts=2]
|
|
110 |
{$r_{x1}$
|
|
111 |
\nodepart{two} $r_1\cdot r$ };
|
|
112 |
%\caption{term 1 \ref{term:1}'s matching configuration}
|
|
113 |
\end{tikzpicture}
|
|
114 |
|
|
115 |
\end{center}
|
|
116 |
For term 1 \ref{term:1}, whatever was before the current
|
|
117 |
input position was fully matched and the regex corresponding
|
|
118 |
to it reduced to $\ONE$,
|
|
119 |
and in the next input token, it will start on $r_1\cdot r$.
|
|
120 |
The resulting value will be something of a similar configuration:
|
|
121 |
\begin{center}\label{value_2parts}
|
|
122 |
%\caption{term 1 \ref{term:1}'s matching configuration}
|
|
123 |
\begin{tikzpicture}[every node/.append style={draw, rounded corners, inner sep=10pt}]
|
|
124 |
\node [rectangle split, rectangle split horizontal, rectangle split parts=2]
|
|
125 |
{$v_{x1}$
|
|
126 |
\nodepart{two} $v_{r_1\cdot r}$ };
|
|
127 |
\end{tikzpicture}
|
|
128 |
\end{center}
|
|
129 |
For term 2 \ref{term:2} we have a similar value partition:
|
|
130 |
\begin{center}\label{value_2parts2}
|
|
131 |
%\caption{term 1 \ref{term:1}'s matching configuration}
|
|
132 |
\begin{tikzpicture}[every node/.append style={draw, rounded corners, inner sep=10pt}]
|
|
133 |
\node [rectangle split, rectangle split horizontal, rectangle split parts=2]
|
|
134 |
{$v_{x2}$
|
|
135 |
\nodepart{two} $v_{r_2\cdot r}$ };
|
|
136 |
\end{tikzpicture}
|
|
137 |
\end{center}
|
|
138 |
\noindent
|
|
139 |
and so on.
|
|
140 |
We note that for term 4 \ref{term:4} its result value
|
|
141 |
after any position beyond the current one will always
|
|
142 |
be the same:
|
|
143 |
\begin{center}\label{value_2parts4}
|
|
144 |
%\caption{term 1 \ref{term:1}'s matching configuration}
|
|
145 |
\begin{tikzpicture}[every node/.append style={draw, rounded corners, inner sep=10pt}]
|
|
146 |
\node [rectangle split, rectangle split horizontal, rectangle split parts=2]
|
|
147 |
{$v_{x4}$
|
|
148 |
\nodepart{two} $v_{r_1\cdot r}$ };
|
|
149 |
\end{tikzpicture}
|
|
150 |
\end{center}
|
|
151 |
And $\POSIX$ rules says that we can eliminate at least one of them.
|
|
152 |
Our algorithm always puts the regex with the longest initial sub-match at the left of the
|
|
153 |
alternatives, so we safely throw away \ref{term:4}.
|
|
154 |
The fact that term 1 and 4 are not immediately in the same alternative
|
|
155 |
expression does not prevent them from being two duplicate matches at
|
|
156 |
the current point of view.
|
|
157 |
To implement this idea into an algorithm, we define a "pruning function"
|
|
158 |
$\textit{prune}$, that takes away parts of all terms in $r$ that belongs to
|
|
159 |
$\textit{acc}$, which acts as an element
|
|
160 |
is a stronger version of $\textit{distinctBy}$.
|
|
161 |
Here is a concise version of it (in the style of Scala):
|
|
162 |
\begin{verbatim}
|
|
163 |
def distinctByPlus(rs: List[ARexp], acc: Set[Rexp] = Set()) :
|
|
164 |
List[ARexp] = rs match {
|
|
165 |
case Nil => Nil
|
|
166 |
case r :: rs =>
|
|
167 |
if(acc.contains(erase(r)))
|
|
168 |
distinctByPlus(rs, acc)
|
|
169 |
else
|
|
170 |
prune(r, acc) :: distinctByPlus(rs, prune(r, acc) +: acc)
|
|
171 |
}
|
|
172 |
|
|
173 |
\end{verbatim}
|
|
174 |
But for the function $\textit{prune}$ there is a difficulty:
|
|
175 |
how could one define the $L$ function in a "computable" way,
|
|
176 |
so that they generate a (lazy infinite) set of strings in $L(r)$.
|
|
177 |
We also need a function that tests whether $L(r_1) \subseteq L(r_2)$
|
|
178 |
is true.
|
|
179 |
For the moment we cut corners and do not define these two functions
|
|
180 |
as they are not used by Antimirov (and will probably not contribute
|
|
181 |
to a bound better than Antimirov's cubic bound).
|
|
182 |
Rather, we do not try to eliminate in every instance of regular expressions
|
|
183 |
that have "language duplicates", but only eliminate the "exact duplicates".
|
|
184 |
For this we need to break up terms $(a+b)\cdot c$ into two
|
|
185 |
terms $a\cdot c$ and $b\cdot c$ before we add them to the accumulator:
|
535
|
186 |
\begin{figure}
|
533
|
187 |
\begin{verbatim}
|
|
188 |
def distinctWith(rs: List[ARexp],
|
|
189 |
pruneFunction: (ARexp, Set[Rexp]) => ARexp,
|
|
190 |
acc: Set[Rexp] = Set()) : List[ARexp] =
|
|
191 |
rs match{
|
|
192 |
case Nil => Nil
|
|
193 |
case r :: rs =>
|
|
194 |
if(acc(erase(r)))
|
|
195 |
distinctWith(rs, pruneFunction, acc)
|
|
196 |
else {
|
|
197 |
val pruned_r = pruneFunction(r, acc)
|
|
198 |
pruned_r ::
|
|
199 |
distinctWith(rs,
|
|
200 |
pruneFunction,
|
|
201 |
turnIntoTerms(erase(pruned_r)) ++: acc
|
|
202 |
)
|
|
203 |
}
|
|
204 |
}
|
|
205 |
\end{verbatim}
|
535
|
206 |
\caption{A Stronger Version of $\textit{distinctBy}$}
|
|
207 |
\end{figure}
|
533
|
208 |
\noindent
|
|
209 |
This process is done by the function $\textit{turnIntoTerms}$:
|
535
|
210 |
\begin{figure}
|
533
|
211 |
\begin{verbatim}
|
|
212 |
def turnIntoTerms(r: Rexp): List[Rexp] = r match {
|
|
213 |
case SEQ(r1, r2) => if(isOne(r1))
|
|
214 |
turnIntoTerms(r2)
|
|
215 |
else
|
|
216 |
turnIntoTerms(r1).map(r11 => SEQ(r11, r2))
|
|
217 |
case ALTS(r1, r2) => turnIntoTerms(r1) ::: turnIntoTerms(r2)
|
|
218 |
case ZERO => Nil
|
|
219 |
case _ => r :: Nil
|
|
220 |
}
|
|
221 |
\end{verbatim}
|
535
|
222 |
\caption{function that breaks up regular expression into "atomic" terms}
|
|
223 |
\end{figure}
|
|
224 |
|
533
|
225 |
\noindent
|
|
226 |
The pruning function can be defined recursively:
|
535
|
227 |
\begin{figure}
|
533
|
228 |
\begin{verbatim}
|
|
229 |
def prune7(r: ARexp, acc: Set[Rexp]) : ARexp = r match{
|
|
230 |
case AALTS(bs, rs) => rs.map(r => prune7(r, acc)).filter(_ != ZERO) match
|
|
231 |
{
|
|
232 |
case Nil => AZERO
|
|
233 |
case r::Nil => fuse(bs, r)
|
|
234 |
case rs1 => AALTS(bs, rs1)
|
|
235 |
}
|
|
236 |
case ASEQ(bs, r1, r2) => prune7(r1, acc.map(r => removeSeqTail(r, erase(r2)))) match {
|
|
237 |
case AZERO => AZERO
|
|
238 |
case r1p if(isOne(erase(r1p))) => fuse(bs ++ mkepsBC(r1p), r2)
|
|
239 |
case r1p => ASEQ(bs, r1p, r2)
|
|
240 |
}
|
|
241 |
case r => if(acc(erase(r))) AZERO else r
|
|
242 |
}
|
|
243 |
\end{verbatim}
|
535
|
244 |
\caption{pruning function}
|
|
245 |
\end{figure}
|
|
246 |
|
|
247 |
|
|
248 |
\pgfplotsset{
|
|
249 |
myplotstyle/.style={
|
|
250 |
legend style={draw=none, font=\small},
|
|
251 |
legend cell align=left,
|
|
252 |
legend pos=north east,
|
|
253 |
ylabel style={align=center, font=\bfseries\boldmath},
|
|
254 |
xlabel style={align=center, font=\bfseries\boldmath},
|
|
255 |
x tick label style={font=\bfseries\boldmath},
|
|
256 |
y tick label style={font=\bfseries\boldmath},
|
|
257 |
scaled ticks=true,
|
|
258 |
every axis plot/.append style={thick},
|
|
259 |
},
|
|
260 |
}
|
533
|
261 |
|
|
262 |
\begin{figure}
|
|
263 |
\centering
|
|
264 |
\begin{tabular}{@{}c@{\hspace{0mm}}c@{\hspace{0mm}}c@{}}
|
|
265 |
\begin{tikzpicture}
|
|
266 |
\begin{axis}[
|
535
|
267 |
%xlabel={$n$},
|
|
268 |
myplotstyle,
|
|
269 |
xlabel={input length},
|
533
|
270 |
ylabel={size},
|
535
|
271 |
]
|
533
|
272 |
\addplot[red,mark=*, mark options={fill=white}] table {strongSimpCurve.data};
|
|
273 |
\end{axis}
|
|
274 |
\end{tikzpicture}
|
|
275 |
&
|
|
276 |
\begin{tikzpicture}
|
|
277 |
\begin{axis}[
|
535
|
278 |
%xlabel={$n$},
|
|
279 |
myplotstyle,
|
|
280 |
xlabel={input length},
|
533
|
281 |
ylabel={size},
|
535
|
282 |
]
|
533
|
283 |
\addplot[blue,mark=*, mark options={fill=white}] table {bsimpExponential.data};
|
|
284 |
\end{axis}
|
|
285 |
\end{tikzpicture}\\
|
535
|
286 |
\multicolumn{2}{l}{Graphs: Runtime for matching $((a^* + (aa)^* + \ldots + (aaaaa)^* )^*)^*$ with strings
|
533
|
287 |
of the form $\underbrace{aa..a}_{n}$.}
|
|
288 |
\end{tabular}
|
|
289 |
\caption{aaaaaStarStar} \label{fig:aaaaaStarStar}
|
|
290 |
\end{figure}
|
532
|
291 |
|
|
292 |
|
|
293 |
|
|
294 |
%----------------------------------------------------------------------------------------
|
|
295 |
% SECTION 1
|
|
296 |
%----------------------------------------------------------------------------------------
|
|
297 |
|
|
298 |
\section{Adding Support for the Negation Construct, and its Correctness Proof}
|
|
299 |
We now add support for the negation regular expression:
|
|
300 |
\[ r ::= \ZERO \mid \ONE
|
|
301 |
\mid c
|
|
302 |
\mid r_1 \cdot r_2
|
|
303 |
\mid r_1 + r_2
|
|
304 |
\mid r^*
|
|
305 |
\mid \sim r
|
|
306 |
\]
|
|
307 |
The $\textit{nullable}$ function's clause for it would be
|
|
308 |
\[
|
|
309 |
\textit{nullable}(~r) = \neg \nullable(r)
|
|
310 |
\]
|
|
311 |
The derivative would be
|
|
312 |
\[
|
|
313 |
~r \backslash c = ~ (r \backslash c)
|
|
314 |
\]
|
|
315 |
|
|
316 |
The most tricky part of lexing for the $~r$ regex
|
|
317 |
is creating a value for it.
|
|
318 |
For other regular expressions, the value aligns with the
|
|
319 |
structure of the regex:
|
|
320 |
\[
|
|
321 |
\vdash \Seq(\Char(a), \Char(b)) : a \cdot b
|
|
322 |
\]
|
|
323 |
But for the $~r$ regex, $s$ is a member of it if and only if
|
|
324 |
$s$ does not belong to $L(r)$.
|
|
325 |
That means when there
|
|
326 |
is a match for the not regex, it is not possible to generate how the string $s$ matched
|
|
327 |
with $r$.
|
|
328 |
What we can do is preserve the information of how $s$ was not matched by $r$,
|
|
329 |
and there are a number of options to do this.
|
|
330 |
|
|
331 |
We could give a partial value when there is a partial match for the regex inside
|
|
332 |
the $\mathbf{not}$ construct.
|
|
333 |
For example, the string $ab$ is not in the language of $(a\cdot b) \cdot c$,
|
|
334 |
A value for it could be
|
|
335 |
\[
|
|
336 |
\vdash \textit{Not}(\Seq(\Char(a), \Char(b))) : ~((a \cdot b ) \cdot c)
|
|
337 |
\]
|
|
338 |
The above example demonstrates what value to construct
|
|
339 |
when the string $s$ is at most a real prefix
|
|
340 |
of the strings in $L(r)$. When $s$ instead is not a prefix of any strings
|
|
341 |
in $L(r)$, it becomes unclear what to return as a value inside the $\textit{Not}$
|
|
342 |
constructor.
|
|
343 |
|
|
344 |
Another option would be to either store the string $s$ that resulted in
|
|
345 |
a mis-match for $r$ or a dummy value as a placeholder:
|
|
346 |
\[
|
533
|
347 |
\vdash \textit{Not}(abcd) : ~( r_1 )
|
532
|
348 |
\]
|
|
349 |
or
|
|
350 |
\[
|
533
|
351 |
\vdash \textit{Not}(\textit{Dummy}) : ~( r_1 )
|
532
|
352 |
\]
|
|
353 |
We choose to implement this as it is most straightforward:
|
|
354 |
\[
|
|
355 |
\mkeps(~(r)) = \textit{if}(\nullable(r)) \; \textit{Error} \; \textit{else} \; \textit{Not}(\textit{Dummy})
|
|
356 |
\]
|
|
357 |
|
|
358 |
%----------------------------------------------------------------------------------------
|
|
359 |
% SECTION 2
|
|
360 |
%----------------------------------------------------------------------------------------
|
|
361 |
|
|
362 |
\section{Bounded Repetitions}
|
|
363 |
|
|
364 |
|