1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 |
(* Modified in part from CSMC330 at UMD *) (* Parser for E -> A + E | A A -> 0|1|2|3|4|5|6|7|8|9 *) #load "str.cma" (* Only for loading into utop, otherwise link with str.cma using ocamlc *) open Str;; (***** Scanner *****) type token = Tok_Num of char | Tok_Sum | Tok_END let re_num = Str.regexp "[0-9]" let re_add = Str.regexp "+" exception IllegalExpression of string let tokenize str = let rec tok pos s = if pos >= String.length s then [Tok_END] else if (Str.string_match re_num s pos) then let token = Str.matched_string s in (Tok_Num token.[0])::(tok (pos+1) s) else if (Str.string_match re_add s pos) then Tok_Sum::(tok (pos+1) s) else raise (IllegalExpression "tokenize") in tok 0 str (***** Parser *****) type exp = Num of int | Sum of exp * exp let rec a_to_str a = match a with Num n -> string_of_int n | Sum (a1,a2) -> "(" ^ (a_to_str a1) ^ " + " ^ (a_to_str a2) ^ ")" ;; let tok_list = ref [] exception ParseError of string let lookahead () = match !tok_list with [] -> raise (ParseError "no tokens") | (h::t) -> h let match_tok a = match !tok_list with (* checks lookahead; advances on match *) | (h::t) when a = h -> tok_list := t | _ -> raise (ParseError "bad match") let rec parse_E () = let a1 = parse_A () in let t = lookahead () in match t with Tok_Sum -> match_tok Tok_Sum; let a2 = parse_E () in Sum(a1,a2) | _ -> a1 (* E -> A *) and parse_A () = let t = lookahead () in match t with Tok_Num c -> let _= match_tok (Tok_Num c) in Num (int_of_string (Char.escaped c)) | _ -> raise (ParseError "parse_A") let parse str = tok_list := (tokenize str); let exp = parse_E () in if !tok_list <> [Tok_END] then raise (ParseError "parse_E") else exp ;; (***** Interpreter ****) let rec eval a = match a with Num n -> n | Sum (a1,a2) -> (eval a1) + (eval a2) ;; let eval_str str = print_string str; print_string "\n"; let e = parse str in print_string "AST produced = " ; print_endline (a_to_str e) ; let v = eval e in print_string "Value of AST = " ; print_int v ; print_endline ""; v ;; eval_str "1+2+3+4+5" ;; |