The next step though will be to replace the Configuration with an integer, and intern all Configurations, along with all other objects.
1067 lines
37 KiB
Python
1067 lines
37 KiB
Python
"""I wanted to try to use the code in `parser.py` to do real work, and as you
|
|
might expect the code did NOT work acceptibly.
|
|
|
|
This version has some performance work done.
|
|
|
|
2023
|
|
"""
|
|
import dataclasses
|
|
import functools
|
|
import typing
|
|
|
|
|
|
###############################################################################
|
|
# LR0
|
|
#
|
|
# We start with LR0 parsers, because they form the basis of everything else.
|
|
###############################################################################
|
|
class Configuration:
|
|
"""A rule being tracked in a state.
|
|
|
|
(Note: technically, lookahead isn't used until we get to LR(1) parsers,
|
|
but if left at its default it's harmless. Ignore it until you get to
|
|
the part about LR(1).)
|
|
"""
|
|
__slots__ = (
|
|
'name',
|
|
'symbols',
|
|
'position',
|
|
'lookahead',
|
|
'next',
|
|
'at_end',
|
|
'_vals',
|
|
'_hash',
|
|
)
|
|
|
|
name: str
|
|
symbols: typing.Tuple[str, ...]
|
|
position: int
|
|
lookahead: typing.Tuple[str, ...]
|
|
next: str | None
|
|
at_end: bool
|
|
|
|
_vals: typing.Tuple
|
|
_hash: int
|
|
|
|
def __init__(self, name, symbols, position, lookahead) -> None:
|
|
self.name = name
|
|
self.symbols = symbols
|
|
self.position = position
|
|
self.lookahead = lookahead
|
|
|
|
at_end = position == len(symbols)
|
|
self.at_end = at_end
|
|
self.next = symbols[position] if not at_end else None
|
|
|
|
self._vals = (name, symbols, position, lookahead)
|
|
self._hash = hash(self._vals)
|
|
|
|
@classmethod
|
|
def from_rule(cls, name: str, symbols: typing.Tuple[str, ...], lookahead=()):
|
|
return Configuration(
|
|
name=name,
|
|
symbols=symbols,
|
|
position=0,
|
|
lookahead=lookahead,
|
|
)
|
|
|
|
def __hash__(self) -> int:
|
|
return self._hash
|
|
|
|
def __eq__(self, value: object, /) -> bool:
|
|
if value is self:
|
|
return True
|
|
if not isinstance(value, Configuration):
|
|
return NotImplemented
|
|
|
|
return (
|
|
value._hash == self._hash and
|
|
value.name == self.name and
|
|
value.position == self.position and
|
|
value.symbols == self.symbols and
|
|
value.lookahead == self.lookahead
|
|
)
|
|
|
|
def __lt__(self, value) -> bool:
|
|
if not isinstance(value, Configuration):
|
|
return NotImplemented
|
|
return self._vals < value._vals
|
|
|
|
def __gt__(self, value) -> bool:
|
|
if not isinstance(value, Configuration):
|
|
return NotImplemented
|
|
return self._vals > value._vals
|
|
|
|
def __le__(self, value) -> bool:
|
|
if not isinstance(value, Configuration):
|
|
return NotImplemented
|
|
return self._vals <= value._vals
|
|
|
|
def __ge__(self, value) -> bool:
|
|
if not isinstance(value, Configuration):
|
|
return NotImplemented
|
|
return self._vals >= value._vals
|
|
|
|
def replace_position(self, new_position):
|
|
return Configuration(
|
|
name=self.name,
|
|
symbols=self.symbols,
|
|
position=new_position,
|
|
lookahead=self.lookahead,
|
|
)
|
|
|
|
def clear_lookahead(self):
|
|
return Configuration(
|
|
name=self.name,
|
|
symbols=self.symbols,
|
|
position=self.position,
|
|
lookahead=(),
|
|
)
|
|
|
|
@property
|
|
def rest(self):
|
|
return self.symbols[(self.position+1):]
|
|
|
|
def at_symbol(self, symbol):
|
|
return self.next == symbol
|
|
|
|
def __str__(self):
|
|
la = ", " + str(self.lookahead) if self.lookahead != () else ""
|
|
return "{name} -> {bits}{lookahead}".format(
|
|
name=self.name,
|
|
bits=' '.join([
|
|
'* ' + sym if i == self.position else sym
|
|
for i, sym in enumerate(self.symbols)
|
|
]) + (' *' if self.at_end else ''),
|
|
lookahead=la,
|
|
)
|
|
|
|
ConfigSet = typing.Tuple[Configuration, ...]
|
|
|
|
class TableBuilder(object):
|
|
def __init__(self):
|
|
self.errors = []
|
|
self.table = []
|
|
self.row = None
|
|
|
|
def flush(self):
|
|
self._flush_row()
|
|
if len(self.errors) > 0:
|
|
raise ValueError("\n\n".join(self.errors))
|
|
return self.table
|
|
|
|
def new_row(self, config_set):
|
|
self._flush_row()
|
|
self.row = {}
|
|
self.current_config_set = config_set
|
|
|
|
def _flush_row(self):
|
|
if self.row:
|
|
actions = {k: v[0] for k,v in self.row.items()}
|
|
self.table.append(actions)
|
|
|
|
|
|
def set_table_reduce(self, symbol, config):
|
|
action = ('reduce', config.name, len(config.symbols))
|
|
self._set_table_action(symbol, action, config)
|
|
|
|
def set_table_accept(self, config):
|
|
action = ('accept',)
|
|
self._set_table_action('$', action, config)
|
|
|
|
def set_table_shift(self, index, config):
|
|
action = ('shift', index)
|
|
self._set_table_action(config.next, action, config)
|
|
|
|
def set_table_goto(self, symbol, index):
|
|
action = ('goto', index)
|
|
self._set_table_action(symbol, action, None)
|
|
|
|
def _set_table_action(self, symbol, action, config):
|
|
"""Set the action for 'symbol' in the table row to 'action'.
|
|
|
|
This is destructive; it changes the table. It raises an error if
|
|
there is already an action for the symbol in the row.
|
|
"""
|
|
assert self.row is not None
|
|
existing, existing_config = self.row.get(symbol, (None, None))
|
|
if existing is not None and existing != action:
|
|
config_old = str(existing_config)
|
|
config_new = str(config)
|
|
max_len = max(len(config_old), len(config_new)) + 1
|
|
error = (
|
|
"Conflicting actions for token '{symbol}':\n"
|
|
" {config_old: <{max_len}}: {old}\n"
|
|
" {config_new: <{max_len}}: {new}\n".format(
|
|
config_old=config_old,
|
|
config_new=config_new,
|
|
max_len=max_len,
|
|
old=existing,
|
|
new=action,
|
|
symbol=symbol,
|
|
)
|
|
)
|
|
self.errors.append(error)
|
|
self.row[symbol] = (action, config)
|
|
|
|
|
|
class GenerateLR0(object):
|
|
"""Generate parser tables for an LR0 parser.
|
|
|
|
The input grammars are of the form:
|
|
|
|
grammar_simple = [
|
|
('E', ['E', '+', 'T']),
|
|
('E', ['T']),
|
|
('T', ['(', 'E', ')']),
|
|
('T', ['id']),
|
|
]
|
|
|
|
Which is to say, they are a list of productions. Each production is a
|
|
tuple where the first element of the tuple is the name of the
|
|
non-terminal being added, and the second elment of the tuple is the
|
|
list of terminals and non-terminals that make up the production.
|
|
|
|
There is currently no support for custom actions or alternation or
|
|
anything like that. If you want alternations that you'll have to lower
|
|
the grammar by hand into the simpler form first.
|
|
|
|
Don't name anything with double-underscores; those are reserved for
|
|
the generator. Don't add '$' either, as it is reserved to mean
|
|
end-of-stream. Use an empty list to indicate nullability, that is:
|
|
|
|
('O', []),
|
|
|
|
means that O can be matched with nothing.
|
|
|
|
Implementation notes:
|
|
- This is implemented in the dumbest way possible, in order to be the
|
|
most understandable it can be. I built this to learn, and I want to
|
|
make sure I can keep learning with it.
|
|
|
|
- We tend to use tuples everywhere. This is because tuples can be
|
|
compared for equality and put into tables and all that jazz. They might
|
|
be a little bit slower in places but like I said, this is for
|
|
learning. (Also, if we need this to run faster we can probably go a
|
|
long way by memoizing results, which is much easier if we have tuples
|
|
everywhere.)
|
|
"""
|
|
|
|
grammar: dict[str, list[typing.Tuple[str, ...]]]
|
|
nonterminals: set[str]
|
|
terminals: set[str]
|
|
alphabet: list[str]
|
|
|
|
def __init__(self, start: str, grammar: list[typing.Tuple[str, list[str]]]):
|
|
"""Initialize the parser generator with the specified grammar and
|
|
start symbol.
|
|
"""
|
|
|
|
# Turn the incoming grammar into a dictionary, indexed by nonterminal.
|
|
#
|
|
# We count on python dictionaries retaining the insertion order, like
|
|
# it or not.
|
|
full_grammar = {}
|
|
for name, rule in grammar:
|
|
rules = full_grammar.get(name)
|
|
if rules is None:
|
|
rules = []
|
|
full_grammar[name] = rules
|
|
rules.append(tuple(rule))
|
|
self.grammar = full_grammar
|
|
|
|
|
|
self.nonterminals = set(self.grammar.keys())
|
|
self.terminals = {
|
|
sym
|
|
for _, symbols in grammar
|
|
for sym in symbols
|
|
if sym not in self.nonterminals
|
|
}
|
|
self.alphabet = list(sorted(self.terminals | self.nonterminals))
|
|
|
|
# Check to make sure they didn't use anything that will give us
|
|
# heartburn later.
|
|
reserved = [a for a in self.alphabet if a.startswith('__') or a == '$']
|
|
if reserved:
|
|
raise ValueError(
|
|
"Can't use {symbols} in grammars, {what} reserved.".format(
|
|
symbols=' or '.join(reserved),
|
|
what="it's" if len(reserved) == 1 else "they're",
|
|
)
|
|
)
|
|
|
|
|
|
self.grammar['__start'] = [(start,)]
|
|
self.terminals.add('$')
|
|
self.alphabet.append('$')
|
|
|
|
|
|
@functools.cache
|
|
def gen_closure_next(self, config: Configuration):
|
|
"""Return the next set of configurations in the closure for
|
|
config.
|
|
|
|
If the position for config is just before a non-terminal, then the
|
|
next set of configurations is configurations for all of the
|
|
productions for that non-terminal, with the position at the
|
|
beginning. (If the position for config is just before a terminal,
|
|
or at the end of the production, then the next set is empty.)
|
|
"""
|
|
next = config.next
|
|
if next is None:
|
|
return ()
|
|
else:
|
|
return tuple(
|
|
Configuration.from_rule(next, rule)
|
|
for rule in self.grammar.get(next, ())
|
|
)
|
|
|
|
@functools.cache
|
|
def gen_closure(self, seeds: typing.Iterable[Configuration]) -> ConfigSet:
|
|
"""Compute the closure for the specified configs. The closure is all
|
|
of the configurations we could be in. Specifically, if the position
|
|
for a config is just before a non-terminal then we must also consider
|
|
configurations where the rule is the rule for the non-terminal and
|
|
the position is just before the beginning of the rule.
|
|
|
|
(We have replaced a recursive version with an iterative one.)
|
|
"""
|
|
closure = set()
|
|
pending = list(seeds)
|
|
while len(pending) > 0:
|
|
config = pending.pop()
|
|
if config in closure:
|
|
continue
|
|
|
|
closure.add(config)
|
|
for next_config in self.gen_closure_next(config):
|
|
pending.append(next_config)
|
|
|
|
return tuple(sorted(closure)) # TODO: Why tuple?
|
|
|
|
@functools.cache
|
|
def gen_successor(self, config_set: typing.Iterable[Configuration], symbol: str) -> ConfigSet:
|
|
"""Compute the successor state for the given config set and the
|
|
given symbol.
|
|
|
|
The successor represents the next state of the parser after seeing
|
|
the symbol.
|
|
"""
|
|
seeds = tuple(
|
|
config.replace_position(config.position + 1)
|
|
for config in config_set
|
|
if config.at_symbol(symbol)
|
|
)
|
|
|
|
closure = self.gen_closure(seeds)
|
|
return closure
|
|
|
|
def gen_all_successors(self, config_set: typing.Iterable[Configuration]) -> list[ConfigSet]:
|
|
"""Return all of the non-empty successors for the given config set."""
|
|
next = []
|
|
for symbol in self.alphabet:
|
|
successor = self.gen_successor(config_set, symbol)
|
|
if len(successor) > 0:
|
|
next.append(successor)
|
|
|
|
return next
|
|
|
|
def gen_sets(self, config_set: typing.Tuple[Configuration,...]) -> typing.Tuple[ConfigSet, ...]:
|
|
"""Generate all configuration sets starting from the provided set."""
|
|
# NOTE: Not a set because we need to maintain insertion order!
|
|
# The first element in the dictionary needs to be the intial
|
|
# set.
|
|
F = {}
|
|
pending = [config_set]
|
|
while len(pending) > 0:
|
|
config_set = pending.pop()
|
|
if config_set in F:
|
|
continue
|
|
# print(f"pending: {len(pending)} F: {len(F)}")
|
|
|
|
F[config_set] = len(F)
|
|
for successor in self.gen_all_successors(config_set):
|
|
pending.append(successor)
|
|
|
|
return tuple(F.keys())
|
|
|
|
|
|
def gen_all_sets(self) -> typing.Tuple[ConfigSet, ...]:
|
|
"""Generate all of the configuration sets for the grammar."""
|
|
seeds = tuple(
|
|
Configuration.from_rule('__start', rule)
|
|
for rule in self.grammar['__start']
|
|
)
|
|
initial_set = self.gen_closure(seeds)
|
|
return self.gen_sets(initial_set)
|
|
|
|
def build_set_index(self, sets: typing.Tuple[ConfigSet, ...]) -> dict[ConfigSet, int]:
|
|
return { s: index for index, s in enumerate(sets) }
|
|
|
|
def find_set_index(self, sets: dict[ConfigSet, int], s: ConfigSet) -> int | None:
|
|
"""Find the specified set in the set of sets, and return the
|
|
index, or None if it is not found.
|
|
"""
|
|
return sets.get(s)
|
|
|
|
def gen_reduce_set(self, config: Configuration) -> typing.Iterable[str]:
|
|
"""Return the set of symbols that indicate we should reduce the given
|
|
configuration.
|
|
|
|
In an LR0 parser, this is just the set of all terminals."""
|
|
del(config)
|
|
return self.terminals
|
|
|
|
def gen_table(self):
|
|
"""Generate the parse table.
|
|
|
|
The parse table is a list of states. The first state in the list is
|
|
the starting state. Each state is a dictionary that maps a symbol to an
|
|
action. Each action is a tuple. The first element of the tuple is a
|
|
string describing what to do:
|
|
|
|
- 'shift': The second element of the tuple is the state
|
|
number. Consume the input and push that state onto the stack.
|
|
|
|
- 'reduce': The second element is the name of the non-terminal being
|
|
reduced, and the third element is the number of states to remove
|
|
from the stack. Don't consume the input; just remove the specified
|
|
number of things from the stack, and then consult the table again,
|
|
this time using the new top-of-stack as the current state and the
|
|
name of the non-terminal to find out what to do.
|
|
|
|
- 'goto': The second element is the state number to push onto the
|
|
stack. In the literature, these entries are treated distinctly from
|
|
the actions, but we mix them here because they never overlap with the
|
|
other actions. (These are always associated with non-terminals, and
|
|
the other actions are always associated with terminals.)
|
|
|
|
- 'accept': Accept the result of the parse, it worked.
|
|
|
|
Anything missing from the row indicates an error.
|
|
"""
|
|
builder = TableBuilder()
|
|
|
|
config_sets = self.gen_all_sets()
|
|
set_index = self.build_set_index(config_sets)
|
|
|
|
for config_set in config_sets:
|
|
builder.new_row(config_set)
|
|
|
|
# Actions
|
|
for config in config_set:
|
|
if config.at_end:
|
|
if config.name != '__start':
|
|
for a in self.gen_reduce_set(config):
|
|
builder.set_table_reduce(a, config)
|
|
else:
|
|
builder.set_table_accept(config)
|
|
|
|
else:
|
|
if config.next in self.terminals:
|
|
successor = self.gen_successor(config_set, config.next)
|
|
index = self.find_set_index(set_index, successor)
|
|
assert index is not None
|
|
builder.set_table_shift(index, config)
|
|
|
|
# Gotos
|
|
for symbol in self.nonterminals:
|
|
successor = self.gen_successor(config_set, symbol)
|
|
index = self.find_set_index(set_index, successor)
|
|
if index is not None:
|
|
builder.set_table_goto(symbol, index)
|
|
|
|
|
|
return builder.flush()
|
|
|
|
def set_table_action(self, errors, row, symbol, action, config):
|
|
"""Set the action for 'symbol' in the table row to 'action'.
|
|
|
|
This is destructive; it changes the table. It raises an error if
|
|
there is already an action for the symbol in the row.
|
|
"""
|
|
existing, existing_config = row.get(symbol, (None, None))
|
|
if existing is not None and existing != action:
|
|
config_old = str(existing_config)
|
|
config_new = str(config)
|
|
max_len = max(len(config_old), len(config_new)) + 1
|
|
error = (
|
|
"Conflicting actions for token '{symbol}':\n"
|
|
" {config_old: <{max_len}}: {old}\n"
|
|
" {config_new: <{max_len}}: {new}\n".format(
|
|
config_old=config_old,
|
|
config_new=config_new,
|
|
max_len=max_len,
|
|
old=existing,
|
|
new=action,
|
|
symbol=symbol,
|
|
)
|
|
)
|
|
errors.append(error)
|
|
row[symbol] = (action, config)
|
|
|
|
|
|
def parse(table, input, trace=False):
|
|
"""Parse the input with the generated parsing table and return the
|
|
concrete syntax tree.
|
|
|
|
The parsing table can be generated by GenerateLR0.gen_table() or by any
|
|
of the other generators below. The parsing mechanism never changes, only
|
|
the table generation mechanism.
|
|
|
|
input is a list of tokens. Don't stick an end-of-stream marker, I'll stick
|
|
one on for you.
|
|
"""
|
|
assert '$' not in input
|
|
input = input + ['$']
|
|
input_index = 0
|
|
|
|
# Our stack is a stack of tuples, where the first entry is the state number
|
|
# and the second entry is the 'value' that was generated when the state was
|
|
# pushed.
|
|
stack : list[typing.Tuple[int, typing.Any]] = [(0, None)]
|
|
while True:
|
|
current_state = stack[-1][0]
|
|
current_token = input[input_index]
|
|
|
|
action = table[current_state].get(current_token, ('error',))
|
|
if trace:
|
|
print("{stack: <20} {input: <50} {action: <5}".format(
|
|
stack=repr([s[0] for s in stack]),
|
|
input=repr(input[input_index:]),
|
|
action=repr(action)
|
|
))
|
|
|
|
if action[0] == 'accept':
|
|
return stack[-1][1]
|
|
|
|
elif action[0] == 'reduce':
|
|
name = action[1]
|
|
size = action[2]
|
|
|
|
value = (name, tuple(s[1] for s in stack[-size:]))
|
|
stack = stack[:-size]
|
|
|
|
goto = table[stack[-1][0]].get(name, ('error',))
|
|
assert goto[0] == 'goto' # Corrupt table?
|
|
stack.append((goto[1], value))
|
|
|
|
elif action[0] == 'shift':
|
|
stack.append((action[1], (current_token, ())))
|
|
input_index += 1
|
|
|
|
elif action[0] == 'error':
|
|
raise ValueError(
|
|
'Syntax error: unexpected symbol {sym}'.format(
|
|
sym=current_token,
|
|
),
|
|
)
|
|
|
|
|
|
###############################################################################
|
|
# SLR(1)
|
|
###############################################################################
|
|
def add_changed(items: set, item)->bool:
|
|
old_len = len(items)
|
|
items.add(item)
|
|
return old_len != len(items)
|
|
|
|
def update_changed(items: set, other: set) -> bool:
|
|
old_len = len(items)
|
|
items.update(other)
|
|
return old_len != len(items)
|
|
|
|
@dataclasses.dataclass(frozen=True)
|
|
class FirstInfo:
|
|
firsts: dict[str, set[str]]
|
|
is_epsilon: set[str]
|
|
|
|
@classmethod
|
|
def from_grammar(
|
|
cls,
|
|
grammar: dict[str, list[typing.Tuple[str,...]]],
|
|
terminals: set[str],
|
|
):
|
|
firsts = {name: set() for name in grammar.keys()}
|
|
for t in terminals:
|
|
firsts[t] = {t}
|
|
|
|
epsilons = set()
|
|
changed = True
|
|
while changed:
|
|
changed = False
|
|
for name, rules in grammar.items():
|
|
f = firsts[name]
|
|
for rule in rules:
|
|
if len(rule) == 0:
|
|
changed = add_changed(epsilons, name) or changed
|
|
continue
|
|
|
|
for index, symbol in enumerate(rule):
|
|
if symbol in terminals:
|
|
changed = add_changed(f, symbol) or changed
|
|
else:
|
|
other_firsts = firsts[symbol]
|
|
changed = update_changed(f, other_firsts) or changed
|
|
|
|
is_last = index == len(rule) - 1
|
|
if is_last and symbol in epsilons:
|
|
# If this is the last symbol and the last
|
|
# symbol can be empty then I can be empty
|
|
# too! :P
|
|
changed = add_changed(epsilons, name) or changed
|
|
|
|
if symbol not in epsilons:
|
|
# If we believe that there is at least one
|
|
# terminal in the first set of this
|
|
# nonterminal then I don't have to keep
|
|
# looping through the symbols in this rule.
|
|
break
|
|
|
|
return FirstInfo(firsts=firsts, is_epsilon=epsilons)
|
|
|
|
@dataclasses.dataclass(frozen=True)
|
|
class FollowInfo:
|
|
follows: dict[str, set[str]]
|
|
|
|
@classmethod
|
|
def from_grammar(
|
|
cls,
|
|
grammar: dict[str, list[typing.Tuple[str,...]]],
|
|
firsts: FirstInfo,
|
|
):
|
|
follows = {name: set() for name in grammar.keys()}
|
|
follows["__start"].add('$')
|
|
|
|
changed = True
|
|
while changed:
|
|
changed = False
|
|
for name, rules in grammar.items():
|
|
for rule in rules:
|
|
epsilon = True
|
|
prev_symbol = None
|
|
for symbol in reversed(rule):
|
|
f = follows.get(symbol)
|
|
if f is None:
|
|
# This particular rule can't produce epsilon.
|
|
epsilon = False
|
|
prev_symbol = symbol
|
|
continue
|
|
|
|
# While epsilon is still set, update the follow of
|
|
# this nonterminal with the follow of the production
|
|
# we're processing. (This also means that the follow
|
|
# of the last symbol in the production is the follow
|
|
# of the entire production, as it should be.)
|
|
if epsilon:
|
|
changed = update_changed(f, follows[name]) or changed
|
|
|
|
# If we're not at the end of the list then the follow
|
|
# of the current symbol contains the first of the
|
|
# next symbol.
|
|
if prev_symbol is not None:
|
|
changed = update_changed(f, firsts.firsts[prev_symbol]) or changed
|
|
|
|
# Now if there's no epsilon in this symbol there's no
|
|
# more epsilon in the rest of the sequence.
|
|
if symbol not in firsts.is_epsilon:
|
|
epsilon = False
|
|
|
|
prev_symbol = symbol
|
|
|
|
return FollowInfo(follows=follows)
|
|
|
|
|
|
|
|
class GenerateSLR1(GenerateLR0):
|
|
"""Generate parse tables for SLR1 grammars.
|
|
|
|
SLR1 parsers can recognize more than LR0 parsers, because they have a
|
|
little bit more information: instead of generating reduce actions for a
|
|
production on all possible inputs, as LR0 parsers do, they generate
|
|
reduce actions only for inputs that are in the 'follow' set of the
|
|
non-terminal.
|
|
|
|
That means SLR1 parsers need to know how to generate 'follow(A)', which
|
|
means they need to know how to generate 'first(A)', which is most of the
|
|
code in this class.
|
|
"""
|
|
_firsts: FirstInfo
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, **kwargs)
|
|
self._firsts = FirstInfo.from_grammar(self.grammar, self.terminals)
|
|
self._follows = FollowInfo.from_grammar(self.grammar, self._firsts)
|
|
|
|
def gen_first(self, symbols: typing.Iterable[str]) -> typing.Tuple[set[str], bool]:
|
|
"""Return the first set for a sequence of symbols.
|
|
|
|
Build the set by combining the first sets of the symbols from left to
|
|
right as long as epsilon remains in the first set. If we reach the end
|
|
and every symbol has had epsilon, then this set also has epsilon.
|
|
|
|
Otherwise we can stop as soon as we get to a non-epsilon first(), and
|
|
our result does not have epsilon.
|
|
"""
|
|
result = set()
|
|
for s in symbols:
|
|
result.update(self._firsts.firsts[s])
|
|
if s not in self._firsts.is_epsilon:
|
|
return (result, False)
|
|
|
|
return (result, True)
|
|
|
|
def gen_follow(self, symbol: str) -> set[str]:
|
|
"""Generate the follow set for the given nonterminal.
|
|
|
|
The follow set for a nonterminal is the set of terminals that can
|
|
follow the nonterminal in a valid sentence. The resulting set never
|
|
contains epsilon and is never empty, since we should always at least
|
|
ground out at '$', which is the end-of-stream marker.
|
|
"""
|
|
assert symbol in self.grammar
|
|
return self._follows.follows[symbol]
|
|
|
|
def gen_reduce_set(self, config: Configuration) -> typing.Iterable[str]:
|
|
"""Return the set of symbols that indicate we should reduce the given
|
|
config.
|
|
|
|
In an SLR1 parser, this is the follow set of the config nonterminal."""
|
|
return self.gen_follow(config.name)
|
|
|
|
|
|
class GenerateLR1(GenerateSLR1):
|
|
"""Generate parse tables for LR1, or "canonical LR" grammars.
|
|
|
|
LR1 parsers can recognize more than SLR parsers. Like SLR parsers, they
|
|
are choosier about when they reduce. But unlike SLR parsers, they specify
|
|
the terminals on which they reduce by carrying a 'lookahead' terminal in
|
|
the configuration. The lookahead of a configuration is computed as the
|
|
closure of a configuration set is computed, so see gen_closure_next for
|
|
details. (Except for the start configuration, which has '$' as its
|
|
lookahead.)
|
|
"""
|
|
def gen_reduce_set(self, config: Configuration) -> typing.Iterable[str]:
|
|
"""Return the set of symbols that indicate we should reduce the given
|
|
config.
|
|
|
|
In an LR1 parser, this is the lookahead of the configuration."""
|
|
return config.lookahead
|
|
|
|
@functools.cache
|
|
def gen_closure_next(self, config: Configuration):
|
|
"""Return the next set of configurations in the closure for
|
|
config.
|
|
|
|
In LR1 parsers, we must compute the lookahead for the configurations
|
|
we're adding to the closure. The lookahead for the new configurations
|
|
is the first() of the rest of this config's production. If that
|
|
contains epsilon, then the lookahead *also* contains the lookahead we
|
|
already have. (This lookahead was presumably generated by the same
|
|
process, so in some sense it is a 'parent' lookahead, or a lookahead
|
|
from an upstream production in the grammar.)
|
|
|
|
(See the documentation in GenerateLR0 for more information on how
|
|
this function fits into the whole process.)
|
|
"""
|
|
config_next = config.next
|
|
if config_next is None:
|
|
return ()
|
|
else:
|
|
next = []
|
|
for rule in self.grammar.get(config_next, ()):
|
|
lookahead, epsilon = self.gen_first(config.rest)
|
|
if epsilon:
|
|
lookahead.update(config.lookahead)
|
|
lookahead = tuple(sorted(lookahead))
|
|
next.append(Configuration.from_rule(config_next, rule, lookahead=lookahead))
|
|
|
|
return tuple(next)
|
|
|
|
def gen_all_sets(self):
|
|
"""Generate all of the configuration sets for the grammar.
|
|
|
|
In LR1 parsers, we must remember to set the lookahead of the start
|
|
symbol to '$'.
|
|
"""
|
|
seeds = tuple(
|
|
Configuration.from_rule('__start', rule, lookahead=('$',))
|
|
for rule in self.grammar['__start']
|
|
)
|
|
initial_set = self.gen_closure(seeds)
|
|
return self.gen_sets(initial_set)
|
|
|
|
|
|
class GenerateLALR(GenerateLR1):
|
|
"""Generate tables for LALR.
|
|
|
|
LALR is smaller than LR(1) but bigger than SLR(1). It works by generating
|
|
the LR(1) configuration sets, but merging configuration sets which are
|
|
equal in everything but their lookaheads. This works in that it doesn't
|
|
generate any shift/reduce conflicts that weren't already in the LR(1)
|
|
grammar. It can, however, introduce new reduce/reduce conflicts, because
|
|
it does lose information. The advantage is that the number of parser
|
|
states is much much smaller in LALR than in LR(1).
|
|
|
|
(Note that because we use immutable state everywhere this generator does
|
|
a lot of copying and allocation.)
|
|
"""
|
|
def merge_sets(self, config_set_a, config_set_b):
|
|
"""Merge the two config sets, by keeping the item cores but merging
|
|
the lookahead sets for each item.
|
|
"""
|
|
assert len(config_set_a) == len(config_set_b)
|
|
merged = []
|
|
for index, a in enumerate(config_set_a):
|
|
b = config_set_b[index]
|
|
assert a.clear_lookahead() == b.clear_lookahead()
|
|
|
|
new_lookahead = a.lookahead + b.lookahead
|
|
new_lookahead = tuple(sorted(set(new_lookahead)))
|
|
merged.append(a.clear_lookahead())
|
|
|
|
return tuple(merged)
|
|
|
|
def sets_equal(self, a, b):
|
|
a_no_la = tuple(s.clear_lookahead() for s in a)
|
|
b_no_la = tuple(s.clear_lookahead() for s in b)
|
|
return a_no_la == b_no_la
|
|
|
|
def gen_sets(self, config_set):
|
|
"""Recursively generate all configuration sets starting from the
|
|
provided set, and merge them with the provided set 'F'.
|
|
|
|
The difference between this method and the one in GenerateLR0, where
|
|
this comes from, is in the part that stops recursion. In LALR we
|
|
compare for set equality *ignoring lookahead*. If we find a match,
|
|
then instead of returning F unchanged, we merge the two equal sets
|
|
and replace the set in F, returning the modified set.
|
|
"""
|
|
F = {}
|
|
pending = [config_set]
|
|
while len(pending) > 0:
|
|
config_set = pending.pop()
|
|
config_set_no_la = tuple(s.clear_lookahead() for s in config_set)
|
|
|
|
existing = F.get(config_set_no_la)
|
|
if existing is not None:
|
|
F[config_set_no_la] = self.merge_sets(config_set, existing)
|
|
else:
|
|
F[config_set_no_la] = config_set
|
|
for successor in self.gen_all_successors(config_set):
|
|
pending.append(successor)
|
|
|
|
# NOTE: We count on insertion order here! The first element must be the
|
|
# starting state!
|
|
return tuple(F.values())
|
|
|
|
def set_without_lookahead(self, config_set: ConfigSet) -> ConfigSet:
|
|
return tuple(sorted(set(c.clear_lookahead() for c in config_set)))
|
|
|
|
def build_set_index(self, sets: typing.Tuple[ConfigSet, ...]) -> dict[ConfigSet, int]:
|
|
index = {}
|
|
for s in sets:
|
|
s_no_la = self.set_without_lookahead(s)
|
|
if s_no_la not in index:
|
|
index[s_no_la] = len(index)
|
|
return index
|
|
|
|
def find_set_index(self, sets: dict[ConfigSet, int], s: ConfigSet) -> int | None:
|
|
"""Find the specified set in the set of sets, and return the
|
|
index, or None if it is not found.
|
|
"""
|
|
s_no_la = self.set_without_lookahead(s)
|
|
return sets.get(s_no_la)
|
|
|
|
|
|
###############################################################################
|
|
# Formatting
|
|
###############################################################################
|
|
def format_node(node):
|
|
"""Print out an indented concrete syntax tree, from parse()."""
|
|
lines = [
|
|
'{name}'.format(name=node[0])
|
|
] + [
|
|
' ' + line
|
|
for child in node[1]
|
|
for line in format_node(child).split('\n')
|
|
]
|
|
return '\n'.join(lines)
|
|
|
|
|
|
def format_table(generator, table):
|
|
"""Format a parser table so pretty."""
|
|
def format_action(state, terminal):
|
|
action = state.get(terminal, ('error',))
|
|
if action[0] == 'accept':
|
|
return 'accept'
|
|
elif action[0] == 'shift':
|
|
return 's' + str(action[1])
|
|
elif action[0] == 'error':
|
|
return ''
|
|
elif action[0] == 'reduce':
|
|
return 'r' + str(action[1])
|
|
|
|
header = " | {terms} | {nts}".format(
|
|
terms=' '.join(
|
|
'{0: <6}'.format(terminal)
|
|
for terminal in sorted(generator.terminals)
|
|
),
|
|
nts=' '.join(
|
|
'{0: <5}'.format(nt)
|
|
for nt in sorted(generator.nonterminals)
|
|
),
|
|
)
|
|
|
|
lines = [
|
|
header,
|
|
'-' * len(header),
|
|
] + [
|
|
"{index: <3} | {actions} | {gotos}".format(
|
|
index=i,
|
|
actions=' '.join(
|
|
'{0: <6}'.format(format_action(row, terminal))
|
|
for terminal in sorted(generator.terminals)
|
|
),
|
|
gotos=' '.join(
|
|
'{0: <5}'.format(row.get(nt, ('error', ''))[1])
|
|
for nt in sorted(generator.nonterminals)
|
|
),
|
|
)
|
|
for i, row in enumerate(table)
|
|
]
|
|
return '\n'.join(lines)
|
|
|
|
|
|
###############################################################################
|
|
# Examples
|
|
###############################################################################
|
|
def examples():
|
|
# OK, this is a very simple LR0 grammar.
|
|
print("grammar_simple:")
|
|
grammar_simple = [
|
|
('E', ['E', '+', 'T']),
|
|
('E', ['T']),
|
|
('T', ['(', 'E', ')']),
|
|
('T', ['id']),
|
|
]
|
|
|
|
gen = GenerateLR0('E', grammar_simple)
|
|
table = gen.gen_table()
|
|
print(format_table(gen, table))
|
|
tree = parse(table, ['id', '+', '(', 'id', ')'])
|
|
print(format_node(tree) + "\n")
|
|
print()
|
|
|
|
# This one doesn't work with LR0, though, it has a shift/reduce conflict.
|
|
print("grammar_lr0_shift_reduce (LR0):")
|
|
grammar_lr0_shift_reduce = grammar_simple + [
|
|
('T', ['id', '[', 'E', ']']),
|
|
]
|
|
try:
|
|
gen = GenerateLR0('E', grammar_lr0_shift_reduce)
|
|
table = gen.gen_table()
|
|
assert False
|
|
except ValueError as e:
|
|
print(e)
|
|
print()
|
|
|
|
# Nor does this: it has a reduce/reduce conflict.
|
|
print("grammar_lr0_reduce_reduce (LR0):")
|
|
grammar_lr0_reduce_reduce = grammar_simple + [
|
|
('E', ['V', '=', 'E']),
|
|
('V', ['id']),
|
|
]
|
|
try:
|
|
gen = GenerateLR0('E', grammar_lr0_reduce_reduce)
|
|
table = gen.gen_table()
|
|
assert False
|
|
except ValueError as e:
|
|
print(e)
|
|
print()
|
|
|
|
# Nullable symbols just don't work with constructs like this, because you can't
|
|
# look ahead to figure out if you should reduce an empty 'F' or not.
|
|
print("grammar_nullable (LR0):")
|
|
grammar_nullable = [
|
|
('E', ['F', 'boop']),
|
|
('F', ['beep']),
|
|
('F', []),
|
|
]
|
|
try:
|
|
gen = GenerateLR0('E', grammar_nullable)
|
|
table = gen.gen_table()
|
|
assert False
|
|
except ValueError as e:
|
|
print(e)
|
|
|
|
print("grammar_lr0_shift_reduce (SLR1):")
|
|
gen = GenerateSLR1('E', grammar_lr0_shift_reduce)
|
|
first, epsilon=gen.gen_first(('E',))
|
|
print(f"First: {str(first)} (epsilon={epsilon})")
|
|
print(f"Follow: {str(gen.gen_follow('E'))}")
|
|
table = gen.gen_table()
|
|
print(format_table(gen, table))
|
|
tree = parse(table, ['id', '+', '(', 'id', '[', 'id', ']', ')'], trace=True)
|
|
print(format_node(tree) + "\n")
|
|
print()
|
|
|
|
# SLR1 can't handle this.
|
|
print("grammar_aho_ullman_1 (SLR1):")
|
|
grammar_aho_ullman_1 = [
|
|
('S', ['L', '=', 'R']),
|
|
('S', ['R']),
|
|
('L', ['*', 'R']),
|
|
('L', ['id']),
|
|
('R', ['L']),
|
|
]
|
|
try:
|
|
gen = GenerateSLR1('S', grammar_aho_ullman_1)
|
|
table = gen.gen_table()
|
|
assert False
|
|
except ValueError as e:
|
|
print(e)
|
|
print()
|
|
|
|
# Here's an example with a full LR1 grammar, though.
|
|
print("grammar_aho_ullman_2 (LR1):")
|
|
grammar_aho_ullman_2 = [
|
|
('S', ['X', 'X']),
|
|
('X', ['a', 'X']),
|
|
('X', ['b']),
|
|
]
|
|
gen = GenerateLR1('S', grammar_aho_ullman_2)
|
|
table = gen.gen_table()
|
|
print(format_table(gen, table))
|
|
parse(table, ['b', 'a', 'a', 'b'], trace=True)
|
|
print()
|
|
|
|
# What happens if we do LALR to it?
|
|
print("grammar_aho_ullman_2 (LALR):")
|
|
gen = GenerateLALR('S', grammar_aho_ullman_2)
|
|
table = gen.gen_table()
|
|
print(format_table(gen, table))
|
|
print()
|
|
|
|
# A fun LALAR grammar.
|
|
print("grammar_lalr:")
|
|
grammar_lalr = [
|
|
('S', ['V', 'E']),
|
|
|
|
('E', ['F']),
|
|
('E', ['E', '+', 'F']),
|
|
|
|
('F', ['V']),
|
|
('F', ['int']),
|
|
('F', ['(', 'E', ')']),
|
|
|
|
('V', ['id']),
|
|
]
|
|
gen = GenerateLALR('S', grammar_lalr)
|
|
table = gen.gen_table()
|
|
print(format_table(gen, table))
|
|
print()
|
|
|
|
if __name__=="__main__":
|
|
examples()
|