Specifications.Assign6
This assignment is due on Thursday 3/20 by 8:00PM. You should put all of your solutions in assign6/lib/assign6.ml
. See the file test/test_assign6.ml
for example behavior of each function.
The purpose of this assignment is to give you a taste of the mini-projects by building an S-expressions calculator. Here is a simple example of an expression in this language.
(?
(< (+ 2 3) (+ 3 3))
0
(+ 1 1))
This expression has type and evaluates to .
In more familiar OCaml-like syntax, this would look like:
if 2 + 3 < 3 + 3 then 0 else 1 + 1
In the mini-projects, you'll be given three things: syntax, typing rules, and semantics. It will then be your task to implement an interpreter. Note that several of the parts below require little to no new code.
Here's the syntax of our toy language.
Next, the typing rules. Expressions can either be integers or Boolean values. These should feel familiar. We'll elide names because we won't be writing derivations for this system. Note that there is no need for a context because there are no local variables!
Finally, the semantics. These should also feel familiar.
In rough terms, lexing (or tokenizing) is the process of converting a string into a list of tokens. Tokens are the abstract units of a programming language. We convert our input program (a string) into a list of tokens so that we don't need to deal with low-level concerns like whitespace when parsing. You're given an implementation of a lexer.
val lex : string -> tok list option
lex s
is Some tks
where tks
is the list of tokens representing s
and None
otherwise. It's a variant of tokenize
from Lab 4, which you can check for more details.
We'll use an ADT to represent expressions in our language, similar to previous assignments.
val parse : string -> expr option
Implement the function parse
so that parse str
is
Some e
if str
represents a well-formed program;None
otherwise.You've already built a parser for S-expressions in Lab 4. Repurpose that code to work for this new expr
ADT (as opposed to nonempty trees) and compose it with the given function lex
.
Implement the function type_of
so that type_of e
is
Some t
where t
is the type of e
if e
is well-typed;None
otherwise.An expression is well-typed if if is derivable according to the above typing rules for some type .
Implement the function eval
so that eval e
is v
is the value of e
given that e
is well-typed. The behavior of the implementation is undefined if e
is not well-typed (i.e., if type_of e
is None
).
Once you've finished the above problems, you can run your interpreter on an actual program. There are several example programs in the directory examples
. If you've done everything correctly, you should be able to run the command
dune exec assign6 examples/file_name
(replace file_name
with the name of one of the files in that directory) which will print out the value of the expression in the file. You can even write your own programs if you'd like.
To be clear, there are no tasks for the assignment in this section, this is just for fun. That said, please look through the file bin/main.ml
. This file chains together the lexer, parser, type checker, and evaluator. In the future, we will be asking you to write this part.
<s> ::= A <s> B
| <a>
<a> ::= A <a>
| B <b>
<b> ::= B B <b>
| B B
Determine a sentence with fewer than 8 symbols which has two distinct leftmost derivations in the above grammar. Also write down these two leftmost derivations.
Write down a grammar over the terminal symbols A
and B
such that every sentence recognized by the grammar is any number of A
s, followed by a sequence of at least 2 B
s. For example AAAABBB
, BBB
, and AABB
should be sentences recognized by the grammar, whereas AAAB
, AB
, and A
should not. You may introduce any nonterminal symbols that you want.