Skip to content

Latest commit

 

History

History
598 lines (419 loc) · 21 KB

internals.md

File metadata and controls

598 lines (419 loc) · 21 KB

PLY Internals

1. Introduction

This document describes classes and functions that make up the internal operation of PLY. Using this programming interface, it is possible to manually build an parser using a different interface specification than what PLY normally uses. For example, you could build a gramar from information parsed in a completely different input format. Some of these objects may be useful for building more advanced parsing engines such as GLR.

It should be stressed that using PLY at this level is not for the faint of heart. Generally, it's assumed that you know a bit of the underlying compiler theory and how an LR parser is put together.

2. Grammar Class

The file ply.yacc defines a class Grammar that is used to hold and manipulate information about a grammar specification. It encapsulates the same basic information about a grammar that is put into a YACC file including the list of tokens, precedence rules, and grammar rules. Various operations are provided to perform different validations on the grammar. In addition, there are operations to compute the first and follow sets that are needed by the various table generation algorithms.

Grammar(terminals)

: Creates a new grammar object. terminals is a list of strings specifying the terminals for the grammar. An instance g of Grammar has the following methods:

g.set_precedence(term,assoc,level)

: Sets the precedence level and associativity for a given terminal term. assoc is one of 'right', 'left', or 'nonassoc' and level is a positive integer. The higher the value of level, the higher the precedence. Here is an example of typical precedence settings:

    g.set_precedence('PLUS',  'left',1)
    g.set_precedence('MINUS', 'left',1)
    g.set_precedence('TIMES', 'left',2)
    g.set_precedence('DIVIDE','left',2)
    g.set_precedence('UMINUS','left',3)

This method must be called prior to adding any productions to the
grammar with `g.add_production()`. The precedence of individual
grammar rules is determined by the precedence of the right-most
terminal.

g.add_production(name,syms,func=None,file='',line=0)

: Adds a new grammar rule. name is the name of the rule, syms is a list of symbols making up the right hand side of the rule, func is the function to call when reducing the rule. file and line specify the filename and line number of the rule and are used for generating error messages.

The list of symbols in `syms` may include character literals and
`%prec` specifiers. Here are some examples:

    g.add_production('expr',['expr','PLUS','term'],func,file,line)
    g.add_production('expr',['expr','"+"','term'],func,file,line)
    g.add_production('expr',['MINUS','expr','%prec','UMINUS'],func,file,line)

If any kind of error is detected, a `GrammarError` exception is
raised with a message indicating the reason for the failure.

g.set_start(start=None)

: Sets the starting rule for the grammar. start is a string specifying the name of the start rule. If start is omitted, the first grammar rule added with add_production() is taken to be the starting rule. This method must always be called after all productions have been added.

g.find_unreachable()

: Diagnostic function. Returns a list of all unreachable non-terminals defined in the grammar. This is used to identify inactive parts of the grammar specification.

g.infinite_cycle()

: Diagnostic function. Returns a list of all non-terminals in the grammar that result in an infinite cycle. This condition occurs if there is no way for a grammar rule to expand to a string containing only terminal symbols.

g.undefined_symbols()

: Diagnostic function. Returns a list of tuples (name, prod) corresponding to undefined symbols in the grammar. name is the name of the undefined symbol and prod is an instance of Production which has information about the production rule where the undefined symbol was used.

g.unused_terminals()

: Diagnostic function. Returns a list of terminals that were defined, but never used in the grammar.

g.unused_rules()

: Diagnostic function. Returns a list of Production instances corresponding to production rules that were defined in the grammar, but never used anywhere. This is slightly different than find_unreachable().

g.unused_precedence()

: Diagnostic function. Returns a list of tuples (term, assoc) corresponding to precedence rules that were set, but never used the grammar. term is the terminal name and assoc is the precedence associativity (e.g., 'left', 'right', or 'nonassoc'.

g.compute_first()

: Compute all of the first sets for all symbols in the grammar. Returns a dictionary mapping symbol names to a list of all first symbols.

g.compute_follow()

: Compute all of the follow sets for all non-terminals in the grammar. The follow set is the set of all possible symbols that might follow a given non-terminal. Returns a dictionary mapping non-terminal names to a list of symbols.

g.build_lritems()

: Calculates all of the LR items for all productions in the grammar. This step is required before using the grammar for any kind of table generation. See the section on LR items below.

The following attributes are set by the above methods and may be useful in code that works with the grammar. All of these attributes should be assumed to be read-only. Changing their values directly will likely break the grammar.

g.Productions

: A list of all productions added. The first entry is reserved for a production representing the starting rule. The objects in this list are instances of the Production class, described shortly.

g.Prodnames

: A dictionary mapping the names of nonterminals to a list of all productions of that nonterminal.

g.Terminals

: A dictionary mapping the names of terminals to a list of the production numbers where they are used.

g.Nonterminals

: A dictionary mapping the names of nonterminals to a list of the production numbers where they are used.

g.First

: A dictionary representing the first sets for all grammar symbols. This is computed and returned by the compute_first() method.

g.Follow

: A dictionary representing the follow sets for all grammar rules. This is computed and returned by the compute_follow() method.

g.Start

: Starting symbol for the grammar. Set by the set_start() method.

For the purposes of debugging, a Grammar object supports the __len__() and __getitem__() special methods. Accessing g[n] returns the nth production from the grammar.

3. Productions

Grammar objects store grammar rules as instances of a Production class. This class has no public constructor--you should only create productions by calling Grammar.add_production(). The following attributes are available on a Production instance p.

p.name

: The name of the production. For a grammar rule such as A : B C D, this is 'A'.

p.prod

: A tuple of symbols making up the right-hand side of the production. For a grammar rule such as A : B C D, this is ('B','C','D').

p.number

: Production number. An integer containing the index of the production in the grammar's Productions list.

p.func

: The name of the reduction function associated with the production. This is the function that will execute when reducing the entire grammar rule during parsing.

p.callable

: The callable object associated with the name in p.func. This is None unless the production has been bound using bind().

p.file

: Filename associated with the production. Typically this is the file where the production was defined. Used for error messages.

p.lineno

: Line number associated with the production. Typically this is the line number in p.file where the production was defined. Used for error messages.

p.prec

: Precedence and associativity associated with the production. This is a tuple (assoc,level) where assoc is one of 'left','right', or 'nonassoc' and level is an integer. This value is determined by the precedence of the right-most terminal symbol in the production or by use of the %prec specifier when adding the production.

p.usyms

: A list of all unique symbols found in the production.

p.lr_items

: A list of all LR items for this production. This attribute only has a meaningful value if the Grammar.build_lritems() method has been called. The items in this list are instances of LRItem described below.

p.lr_next

: The head of a linked-list representation of the LR items in p.lr_items. This attribute only has a meaningful value if the Grammar.build_lritems() method has been called. Each LRItem instance has a lr_next attribute to move to the next item. The list is terminated by None.

p.bind(dict)

: Binds the production function name in p.func to a callable object in dict. This operation is typically carried out in the last step prior to running the parsing engine and is needed since parsing tables are typically read from files which only include the function names, not the functions themselves.

Production objects support the __len__(), __getitem__(), and __str__() special methods. len(p) returns the number of symbols in p.prod and p[n] is the same as p.prod[n].

4. LRItems

The construction of parsing tables in an LR-based parser generator is primarily done over a set of "LR Items". An LR item represents a stage of parsing one of the grammar rules. To compute the LR items, it is first necessary to call Grammar.build_lritems(). Once this step, all of the productions in the grammar will have their LR items attached to them.

Here is an interactive example that shows what LR items look like if you interactively experiment. In this example, g is a Grammar object:

>>> g.build_lritems()
>>> p = g[1]
>>> p
Production(statement -> ID = expr)
>>>

In the above code, p represents the first grammar rule. In this case, a rule 'statement -> ID = expr'.

Now, let's look at the LR items for p:

>>> p.lr_items
[LRItem(statement -> . ID = expr), 
 LRItem(statement -> ID . = expr), 
 LRItem(statement -> ID = . expr), 
 LRItem(statement -> ID = expr .)]
>>>

In each LR item, the dot (.) represents a specific stage of parsing. In each LR item, the dot is advanced by one symbol. It is only when the dot reaches the very end that a production is successfully parsed.

An instance lr of LRItem has the following attributes that hold information related to that specific stage of parsing.

lr.name

: The name of the grammar rule. For example, 'statement' in the above example.

lr.prod

: A tuple of symbols representing the right-hand side of the production, including the special '.' character. For example, ('ID','.','=','expr').

lr.number

: An integer representing the production number in the grammar.

lr.usyms

: A set of unique symbols in the production. Inherited from the original Production instance.

lr.lr_index

: An integer representing the position of the dot (.). You should never use lr.prod.index() to search for it--the result will be wrong if the grammar happens to also use (.) as a character literal.

lr.lr_after

: A list of all productions that can legally appear immediately to the right of the dot (.). This list contains Production instances. This attribute represents all of the possible branches a parse can take from the current position. For example, suppose that lr represents a stage immediately before an expression like this:

    >>> lr
    LRItem(statement -> ID = . expr)
    >>>

Then, the value of `lr.lr_after` might look like this, showing all
productions that can legally appear next:

    >>> lr.lr_after
    [Production(expr -> expr PLUS expr), 
     Production(expr -> expr MINUS expr), 
     Production(expr -> expr TIMES expr), 
     Production(expr -> expr DIVIDE expr), 
     Production(expr -> MINUS expr), 
     Production(expr -> LPAREN expr RPAREN), 
     Production(expr -> NUMBER), 
     Production(expr -> ID)]
    >>>

lr.lr_before

: The grammar symbol that appears immediately before the dot (.) or None if at the beginning of the parse.

lr.lr_next

: A link to the next LR item, representing the next stage of the parse. None if lr is the last LR item.

LRItem instances also support the __len__() and __getitem__() special methods. len(lr) returns the number of items in lr.prod including the dot (.). lr[n] returns lr.prod[n].

It goes without saying that all of the attributes associated with LR items should be assumed to be read-only. Modifications will very likely create a small black-hole that will consume you and your code.

5. LRTable

The LRTable class represents constructed LR parsing tables on a grammar.

LRTable(grammar, log=None)

: Create the LR parsing tables on a grammar. grammar is an instance of Grammar and log is a logger object used to write debugging information. The debugging information written to log is the same as what appears in the parser.out file created by yacc. By supplying a custom logger with a different message format, it is possible to get more information (e.g., the line number in yacc.py used for issuing each line of output in the log).

An instance lr of LRTable has the following attributes.

lr.grammar

: A link to the Grammar object used to construct the parsing tables.

lr.lr_method

: The LR parsing method used (e.g., 'LALR')

lr.lr_productions

: A reference to grammar.Productions. This, together with lr_action and lr_goto contain all of the information needed by the LR parsing engine.

lr.lr_action

: The LR action dictionary that implements the underlying state machine. The keys of this dictionary are the LR states.

lr.lr_goto

: The LR goto table that contains information about grammar rule reductions.

lr.sr_conflicts

: A list of tuples (state,token,resolution) identifying all shift/reduce conflicts. state is the LR state number where the conflict occurred, token is the token causing the conflict, and resolution is a string describing the resolution taken. resolution is either 'shift' or 'reduce'.

lr.rr_conflicts

: A list of tuples (state,rule,rejected) identifying all reduce/reduce conflicts. state is the LR state number where the conflict occurred, rule is the production rule that was selected and rejected is the production rule that was rejected. Both rule and rejected are instances of Production. They can be inspected to provide the user with more information.

lrtab.bind_callables(dict)

: This binds all of the function names used in productions to callable objects found in the dictionary dict. During table generation and when reading LR tables from files, PLY only uses the names of action functions such as 'p_expr', 'p_statement', etc. In order to actually run the parser, these names have to be bound to callable objects. This method is always called prior to running a parser.

6. LRParser

The LRParser class implements the low-level LR parsing engine.

LRParser(lrtab, error_func)

: Create an LRParser. lrtab is an instance of LRTable containing the LR production and state tables. error_func is the error function to invoke in the event of a parsing error.

An instance p of LRParser has the following methods:

p.parse(input=None,lexer=None,debug=0,tracking=0)

: Run the parser. input is a string, which if supplied is fed into the lexer using its input() method. lexer is an instance of the Lexer class to use for tokenizing. If not supplied, the last lexer created with the lex module is used. debug is a boolean flag that enables debugging. tracking is a boolean flag that tells the parser to perform additional line number tracking.

p.restart()

: Resets the parser state for a parse already in progress.

7. ParserReflect

The ParserReflect class is used to collect parser specification data from a Python module or object. This class is what collects all of the p_rule() functions in a PLY file, performs basic error checking, and collects all of the needed information to build a grammar. Most of the high-level PLY interface as used by the yacc() function is actually implemented by this class.

ParserReflect(pdict, log=None)

: Creates a ParserReflect instance. pdict is a dictionary containing parser specification data. This dictionary typically corresponds to the module or class dictionary of code that implements a PLY parser. log is a logger instance that will be used to report error messages.

An instance p of ParserReflect has the following methods:

p.get_all()

: Collect and store all required parsing information.

p.validate_all()

: Validate all of the collected parsing information. This is a seprate step from p.get_all() as a performance optimization. In order to increase parser start-up time, a parser can elect to only validate the parsing data when regenerating the parsing tables. The validation step tries to collect as much information as possible rather than raising an exception at the first sign of trouble. The attribute p.error is set if there are any validation errors. The value of this attribute is also returned.

p.signature()

: Compute a signature representing the contents of the collected parsing data. The signature value should change if anything in the parser specification has changed in a way that would justify parser table regeneration. This method can be called after p.get_all(), but before p.validate_all().

The following attributes are set in the process of collecting data:

p.start

: The grammar start symbol, if any. Taken from pdict['start'].

p.error_func

: The error handling function or None. Taken from pdict['p_error'].

p.tokens

: The token list. Taken from pdict['tokens'].

p.prec

: The precedence specifier. Taken from pdict['precedence'].

p.preclist

: A parsed version of the precedence specified. A list of tuples of the form (token,assoc,level) where token is the terminal symbol, assoc is the associativity (e.g., 'left') and level is a numeric precedence level.

p.grammar

: A list of tuples (name, rules) representing the grammar rules. name is the name of a Python function or method in pdict that starts with "p_". rules is a list of tuples (filename,line,prodname,syms) representing the grammar rules found in the documentation string of that function. filename and line contain location information that can be used for debugging. prodname is the name of the production. syms is the right-hand side of the production. If you have a function like this:

    def p_expr(p):
        '''expr : expr PLUS expr
                | expr MINUS expr
                | expr TIMES expr
                | expr DIVIDE expr'''

then the corresponding entry in `p.grammar` might look like this:

    ('p_expr', [ ('calc.py',10,'expr', ['expr','PLUS','expr']),
                 ('calc.py',11,'expr', ['expr','MINUS','expr']),
                 ('calc.py',12,'expr', ['expr','TIMES','expr']),
                 ('calc.py',13,'expr', ['expr','DIVIDE','expr'])
               ])

p.pfuncs

: A sorted list of tuples (line, file, name, doc) representing all of the p_ functions found. line and file give location information. name is the name of the function. doc is the documentation string. This list is sorted in ascending order by line number.

p.files

: A dictionary holding all of the source filenames that were encountered while collecting parser information. Only the keys of this dictionary have any meaning.

p.error

: An attribute that indicates whether or not any critical errors occurred in validation. If this is set, it means that that some kind of problem was detected and that no further processing should be performed.

8. High-level operation

Using all of the above classes requires some attention to detail. The yacc() function carries out a very specific sequence of operations to create a grammar. This same sequence should be emulated if you build an alternative PLY interface.

1. A ParserReflect object is created and raw grammar specification data is collected.

2. A Grammar object is created and populated with information from the specification data.

3. A LRTable object is created to run the LALR algorithm over the Grammar object.

4. Productions in the LRTable and bound to callables using the bind_callables() method.

5. A LRParser object is created from from the information in the LRTable object.