Update pl0 for ebnf changes, and add more tests
parent
eef6ae7827
commit
208c88c449
|
@ -1,7 +1,7 @@
|
||||||
! Copyright (C) 2007 Chris Double.
|
! Copyright (C) 2007 Chris Double.
|
||||||
! See http://factorcode.org/license.txt for BSD license.
|
! See http://factorcode.org/license.txt for BSD license.
|
||||||
!
|
!
|
||||||
USING: kernel tools.test peg peg.pl0 ;
|
USING: kernel tools.test peg peg.pl0 multiline sequences ;
|
||||||
IN: peg.pl0.tests
|
IN: peg.pl0.tests
|
||||||
|
|
||||||
{ "abc" } [
|
{ "abc" } [
|
||||||
|
@ -11,3 +11,89 @@ IN: peg.pl0.tests
|
||||||
{ 55 } [
|
{ 55 } [
|
||||||
"55abc" number parse parse-result-ast
|
"55abc" number parse parse-result-ast
|
||||||
] unit-test
|
] unit-test
|
||||||
|
|
||||||
|
{ t } [
|
||||||
|
<"
|
||||||
|
VAR x, squ;
|
||||||
|
|
||||||
|
PROCEDURE square;
|
||||||
|
BEGIN
|
||||||
|
squ := x * x
|
||||||
|
END;
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
x := 1;
|
||||||
|
WHILE x <= 10 DO
|
||||||
|
BEGIN
|
||||||
|
CALL square;
|
||||||
|
x := x + 1;
|
||||||
|
END
|
||||||
|
END.
|
||||||
|
"> program parse parse-result-remaining empty?
|
||||||
|
] unit-test
|
||||||
|
|
||||||
|
{ f } [
|
||||||
|
<"
|
||||||
|
CONST
|
||||||
|
m = 7,
|
||||||
|
n = 85;
|
||||||
|
|
||||||
|
VAR
|
||||||
|
x, y, z, q, r;
|
||||||
|
|
||||||
|
PROCEDURE multiply;
|
||||||
|
VAR a, b;
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
a := x;
|
||||||
|
b := y;
|
||||||
|
z := 0;
|
||||||
|
WHILE b > 0 DO BEGIN
|
||||||
|
IF ODD b THEN z := z + a;
|
||||||
|
a := 2 * a;
|
||||||
|
b := b / 2;
|
||||||
|
END
|
||||||
|
END;
|
||||||
|
|
||||||
|
PROCEDURE divide;
|
||||||
|
VAR w;
|
||||||
|
BEGIN
|
||||||
|
r := x;
|
||||||
|
q := 0;
|
||||||
|
w := y;
|
||||||
|
WHILE w <= r DO w := 2 * w;
|
||||||
|
WHILE w > y DO BEGIN
|
||||||
|
q := 2 * q;
|
||||||
|
w := w / 2;
|
||||||
|
IF w <= r THEN BEGIN
|
||||||
|
r := r - w;
|
||||||
|
q := q + 1
|
||||||
|
END
|
||||||
|
END
|
||||||
|
END;
|
||||||
|
|
||||||
|
PROCEDURE gcd;
|
||||||
|
VAR f, g;
|
||||||
|
BEGIN
|
||||||
|
f := x;
|
||||||
|
g := y;
|
||||||
|
WHILE f # g DO BEGIN
|
||||||
|
IF f < g THEN g := g - f;
|
||||||
|
IF g < f THEN f := f - g;
|
||||||
|
END;
|
||||||
|
z := f
|
||||||
|
END;
|
||||||
|
|
||||||
|
BEGIN
|
||||||
|
x := m;
|
||||||
|
y := n;
|
||||||
|
CALL multiply;
|
||||||
|
x := 25;
|
||||||
|
y := 3;
|
||||||
|
CALL divide;
|
||||||
|
x := 84;
|
||||||
|
y := 36;
|
||||||
|
CALL gcd;
|
||||||
|
END.
|
||||||
|
"> program parse parse-result-remaining empty?
|
||||||
|
] unit-test
|
|
@ -1,30 +1,31 @@
|
||||||
! Copyright (C) 2007 Chris Double.
|
! Copyright (C) 2007 Chris Double.
|
||||||
! See http://factorcode.org/license.txt for BSD license.
|
! See http://factorcode.org/license.txt for BSD license.
|
||||||
USING: kernel arrays strings math.parser sequences
|
USING: kernel arrays strings math.parser sequences
|
||||||
peg peg.ebnf peg.parsers memoize ;
|
peg peg.ebnf peg.parsers memoize namespaces ;
|
||||||
IN: peg.pl0
|
IN: peg.pl0
|
||||||
|
|
||||||
#! Grammar for PL/0 based on http://en.wikipedia.org/wiki/PL/0
|
#! Grammar for PL/0 based on http://en.wikipedia.org/wiki/PL/0
|
||||||
MEMO: ident ( -- parser )
|
MEMO: ident ( -- parser )
|
||||||
CHAR: a CHAR: z range
|
[
|
||||||
CHAR: A CHAR: Z range 2array choice repeat1
|
CHAR: a CHAR: z range ,
|
||||||
[ >string ] action ;
|
CHAR: A CHAR: Z range ,
|
||||||
|
] choice* repeat1 [ >string ] action ;
|
||||||
|
|
||||||
MEMO: number ( -- parser )
|
MEMO: number ( -- parser )
|
||||||
CHAR: 0 CHAR: 9 range repeat1 [ string>number ] action ;
|
CHAR: 0 CHAR: 9 range repeat1 [ string>number ] action ;
|
||||||
|
|
||||||
<EBNF
|
<EBNF
|
||||||
program = block '.' .
|
program = block "."
|
||||||
block = [ 'const' ident '=' number { ',' ident '=' number } ';' ]
|
block = [ "CONST" ident "=" number { "," ident "=" number } ";" ]
|
||||||
[ 'var' ident { ',' ident } ';' ]
|
[ "VAR" ident { "," ident } ";" ]
|
||||||
{ 'procedure' ident ';' [ block ';' ] } statement .
|
{ "PROCEDURE" ident ";" [ block ";" ] } statement
|
||||||
statement = [ ident ':=' expression | 'call' ident |
|
statement = [ ident ":=" expression | "CALL" ident |
|
||||||
'begin' statement {';' statement } 'end' |
|
"BEGIN" statement {";" statement } "END" |
|
||||||
'if' condition 'then' statement |
|
"IF" condition "THEN" statement |
|
||||||
'while' condition 'do' statement ] .
|
"WHILE" condition "DO" statement ]
|
||||||
condition = 'odd' expression |
|
condition = "ODD" expression |
|
||||||
expression ('=' | '#' | '<=' | '<' | '>=' | '>') expression .
|
expression ("=" | "#" | "<=" | "<" | ">=" | ">") expression
|
||||||
expression = ['+' | '-'] term {('+' | '-') term } .
|
expression = ["+" | "-"] term {("+" | "-") term }
|
||||||
term = factor {('*' | '/') factor } .
|
term = factor {("*" | "/") factor }
|
||||||
factor = ident | number | '(' expression ')'
|
factor = ident | number | "(" expression ")"
|
||||||
EBNF>
|
EBNF>
|
||||||
|
|
Loading…
Reference in New Issue