⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 parser2.sml

📁 这是我们参加06年全国开源软件的竞赛作品
💻 SML
📖 第 1 页 / 共 2 页
字号:
(* ML-Yacc Parser Generator (c) 1989 Andrew W. Appel, David R. Tarditi  * * $Log: parser2.sml,v $ * Revision 1.1.1.1  1999/12/03 19:59:22  dbm * Import of 110.0.6 src * * Revision 1.2  1997/08/26 19:18:54  jhr *   Replaced used of "abstraction" with ":>". *# Revision 1.1.1.1  1997/01/14  01:38:04  george#   Version 109.24# * Revision 1.3  1996/10/03  03:36:58  jhr * Qualified identifiers that are no-longer top-level (quot, rem, min, max). * * Revision 1.2  1996/02/26  15:02:29  george *    print no longer overloaded. *    use of makestring has been removed and replaced with Int.toString .. *    use of IO replaced with TextIO * * Revision 1.1.1.1  1996/01/31  16:01:42  george * Version 109 *  *)(* parser.sml:  This is a parser driver for LR tables with an error-recovery   routine added to it.  The routine used is described in detail in this   article:	'A Practical Method for LR and LL Syntactic Error Diagnosis and	 Recovery', by M. Burke and G. Fisher, ACM Transactions on	 Programming Langauges and Systems, Vol. 9, No. 2, April 1987,	 pp. 164-197.    This program is an implementation is the partial, deferred method discussed    in the article.  The algorithm and data structures used in the program    are described below.      This program assumes that all semantic actions are delayed.  A semantic    action should produce a function from unit -> value instead of producing the    normal value.  The parser returns the semantic value on the top of the    stack when accept is encountered.  The user can deconstruct this value    and apply the unit -> value function in it to get the answer.    It also assumes that the lexer is a lazy stream.    Data Structures:    ----------------		* The parser:	   The state stack has the type		 (state * (semantic value * line # * line #)) list	   The parser keeps a queue of (state stack * lexer pair).  A lexer pair	 consists of a terminal * value pair and a lexer.  This allows the 	 parser to reconstruct the states for terminals to the left of a	 syntax error, and attempt to make error corrections there.	   The queue consists of a pair of lists (x,y).  New additions to	 the queue are cons'ed onto y.  The first element of x is the top	 of the queue.  If x is nil, then y is reversed and used	 in place of x.    Algorithm:    ----------	* The steady-state parser:  	    This parser keeps the length of the queue of state stacks at	a steady state by always removing an element from the front when	another element is placed on the end.	    It has these arguments:	   stack: current stack	   queue: value of the queue	   lexPair ((terminal,value),lex stream)	When SHIFT is encountered, the state to shift to and the value are	are pushed onto the state stack.  The state stack and lexPair are	placed on the queue.  The front element of the queue is removed.	When REDUCTION is encountered, the rule is applied to the current	stack to yield a triple (nonterm,value,new stack).  A new	stack is formed by adding (goto(top state of stack,nonterm),value)	to the stack.	When ACCEPT is encountered, the top value from the stack and the	lexer are returned.	When an ERROR is encountered, fixError is called.  FixError	takes the arguments to the parser, fixes the error if possible and        returns a new set of arguments.	* The distance-parser:	This parser includes an additional argument distance.  It pushes	elements on the queue until it has parsed distance tokens, or an	ACCEPT or ERROR occurs.  It returns a stack, lexer, the number of	tokens left unparsed, a queue, and an action option.*)signature FIFO =   sig type 'a queue      val empty : 'a queue      exception Empty      val get : 'a queue -> 'a * 'a queue      val put : 'a * 'a queue -> 'a queue  end(* drt (12/15/89) -- the functor should be used in development work, but   it wastes space in the release version.functor ParserGen(structure LrTable : LR_TABLE		  structure Stream : STREAM) : LR_PARSER =*)structure LrParser :> LR_PARSER =   struct      structure LrTable = LrTable      structure Stream = Stream      structure Token : TOKEN =	struct	    structure LrTable = LrTable	    datatype ('a,'b) token = TOKEN of LrTable.term * ('a * 'b * 'b)	    val sameToken = fn (TOKEN(t,_),TOKEN(t',_)) => t=t'        end      open LrTable      open Token      val DEBUG1 = false      val DEBUG2 = false      exception ParseError      exception ParseImpossible of int      structure Fifo :> FIFO =        struct	  type 'a queue = ('a list * 'a list)	  val empty = (nil,nil)	  exception Empty	  fun get(a::x, y) = (a, (x,y))	    | get(nil, nil) = raise Empty	    | get(nil, y) = get(rev y, nil) 	  fun put(a,(x,y)) = (x,a::y)        end      type ('a,'b) elem = (state * ('a * 'b * 'b))      type ('a,'b) stack = ('a,'b) elem list      type ('a,'b) lexv = ('a,'b) token      type ('a,'b) lexpair = ('a,'b) lexv * (('a,'b) lexv Stream.stream)      type ('a,'b) distanceParse =		 ('a,'b) lexpair *		 ('a,'b) stack * 		 (('a,'b) stack * ('a,'b) lexpair) Fifo.queue *		 int ->		   ('a,'b) lexpair *		   ('a,'b) stack * 		   (('a,'b) stack * ('a,'b) lexpair) Fifo.queue *		   int *		   action option      type ('a,'b) ecRecord =	 {is_keyword : term -> bool,          preferred_change : (term list * term list) list,	  error : string * 'b * 'b -> unit,	  errtermvalue : term -> 'a,	  terms : term list,	  showTerminal : term -> string,	  noShift : term -> bool}      local 	 val print = fn s => TextIO.output(TextIO.stdOut,s)	 val println = fn s => (print s; print "\n")	 val showState = fn (STATE s) => "STATE " ^ (Int.toString s)      in        fun printStack(stack: ('a,'b) stack, n: int) =         case stack           of (state,_) :: rest =>                 (print("\t" ^ Int.toString n ^ ": ");                  println(showState state);                  printStack(rest, n+1))            | nil => ()                        fun prAction showTerminal		 (stack as (state,_) :: _, next as (TOKEN (term,_),_), action) =             (println "Parse: state stack:";              printStack(stack, 0);              print("       state="                         ^ showState state	                         ^ " next="                         ^ showTerminal term                         ^ " action="                        );              case action                of SHIFT state => println ("SHIFT " ^ (showState state))                 | REDUCE i => println ("REDUCE " ^ (Int.toString i))                 | ERROR => println "ERROR"		 | ACCEPT => println "ACCEPT")        | prAction _ (_,_,action) = ()     end    (* ssParse: parser which maintains the queue of (state * lexvalues) in a	steady-state.  It takes a table, showTerminal function, saction	function, and fixError function.  It parses until an ACCEPT is	encountered, or an exception is raised.  When an error is encountered,	fixError is called with the arguments of parseStep (lexv,stack,and	queue).  It returns the lexv, and a new stack and queue adjusted so	that the lexv can be parsed *)	    val ssParse =      fn (table,showTerminal,saction,fixError,arg) =>	let val prAction = prAction showTerminal	    val action = LrTable.action table	    val goto = LrTable.goto table	    fun parseStep(args as			 (lexPair as (TOKEN (terminal, value as (_,leftPos,_)),				      lexer				      ),			  stack as (state,_) :: _,			  queue)) =	      let val nextAction = action (state,terminal)	          val _ = if DEBUG1 then prAction(stack,lexPair,nextAction)			  else ()	      in case nextAction		 of SHIFT s =>		  let val newStack = (s,value) :: stack		      val newLexPair = Stream.get lexer		      val (_,newQueue) =Fifo.get(Fifo.put((newStack,newLexPair),							    queue))		  in parseStep(newLexPair,(s,value)::stack,newQueue)		  end		 | REDUCE i =>		     (case saction(i,leftPos,stack,arg)		      of (nonterm,value,stack as (state,_) :: _) =>		          parseStep(lexPair,(goto(state,nonterm),value)::stack,				    queue)		       | _ => raise (ParseImpossible 197))		 | ERROR => parseStep(fixError args)		 | ACCEPT => 			(case stack			 of (_,(topvalue,_,_)) :: _ =>				let val (token,restLexer) = lexPair				in (topvalue,Stream.cons(token,restLexer))				end			  | _ => raise (ParseImpossible 202))	      end	    | parseStep _ = raise (ParseImpossible 204)	in parseStep	end    (*  distanceParse: parse until n tokens are shifted, or accept or	error are encountered.  Takes a table, showTerminal function, and	semantic action function.  Returns a parser which takes a lexPair	(lex result * lexer), a state stack, a queue, and a distance	(must be > 0) to parse.  The parser returns a new lex-value, a stack	with the nth token shifted on top, a queue, a distance, and action	option. *)    val distanceParse =      fn (table,showTerminal,saction,arg) =>	let val prAction = prAction showTerminal	    val action = LrTable.action table	    val goto = LrTable.goto table	    fun parseStep(lexPair,stack,queue,0) = (lexPair,stack,queue,0,NONE)	      | parseStep(lexPair as (TOKEN (terminal, value as (_,leftPos,_)),				      lexer				     ),			  stack as (state,_) :: _,			  queue,distance) =	      let val nextAction = action(state,terminal)	          val _ = if DEBUG1 then prAction(stack,lexPair,nextAction)			  else ()	      in case nextAction		 of SHIFT s =>		  let val newStack = (s,value) :: stack		      val newLexPair = Stream.get lexer		  in parseStep(newLexPair,(s,value)::stack,			       Fifo.put((newStack,newLexPair),queue),distance-1)		  end		 | REDUCE i =>		    (case saction(i,leftPos,stack,arg)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -