-- demo\rosetta\Arithmetic_evaluation.exw sequence opstack = {} -- atom elements are literals, -- sequence elements are subexpressions -- on completion length(opstack) should be 1 object token constant op_p_p = 0 -- 1: expressions stored as op,p1,p2 -- p_op_p -- 0: expressions stored as p1,op,p2 -- p_p_op -- -1: expressions stored as p1,p2,op object op = 0 -- 0 if none, else "+", "-", "*", "/", "^", "%", or "u-" string s -- the expression being parsed integer ch integer sidx procedure err(string msg) printf(1,"%s\n%s^ %s\n\nPressEnter...",{s,repeat(' ',sidx-1),msg}) {} = wait_key() abort(0) end procedure procedure nxtch(object msg="eof") sidx += 1 if sidx>length(s) then if string(msg) then err(msg) end if ch = -1 else ch = s[sidx] end if end procedure procedure skipspaces() while find(ch," \t\r\n")!=0 do nxtch(0) end while end procedure procedure get_token() atom n, fraction integer dec skipspaces() if ch=-1 then token = "eof" return end if if ch>='0' and ch<='9' then n = ch-'0' while 1 do nxtch(0) if ch<'0' or ch>'9' then exit end if n = n*10+ch-'0' end while if ch='.' then dec = 1 fraction = 0 while 1 do nxtch(0) if ch<'0' or ch>'9' then exit end if fraction = fraction*10 + ch-'0' dec *= 10 end while n += fraction/dec end if -- if find(ch,"eE") then -- you get the idea -- end if token = n return end if if find(ch,"+-/*()^%")=0 then err("syntax error") end if token = s[sidx..sidx] nxtch(0) return end procedure procedure Match(string t) if token!=t then err(t&" expected") end if get_token() end procedure procedure PopFactor() object p2 = opstack[$] if op="u-" then if op_p_p=1 then -- op_p_p opstack[$] = {op,0,p2} elsif op_p_p=0 then -- p_op_p opstack[$] = {0,op,p2} else -- -1 -- p_p_op opstack[$] = {0,p2,op} end if else opstack = opstack[1..$-1] if op_p_p=1 then -- op_p_p opstack[$] = {op,opstack[$],p2} elsif op_p_p=0 then -- p_op_p opstack[$] = {opstack[$],op,p2} else -- -1 -- p_p_op opstack[$] = {opstack[$],p2,op} end if end if op = 0 end procedure procedure PushFactor(atom t) if op!=0 then PopFactor() end if opstack = append(opstack,t) end procedure procedure PushOp(string t) if op!=0 then PopFactor() end if op = t end procedure procedure Factor() if atom(token) then PushFactor(token) if ch!=-1 then get_token() end if elsif token="-" then get_token() -- Factor() Expr(3) -- makes "-3^2" yield -9 (ie -(3^2)) not 9 (ie (-3)^2). if op!=0 then PopFactor() end if if integer(opstack[$]) then opstack[$] = -opstack[$] else PushOp("u-") end if elsif token="(" then get_token() Expr(0) Match(")") elsif token="+" then -- (ignore) nxtch() Factor() else err("syntax error") end if end procedure constant {operators, precedence, associativity} = columnize({{"^",3,0}, {"%",2,1}, {"*",2,1}, {"/",2,1}, {"+",1,1}, {"-",1,1}, $}) procedure Expr(integer p) -- -- Parse an expression, using precedence climbing. -- -- p is the precedence level we should parse to, eg/ie -- 4: Factor only (may as well just call Factor) -- 3: "" and ^ -- 2: "" and *,/,% -- 1: "" and +,- -- 0: full expression (effectively the same as 1) -- obviously, parentheses override any setting of p. -- integer k, thisp Factor() while 1 do k = find(token,operators) -- *,/,+,- if k=0 then exit end if thisp = precedence[k] if thisp