Only calculate lookahead tokens when needed - #28
Lookahead tokens are only need if either: (1) There is more than one rule that could be reduced in a given parser state, or (2) There are shift actions for a state and at least one rule that could be reduced in the same state (to warn about shift/reduce conflicts).
This commit is contained in:
parent
61339aeae9
commit
fec2c28693
@ -65,10 +65,10 @@ class Propane
|
|||||||
state_id: state_id,
|
state_id: state_id,
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
if item_set.reduce_actions
|
unless item_set.reduce_rules.empty?
|
||||||
shift_entries.each do |shift_entry|
|
shift_entries.each do |shift_entry|
|
||||||
token = shift_entry[:symbol]
|
token = shift_entry[:symbol]
|
||||||
if item_set.reduce_actions.include?(token)
|
if get_lookahead_reduce_actions_for_item_set(item_set).include?(token)
|
||||||
rule = item_set.reduce_actions[token]
|
rule = item_set.reduce_actions[token]
|
||||||
@warnings << "Shift/Reduce conflict (state #{item_set.id}) between token #{token.name} and rule #{rule.name} (defined on line #{rule.line_number})"
|
@warnings << "Shift/Reduce conflict (state #{item_set.id}) between token #{token.name} and rule #{rule.name} (defined on line #{rule.line_number})"
|
||||||
end
|
end
|
||||||
@ -114,7 +114,7 @@ class Propane
|
|||||||
# @return [void]
|
# @return [void]
|
||||||
def build_reduce_actions!
|
def build_reduce_actions!
|
||||||
@item_sets.each do |item_set|
|
@item_sets.each do |item_set|
|
||||||
item_set.reduce_actions = build_reduce_actions_for_item_set(item_set)
|
build_reduce_actions_for_item_set(item_set)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -123,44 +123,51 @@ class Propane
|
|||||||
# @param item_set [ItemSet]
|
# @param item_set [ItemSet]
|
||||||
# ItemSet (parser state)
|
# ItemSet (parser state)
|
||||||
#
|
#
|
||||||
# @return [nil, Hash]
|
# @return [void]
|
||||||
# If no reduce actions are possible for the given item set, nil.
|
|
||||||
# Otherwise, a mapping of lookahead Tokens to the Rules to reduce.
|
|
||||||
def build_reduce_actions_for_item_set(item_set)
|
def build_reduce_actions_for_item_set(item_set)
|
||||||
# To build the reduce actions, we start by looking at any
|
# To build the reduce actions, we start by looking at any
|
||||||
# "complete" items, i.e., items where the parse position is at the
|
# "complete" items, i.e., items where the parse position is at the
|
||||||
# end of a rule. These are the only rules that are candidates for
|
# end of a rule. These are the only rules that are candidates for
|
||||||
# reduction in the current ItemSet.
|
# reduction in the current ItemSet.
|
||||||
reduce_rules = Set.new(item_set.items.select(&:complete?).map(&:rule))
|
item_set.reduce_rules = Set.new(item_set.items.select(&:complete?).map(&:rule))
|
||||||
|
|
||||||
if reduce_rules.size == 1
|
if item_set.reduce_rules.size == 1
|
||||||
item_set.reduce_rule = reduce_rules.first
|
item_set.reduce_rule = item_set.reduce_rules.first
|
||||||
end
|
end
|
||||||
|
|
||||||
if reduce_rules.size == 0
|
if item_set.reduce_rules.size > 1
|
||||||
nil
|
# Force item_set.reduce_actions to be built to store the lookahead
|
||||||
else
|
# tokens for the possible reduce rules if there is more than one.
|
||||||
build_lookahead_reduce_actions_for_item_set(reduce_rules, item_set)
|
get_lookahead_reduce_actions_for_item_set(item_set)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Build the reduce actions for a single item set (parser state).
|
# Get the reduce actions for a single item set (parser state).
|
||||||
#
|
#
|
||||||
# @param reduce_rules [Set<Rule>]
|
|
||||||
# Rules to look for lookahead tokens after.
|
|
||||||
# @param item_set [ItemSet]
|
# @param item_set [ItemSet]
|
||||||
# ItemSet (parser state)
|
# ItemSet (parser state)
|
||||||
#
|
#
|
||||||
# @return [Hash]
|
# @return [Hash]
|
||||||
# Mapping of lookahead Tokens to the Rules to reduce.
|
# Mapping of lookahead Tokens to the Rules to reduce.
|
||||||
def build_lookahead_reduce_actions_for_item_set(reduce_rules, item_set)
|
def get_lookahead_reduce_actions_for_item_set(item_set)
|
||||||
|
item_set.reduce_actions ||= build_lookahead_reduce_actions_for_item_set(item_set)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Build the reduce actions for a single item set (parser state).
|
||||||
|
#
|
||||||
|
# @param item_set [ItemSet]
|
||||||
|
# ItemSet (parser state)
|
||||||
|
#
|
||||||
|
# @return [Hash]
|
||||||
|
# Mapping of lookahead Tokens to the Rules to reduce.
|
||||||
|
def build_lookahead_reduce_actions_for_item_set(item_set)
|
||||||
# We will be looking for all possible tokens that can follow instances of
|
# We will be looking for all possible tokens that can follow instances of
|
||||||
# these rules. Rather than looking through the entire grammar for the
|
# these rules. Rather than looking through the entire grammar for the
|
||||||
# possible following tokens, we will only look in the item sets leading
|
# possible following tokens, we will only look in the item sets leading
|
||||||
# up to this one. This restriction gives us a more precise lookahead set,
|
# up to this one. This restriction gives us a more precise lookahead set,
|
||||||
# and allows us to parse LALR grammars.
|
# and allows us to parse LALR grammars.
|
||||||
item_sets = Set[item_set] + item_set.leading_item_sets
|
item_sets = Set[item_set] + item_set.leading_item_sets
|
||||||
reduce_rules.reduce({}) do |reduce_actions, reduce_rule|
|
item_set.reduce_rules.reduce({}) do |reduce_actions, reduce_rule|
|
||||||
lookahead_tokens_for_rule = build_lookahead_tokens_to_reduce(reduce_rule, item_sets)
|
lookahead_tokens_for_rule = build_lookahead_tokens_to_reduce(reduce_rule, item_sets)
|
||||||
lookahead_tokens_for_rule.each do |lookahead_token|
|
lookahead_tokens_for_rule.each do |lookahead_token|
|
||||||
if existing_reduce_rule = reduce_actions[lookahead_token]
|
if existing_reduce_rule = reduce_actions[lookahead_token]
|
||||||
|
@ -2,7 +2,7 @@ class Propane
|
|||||||
class Parser
|
class Parser
|
||||||
|
|
||||||
# Represent a parser "item set", which is a set of possible items that the
|
# Represent a parser "item set", which is a set of possible items that the
|
||||||
# parser could currently be parsing.
|
# parser could currently be parsing. This is equivalent to a parser state.
|
||||||
class ItemSet
|
class ItemSet
|
||||||
|
|
||||||
# @return [Set<Item>]
|
# @return [Set<Item>]
|
||||||
@ -25,6 +25,10 @@ class Propane
|
|||||||
# Rule to reduce if there is only one possibility.
|
# Rule to reduce if there is only one possibility.
|
||||||
attr_accessor :reduce_rule
|
attr_accessor :reduce_rule
|
||||||
|
|
||||||
|
# @return [Set<Rule>]
|
||||||
|
# Set of rules that could be reduced in this parser state.
|
||||||
|
attr_accessor :reduce_rules
|
||||||
|
|
||||||
# @return [nil, Hash]
|
# @return [nil, Hash]
|
||||||
# Reduce actions, mapping lookahead tokens to rules, if there is
|
# Reduce actions, mapping lookahead tokens to rules, if there is
|
||||||
# more than one rule that could be reduced.
|
# more than one rule that could be reduced.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user