specification | programs |
grammar | javatokens.jj | programs |
JAVATOKENS SPECIFICATION
JavaTokens is a representation for a stream of Tokens for MacroJava.
A JavaTokens file may optionally end with "error" to indicate lexical error in the input.
specification | minijava.jj | programs |
grammar | miniIR.jj | programs |
MiniIR (Minijava Intermediate Representation) is an intermediate
representation for Minijava. It has simplified expressions, exposes the
runtime model (integer is of size 4, field and method references require
accessing of the address of the object, "new" operator is replaced with
system call HALLOCATE).
procedures:
If we have a procedurearrays:
ProcLabel [5] Stmt
then it takes five arguments, called (temp 0), (temp 1), ..., (temp 4). Other temporaries can be used without declaration and treated as locals. It executes the statement Stmt.
The index of an array is a multiple of 4, starting at 0.NOOP:
Does nothing.ERROR:
Terminates the program execution with an error message.CJUMP SimpleExp Label:
If SimpleExp evaluates to 1, then continue with the next statement, otherwise jump to Label.HSTORE Temp_1 IntegerLiteral Temp_2:
Temp_1 evaluates to an addressHLOAD Temp_1 Temp_2 IntegerLiteral:
IntegerLiteral is an offset
Temp_2 evaluates to the value that should be stored.
Temp_1 is the temporary into which a value should be loadedLT: is "<"
Temp_2 evaluates to an address
IntegerLiteral is an offset.
HALLOCATE SimpleExp:
SimpleExp evaluates to an integer, and that corresponding amount of heapspace is allocated, and the return address of that is returned as the result.
grammar | microIR.jj | programs |
microIR is a subset of miniIR. The grammar for microIR differs
from that of miniIR in two ways:
- StmtExp is not an Exp in microIR, and
- in many places, Exp is replaced with SimpleExp or Temp.
grammar | miniRA.jj | programs |
MiniRA is a MIPS-oriented language. It is a lot like microIR, but with the
following changes:
1) MiniRA supports C-style comments.
2) Labels now have global rather than local scope.
3) Instead of an unbounded number of temporaries with local scope,
MiniRA has 24 global machine registers with global scope. The
registers s0-s7 and t0-t9 can be allocated for general use.
Registers a0-a3 are reserved to pass arguments to a procedure call.
Register v0 is reserved to return a result from a procedure call,
and v0 and v1 are also used as temporary registers when values need
to be loaded from the stack.
s0-s7 are callee saved. t0-t9 are caller saved.
4) Values can be loaded from and stored to the stack with the ALOAD
and ASTORE instructions, where (SPILLEDARG i) denotes the ith value
on the stack, with the first value at (SPILLEDARG 0). For example:
ALOAD s3 (SPILLEDARG 1)loads the second value from the stack into register s3.
5) A procedure body is no longer a StmtExp but a StmtList. The return
value should be put in register v0.
6) Call is now a statement. As mentioned above, the "a" registers are
used to send arguments. If there are more than 4 arguments to the
call, you need to use the PASSARG stmt, which saves the extra args
to the stack. For whatever reason, PASSARG starts at 1, but
SPILLEDARG starts at 0, so in general an argument passed as
(PASSARG i) is accessed in the body of the procedure as (SPILLEDARG
i-1).
Here's an example. Consider a call to some procedure with label P and
arguments stored in registers t1, t2, t3, t4, and t5, and where the return
value should go in t6.
MOVE a0 t1 // first move 4 args to the "a" registers7) A procedure has three integers in its header, for example:
MOVE a1 t2
MOVE a2 t3
MOVE a3 t4
PASSARG 1 t5 // if there are more args save them to the stack.
// note that PASSARG is 1-based, not 0-based!
CALL P
MOVE t6 v0 // the return value is in v0
procA [5] [3] [4]The first integer has the same meaning as the integer in microIR, i.e. the number of args taken by the procedure.
The second integer is for the number of stack slots that the
procedure requires. This is the total number, including space for
arguments (if necessary), space for any spilled temps, and space for
any registers that have to be saved.
The third integer is the maximum arguments of a call in the body of
procA. i.e. if procA makes a call to procB that takes 3 args, to
procC that takes 2 args, and to procD that takes 4 args, then since
4 is the maximum args a call in the body of procA uses, this
integer is set to 4.
grammar | mips.jj | programs |
[ Taken from Yannis Smaragdakis] The MIPS registers of interest together with the conventions of their uses are:
Some examples from MIPS code:
To insert the value from $t0 to the top of the
stack:
sw $t0, 0($sp)
add $sp, $sp, -4
To add $t0, $t1 and produce the result in $t2:
add $t2, $t0, $t1
To multiply $t0, $t1 and produce the result in
$t2:
mult $t0, $t1
mflo $t2
To move from one point of the stack to another
(x = y where variable x has been assigned during register allocation
to the 3rd position from the top of the stack, and y has been
assigned the 6th position, counting backwards) under our convention
for v1:
lw $v1, -24($sp)
sw $v1, -12($sp)
To do a function call:
jal foo
To return from a function call:
jr $ra
To save a long integer (>32768) to $t0, you do:
li $t0, 1000000
The "MOVE t2 Fac_ComputeFac" of miniRA would
then become:
la $t2, Fac_ComputeFac
To translate the miniRA "CALL t0 " you will use
the pseudo-instruction jalr:
jalr $t0
You can figure out the rest for yourselves. A
very good reference is
here. A difference from the standard MIPS conventions of
register usage is that you can use $v1 for transfers from the stack.
Before a function call you should save all (live) $t0-$t9 variables
as well as $ra to
the stack. On the above site you will find some MIPS
pseudo-instructions that will be translated into simpler binary code
by the assembler. You can use these pseudo-instructions freely.
You can execute your MIPS code in SPIM. SPIM is
a MIPS emulator that offers some very simple system calls. You will
need calls for memory allocation, integer printing and character
printing.
To do a memory allocation of 40 bytes and get
the address of the allocated memory in $v0:
add $a0, $0, 40
add $v0,
$0, 9
syscall
To call the function that prints an integer
from $t4:
move $a0, $t4
add $v0, $0, 1
syscall
To call a function that prints a line feed:
add $a0, $0, 10
add $v0,
$0, 11
syscall
The following is an example of MIPS code that runs on SPIM. Given this C program, a possible translation to MIPS code is here. The same code with "dynamic" calls is here. The print_int and print_char functions are identical to those offered by SPIM through syscall. (The C program is given in this form to make its translation more understandable.)
Details about the MIPS assembly can be found at spim homepage.
SPIM Quick reference can be found here.
Quick details about MIPS integer instruction set can be here.
A Java based MIPS simulator can be found here.