Homework 5: MiniScheme Beginnings

Due: Friday, October 18 at 23:59
First Commit: Monday, October 14 at 23:59

This assignment helps you set the foundation for the MiniScheme project. Specifically, you’ll start by implementing environments and then continue to implementing to MiniScheme parts A & B.

This is the first of three homework assignments that constitute the MiniScheme project.

Your implementations will be in several files. The start of each file should be

#lang racket
; Your name(s) here.

Preliminaries

Click on the assignment link. You should make a new team as normal. However, you cannot change teams for subsequent MiniScheme Homeworks, so your team name and choice of partner should reflect this. I might recommend a variant of “molly-elinor-minischeme” as a team name.

As a reminder, if you’re working with a partner, one partner should create a new team. The second partner should click the link and choose the appropriate team. It’s extremely helpful for the graders if you include your name and your partner’s name in the team name. (Please don’t choose the wrong team, there’s a maximum of two people and if you join the wrong one, you’ll prevent the correct person from joining.)

Once you have accepted the assignment and created/joined a team, you can clone the repository on your computer by following the instruction and begin working. But before you do, read the entire assignment and be sure to check out the expected coding style, as posted on Ed.

Submission

To submit your homework, you must commit and push to GitHub before the deadline.

Your repository should contain the following files

It may also a .gitignore file which tells Git to ignore files matching patterns in your working directory.

Any additional files you have added to your repository should be removed from the main branch. (You’re free to make other branches, if you desire, but make sure main contains the version of the code you want graded.)

Make sure you put your name (and your partner’s name if you’re working with one) as a comment at the top of each file.

MiniScheme Environments

As a reminder, we will be building an interpreter over the next few weeks for a subset of the Scheme language. The first thing we need to do is develop a way to hold bindings and their values to execute code. In other words: we need environments!

Overview

We need to create an environment to hold the data for the expressions we will interpret. The scoping rules for Scheme determine the structure of this environment. Consider the following three examples. First

(let ([x 2] [y 3])
  (+ x y))

This has an environment with bindings for x and y created in a let-block. These bindings are used in the body of the let.

Next, consider the following. This has a let-block that creates a lambda expression, which is called in the body of the let:

(let ([f (lambda (x) (+ x 4))])
  (f 5))

When this is evaluated we want to bind x to the value of the argument, 5, and then evaluate the body of f using that binding.

Finally, we combine these. At the outer level in the following expression we have a let-block with bindings for x and y. The body is another, nested, let, which binds a lambda expression with a parameter x. The body of the interior let is a call to the function.

(let ([x 2] [y 3])
  (let ([f (lambda(x) (+ x y))])
    (f 5)))

When we evaluate this we first make an environment with bindings of x and y to 2 and 3 respectively, then we use this to evaluate the inner let-expression. In that expression we make a binding of f to the value of the lambda-expression (a closure), and then we call f with argument 5. This requires us to evaluate the body of f with x bound to 5. The body of f does not have a binding for y, so we look it up in the outer environment and see that its value is 3. Finally, we evaluates (+ x y) with x bound to 5 and y bound to 3, yielding 8 for the value of the full expression.

Environments are extended in two ways. Let expressions have bindings that extend the current environment; the body of the let is evaluated in the extended environment. Lambda-expressions do not extend the environment; they evaluate to closures that store the current environment from the location where the (lambda (...) ...) is evaluated.

When the closure that results from a lambda-expression is called, the closure’s environment is extended with the parameters from the lambda-expression being bound to the values of the arguments.

We will define the environment as an association list, where symbols are associated with values. There are two ways we might do this. In the first example above, where x is bound to 2 and y to 3, we might use the list '((x 2) (y 3)), or we might use '((x y) (2 3)). The former structure is closer to the way the bindings appear in let-expressions; the latter is closer to the components of a call. The former structure might appear simpler, but the latter is actually easier to code and we will go with that.

Scheme, and many other languages you have and likely will learn, employs lexical scoping. When we want to resolve the binding for a free variable, we look first in the current scope, then in the surrounding scope, then in the scope that surrounds that, until the variable is found or we reach the outermost scope. To implement this our environments will be structured as a “list” (hint: what should it actually be?) with four elements: the symbol 'env, a list of symbols, a list of values, and the previous environment. The top-most environment does not have a previous-environment (by definition) so we’ll use null to represent the empty environment.

Thus, the environment for the expression

(let ([x 2] [y 3])
  (let ([z 4] [x 5])
    (+ x (+ y z))))

will be something like

'(env (z x) (4 5)
      (env (x y) (2 3)
           ()))

When we resolve the bindings for x, y and z to evaluate (+ x (+ y z)) we find the binding 5 for x (there are two bindings for x, but the one we want is is the first one we come to), and of course we find 4 for z and 3 for y. This leads to the correct value, 12 for the expression.

Similarly, in the expression

(let ([x 2] [y 3])
  (let ([f (lambda(x) (+ x y))])
    (f 5)))

we evaluate the call (f 5) by evaluating the body of f in an environment that first has x bound to 5, and then has the environment surrounding the definition of f.

You will see in later parts how this environment is created. At present we need to create the tools that will allow this.

Part 1: The environment data type

The two most important features of an environment are that we need to be able to look up symbols to get the values they are bound to, and we need to be able to extend an environment with new bindings to get a new environment. We’ll define an environment as either the empty environment, with no bindings, or an extended environment with a list of symbols, a corresponding list of the values those symbols are bound to, and a previous environment that is being extended. Here is the definition of the empty environment.

; The empty environment is null.
(define empty-env null)
(define empty-env? null?)

How will we build the environment? Well, we want a constructor such that the following works:

(env '(x y) '(1 2) empty-env)

Recognizer

(env? x)

And accessors.

(env-syms e)
(env-vals e)
(env-previous e)

Now, define the new data type in env.rkt. Use built-in Racket functionality to define this data type: do NOT implement the constructor, recognizers, or accessors yourself!

Note that env is the procedure you will use every time you need to extend an environment when evaluating a let-expression or a function call. For example, when evaluating the expression (let ([x 1] [y 2]) ...) we might use the following.

(define env-a
  (env '(x y) '(1 2) empty-env))

We can then further extend the environment as follows:

(define env-b
  (env '(x z) '(5 7) env-a))

Part 2: Implementing the environment lookup

After Part 1 above, env.rkt contains code which describes the environment data type. Part 2 asks you to implement env-lookup (35 points) and write thoughtful related tests (see details below) (15 points).

Add to env.rkt the procedure (env-lookup environment symbol), which takes an environment and a symbol and returns the first binding for that symbol in the environment. For example, with environments env-a and env-b defined as

(define env-a
  (env '(x y) '(1 2) empty-env))
(define env-b
  (env '(x z) '(5 7) env-a))

we should have the following behavior:

If env-lookup does not find a binding for the symbol you should throw an error as below:

(error 'env-lookup "No binding for ~s" symbol)

Make sure you (provide env-lookup) at the top of env.rkt so that modules you’ll write in future parts (and in the tests described below) can use that procedure.

The file env-tests.rkt contains a test-env environment that maps x to 1 and y to 2. In addition it defines an env-tests test suite that tests the basic behavior of the environment data type. Extend this test suite with additional tests for env-lookup. You will probably want to define new extended environments for your tests.

In particular, be sure to test at least the following situations:

All testing for MiniScheme runs through tests.rkt. Run this file to call the tests you wrote in env-tests.rkt.

Now that you have a working environment, it would be a great time to commit your code!

Overview of MiniScheme A & B

We will start with a very basic version of our MiniScheme language, version A, and gradually add language features. As we do so, we will update our parser and interpreter to implement the new language features.

As mentioned in lecture, the parser and interpreter for MiniScheme will reside in files parse.rkt and interp.rkt, respectively. After you complete each of the remaining parts, you should be sure to commit your code. That way, you can always retrieve an earlier version if you need to.

Here’s what you’ll have in each file.

As you go, think about where to put new definitions. Racket has trouble with circular requirements (such as parse.rkt requiring interp.rkt and interp.rkt requiring parse.rkt). You probably want your parse.rkt and env.rkt files to not require any other module, and your interp.rkt file to require both parse.rkt and env.rkt.

In addition to those files, we’ll have some testing files.

Start by creating files parse.rkt and interp.rkt. Make sure they include the #lang line and your name(s).

MiniScheme A: Numbers

Version A of MiniScheme is given by the following grammar.

EXP → number                      parse into lit-exp

Parsing

Our parse procedure will take an input expression and return a parse tree for an EXP. The only expressions in MiniScheme A are numbers.

In parse.rkt we need a data type to hold EXP nodes that represent numbers. We’ll call this data type lit-exp. You need to implement it so that it contains the numerical value being represented, has a constructor, recognizer, and getter (a transparent struct is recommended!).

In the instructions I will use the names lit-exp (constructor), lit-exp? (recognizer), and lit-exp-num (getter), although you can call them whatever you’d like.

The parse function (parse input) should create a lit-exp when it sees a number (and throw an error otherwise).

Write the parse function and and the code that implements the lit-exp data type, in parse.rkt. Make this file into a module by adding the appropriate provide lines to make the parse procedure and the procedures for lit-exp (or whatever you call it) available to other modules.

In parse-tests.rkt, add tests for parse. In particular, you should test that (parse 5) returns something that lit-exp? evaluates to true. The test-pred procedure, as shown below, will likely be helpful.

(test-pred "Literal"
           lit-exp?
           (parse 5))

Also test that when you parse a number, you can extract the number from the resultant parse tree using lit-exp-num.

Evaluating

For the interpreter, you know that Scheme evaluates all integers as themselves. Therefore, eval-exp is simple. Write (eval-exp tree e) to evaluate a lit-exp and throw an error otherwise. Save this code as interp.rkt. Make sure you require parse.rkt and (for the next step) env.rkt. Similarly, make sure you provide eval-exp for use by other modules.

If you run interp.rkt in DrRacket to load the interpreter and parser into memory, you can type the following:

> (define T (parse '23))
> (eval-exp T empty-env)

and it should print the value 23.

It quickly becomes tedious to always invoke your interpreter by specifically calling eval-exp after calling the parser on the quoted expression. It would be nice if we could write a read-eval-print loop for MiniScheme. This is precisely what minischeme.rkt does.

Running minischeme.rkt in DrRacket will give you an input box that allows you to type expressions and get back their value as determined by your parse and interp modules. For example, if you enter the MiniScheme expression 23 this evaluates it and prints its value 23.

Running into issues? The read-eval-print procedures assumes that your parse procedure is named parse that your evaluator is called eval-exp that takes as arguments a parse tree and an environment, in that order; and an initial environment named init-env. Please keep these names consistent so that testing & minischeme.rkt works throughout the project!

Now inside interp.rkt, define and provide an init-env. For the init-env, use your env constructor to create an environment mapping x to 23 and y to 42. These will be helpful in future parts, so its a good idea to define them now.

The last thing we need to do for MiniScheme A is to write some tests. In interp-tests.rkt, write a test like

(test-equal? "Number"
             (eval-exp (lit-exp 5) empty-env)
             5)

Note that we’re explicitly passing the parse tree (lit-exp 5) rather than calling (parse 5). This lets you separately test parsing from evaluating. In subsequent parts, you’ll be modifying parse. You don’t want a bug in parse to show up as a bug in eval-exp. By not using parse in your interpreter tests, you can keep the two separate.

Congratulations, at this point you should have a working interpreter for MiniScheme A! Make sure you commit your work.

MiniScheme B: Variables & Values

Let’s extend MiniScheme A by adding symbols. So, the grammar for MiniScheme B is:

EXP → number                      parse into lit-exp
    | symbol                      parse into var-exp

The parser is a simple modification to parse. You need to add a line to (parse input) to handle the case where (symbol? input) is #t. You will need a new data type to handle this (I call it var-exp).

To evaluate a variable expression, MiniScheme B needs to be able to look up references. We evaluate a var-exp tree node in a given environment by calling lookup in that environment on the var-exp-symbol. Since we asked you to include bindings for symbols x and y in the initial environment in Part 4, you should be able to evaluate the MiniScheme expressions x or y to get their values. Any other symbol at this point should give you an error message.

Make sure to add tests to both parse-tests.rkt and interp-tests.rkt to address MiniScheme B.

Common Questions

Finishing up

Make sure you commit and push. And make sure your name is in all of the files you submit. Your friendly graders thank you!