Skip to content

Latest commit

 

History

History
355 lines (306 loc) · 12 KB

README.md

File metadata and controls

355 lines (306 loc) · 12 KB

CSE 231 PA2

Representation of values

Literals can be one of bool, int or None type.

  • For numbers we use int. In typescript we use number datatype and correspondingly i32 is used in WASM. Therefore, a number 32 would be represented as i32.const 32 in WASM.

  • For boolean values we use bool. In typescript we use boolean datatype. The value can be either True or False which is represented by true and false respectively in typescript. In WASM True is represented by integer constant 1 i.e., i32.const 1 and False is represented by integer constant 0 i.e., i32.const 0.

  • None value is a separate datatype in python. We use {tag:"none} to represent None in typescript and i32.const 0 in WASM.

Therefore, in WASM the represnetation for integer 0, boolean False and None is the same. In Typescript we use enum Type to distinguish them. The attribute a is used to set the type of any expression. For boolean values it is set to "bool". This is utilised in calling the appropriate print function as shown in the code snippet. The print_bool function displays False if the argument value is 0 else displays True.

Code snippet from ast.ts

export enum Type { int = "int", bool = "bool", none = "None" }

export type Literal<A> = { a?: A, tag: "num", value: number }
  | { a?: A, tag: "bool", value: boolean }
  | { a?: A, tag: "none" }

Code snippet for print

//compile.ts
...
case "builtin1":
    const argStmts = codeGenExpr(expr.arg, localEnv);
    var funName = expr.name;
    if (funName === 'print') {
        switch (expr.arg.a) {
        case "int":
            funName = 'print_num'
            break;
        case "bool":
            funName = 'print_bool'
            break;
        case "None":
            funName = 'print_none'
            break;
        }
    }
    return argStmts.concat([`(call $${funName})`]);
...

//webstart.ts
...
print_bool: (arg: any) => {
    if (arg === 0) { display("False"); }
    else { display("True"); }
    return 0;
}
...

Example program

x:int = 5
def func(y:int):
    z:int = 4
    if y < z:
        print(z)
    else:
        print(y)
func(x)

ast.ts

export type Program<A> = { 
    a?: A, varinits: VarDef<A>[], 
    fundefs: FunDef<A>[], 
    stmts: Stmt<A>[] 
}

export type VarDef<A> = {
  a?: A, name: string,
  type: Type,
  init: Literal<A>
}

export type FunDef<A> = {
  a?: A, name: string,
  params: TypedVar<A>[],
  ret: Type,
  inits: VarDef<A>[],
  body: Stmt<A>[]
}

export type TypedVar<A> = { a?: A, name: string, type: Type }

The entire program is a set of global variables, function definitions and statements. The global variables are represented by the key varinits which is an array of type VarDef. The function parameter y is represented by params key which is an array of TypedVar in FunDef whereas variables declared within the scope of the function is represnted by inits which is an array of VarDef. Therefore, in the above example program, x is stored in the varinits array which is part of Program and thus available in global scope. The paramter y is stored in params and z is stored in inits which are both a part of FunDef used to represent the function func.

typecheck.ts

export type TypeEnv = {
  vars: Map<string, Type>,
  funcs: Map<string, [Type[], Type]>,
  retType: Type //stores the return type
}

function typeCheckVarInits(inits: VarDef<null>[],
 env: TypeEnv): VarDef<Type>[] {
  const typedInits: VarDef<Type>[] = [];
  inits.forEach(init => {
    const initType = typeCheckLiteral(init.init);
    if (initType.a !== init.type) {
      throw new TypeError(`Expected type \`${init.type}\`;
       got type \`${initType.a}\``);

    }
    env.vars.set(init.name, init.type);
    typedInits.push({ ...init, a: init.type, init: initType });
  });

  return typedInits;
}
function typeCheckParams(params: TypedVar<null>[]): TypedVar<Type>[] {
  return params.map(param => {
    return { ...param, a: param.type }
  });
}
function typeCheckFunDef(fundef: FunDef<null>,
 env: TypeEnv): FunDef<Type> {
  
  // add params to env
  const localEnv: TypeEnv = { ...env,
   vars: new Map(env.vars),
    funcs: new Map(env.funcs) };
  fundef.params.forEach(param => {
    localEnv.vars.set(param.name, param.type);
  });

  const typedParams = typeCheckParams(fundef.params);

  // check inits and add to env
  const typedInits = typeCheckVarInits(fundef.inits, localEnv);

  localEnv.retType = fundef.ret;
  // check body
  const typedStmts = typeCheckStmts(fundef.body, localEnv)

  return { ...fundef, params: typedParams, inits: typedInits, body: typedStmts }
}
...
case "call":
    if (!env.funcs.has(expr.name)) {
    throw new ReferenceError(`Not a Function \`${expr.name}\``);
    }
    const typedArgs = expr.args.map((arg) => typeCheckExpr(arg, env));
    const [actualArgs, retType] = env.funcs.get(expr.name)
    if (actualArgs.length !== typedArgs.length)
    throw new Error(`Expected ${actualArgs.length} arguments for 
    ${expr.name} got ${typedArgs.length}`)
    for (let index = 0; index < typedArgs.length; index++) {
    if (typedArgs[index].a !== actualArgs[index])
        throw new TypeError(`Expected type \`${actualArgs[index]}\`;
         got type \`${typedArgs[index].a}\` for parameter ${index}`);
    }
    return { ...expr, args: typedArgs, a: retType };
...

In typeCheck.ts, all these variables are typechecked. For x:int=5 it is ensured that x is initialised to only literals of type int by typeCheckVarInits as shown in the code snippet. Similarly it is checked for z:int=4. For paramters in the function definition we annotate the type. During the function call the types of the passed arguments are checked against the types declared in the function defintions which is stored in env which is of type TypeEnv.

In compile.ts we generate code for all the global variables separately as shown in the snippet. For function definitions we first generate code for paramters (which are all i32 since all types are essentially represented by i32 as described in Section 1). Then code is genertaed for variables initialised in function scope using local.

compile.ts

...
var globals:string[] = [];
  typedAst.varinits.forEach((v) => {
    globals.push(`(global $${v.name} (mut i32) ${resolveLiteral(v.init)})`);
  });
...

function codeGenFun(fundef: FunDef<Type>, localEnv:TypeEnv): Array<string> {
  // Construct the environment for the function body
  const funEnv:TypeEnv = { vars: new Map(), funcs: new Map, retType: "None" };
  // Construct the code for params and variable declarations in the body
  fundef.inits.forEach(init => {
    funEnv.vars.set(init.name,init.type);
  });
  fundef.params.forEach(param => {
    funEnv.vars.set(param.name,param.type);
  });
  // Construct the code for params and variable declarations in the body
  const params = fundef.params.map(p => `(param $${p.name} i32)`).join(" ");
  const varDecls = fundef.inits.map(v => 
    `(local $${v.name} i32)\n${resolveLiteral(v.init)}\n(local.set $${v.name})`).join("\n");

  const stmts = fundef.body.map(s => codeGenStmt(s,funEnv, false)).flat();
  const stmtsBody = stmts.join("\n");
  return [`(func $${fundef.name} ${params} (result i32)
    (local $$last i32)
    ${varDecls}
    ${stmtsBody}
    (i32.const 0))`];
}

Infinite loop

x:int = 9
while x<10:
    print(x)

When this infinite loop is run on a web browser. The print satement is executed infinitely and the output can be seen on the console logs. However, nothing is displayed on the output and the webpage becomes unresponsive and ultimately hangs.

Screenshots

  1. A program that reports a type error for adding a number and a boolean where one operand is a call expression and the other operand is a variable.

    x:int = 10
    def func(x:int)->bool:
        return x>5
    x + func(x)

  2. A program that has a type error in a conditional position, where that position is a non-global identifier.

    p : bool = True
    def f(q : bool) -> int:
        if q < 25:
            return 99
        else:
            return 500
    print(f(p))

  3. A program that with a loop that has multiple iterations, and calls a function on each iteration

    x:int = 5
    def my_print(x:int):
        print(x)
    while x>0:
        my_print(x)
        x = x-1    

  4. A program that returns from the body of a loop, and not on the first iteration of the loop

    def func(x:int)->int:
        while x > 0:
            x = x-2
            if x < 0:
                return -x
        return x
    print(func(4))
    func(5)

  5. Printing an integer and a boolean

    x:int = 4
    y:bool = True
    print(x)
    print(y)    

  6. A recursive function that terminates (e.g. no stack overflow)

    def sum(n : int) -> int:
        if n < 1: return 0
        else: return sum(n - 1) + n
    sum(4)

  7. Two mutually-recursive functions that terminate.

    def is_even(n:int) -> bool:
        if (n == 0):
            return True
        else:
            return is_odd(n - 1)
    def is_odd(n:int) -> bool:
        if (n == 0):
            return False
        else:
            return is_even(n - 1)
    print(is_even(4))
    print(is_odd(4))
    print(is_even(3))
    print(is_odd(3))

Type Errors

In (1) TypeError occurs as we try to add a bool and an int which is not supported. In case of Binary operators for operators +,-,*,//,%,<,<=,>,>=, it is ensured that both left and right hand operators are int. For the operators ==,!= it is ensured that left and right are either both int or both bool. For is, it is ensured that both left and right hand operators are None. It can also be noted that type of evaluated expression where the operators are +,-,*,//,% is int and for the operators <,<=,>,>=,==,!=,is is bool.

Code Snippet:

function typeCheckExpr(expr: Expr<null>, env: TypeEnv): Expr<Type> {
  switch (expr.tag) {
    case "binexpr":
      const left = typeCheckExpr(expr.left, env);
      var right = typeCheckExpr(expr.right, env);
      switch (expr.op) {
        case BinOp.Add:
        case BinOp.Sub:
        case BinOp.Mul:
        case BinOp.Div:
        case BinOp.Mod:
          if (left.a !== "int" || right.a !== "int") {
            throw new TypeError(`Cannot apply operator \`${expr.op}
            \` on types \`${left.a}\` and \`${right.a}\``)
          }
          return { ...expr, left, right, a: "int" }
        case BinOp.Lesser:
        case BinOp.LessEq:
        case BinOp.GreatEq:
        case BinOp.Greater:
          if (left.a !== "int" || right.a !== "int") {
            throw new TypeError(`Cannot apply operator \`${expr.op}
            \` on types \`${left.a}\` and \`${right.a}\``)
          }
          return { ...expr, left, right, a: "bool" }
        case BinOp.Equals:
        case BinOp.NotEquals:
          if (left.a !== right.a || right.a === "None") {
            throw new TypeError(`Cannot apply operator \`${expr.op}
            \` on types \`${left.a}\` and \`${right.a}\``)
          }
          return { ...expr, left, right, a: "bool" }
        case BinOp.Is:
          if (left.a !== "None" || right.a !== "None") {
            throw new TypeError(`Cannot apply operator \`${expr.op}
            \` on types \`${left.a}\` and \`${right.a}\``)
          }
          return { ...expr, left, right, a: "bool" }
      }
      break;
      ...
  }

Resources

  • TA Office hours was very helpful.
  • lecture 4 code.
  • WASM instruction guides.