34 Section Handout
34 Section Handout
34 Section Handout
This problem asks you to add this extension to your Decaf compiler. You must support
one-dimensional arrays whose element types can be any legal Decaf type, except stack-
allocated arrays themselves. That is, it is legal to declare
int[][n] legitimateArray;,
(Remember that Decaf uses the notation int[] a, unlike C’s int a[].)
The storage for the array elements should be allocated on the stack. In this way, the
storage is automatically freed when the function returns. Array bounds checking should
be performed as for ordinary, heap-allocated arrays.
is illegal.
As in the project, we’ll provide the necessary TAC instructions for your intermediate code
generator. We define a new TAC instruction AllocA that can be emitted to allocate space
on the stack for a stack-allocated array:
In order for this to work, you’d need to also implement one more method in the MIPS
g) Given the MIPS’s procedure calling convention, explain where in the stack frame
you would allocate space for the elements of stack-allocated arrays.
h) Assuming this new stack frame layout, will this extension affect how you assign
offsets to the locations of your local variables and temporaries?
i) Implement Mips::EmitAllocA. Assume that the szLoc argument refers to a location
in which a value of 4*n + 4 is stored, where n > 0 is the number of elements in the
array. In other words, the required size of the array has already been computed
and checked. Assume further that AllocA shall only allocate the memory for the
new array, it shall not set up the array header. In short, AllocA replaces the LCall
_Alloc sequence found in the TAC sequence emitted to allocate ordinary, heap-
based arrays.
3
c) This change will introduce new conflicts, because the grammar will have to
distinguish between a variable declaration A[n] b; and a possible first statement
starting with A[n]… in the following block of statements. Your parser would have
explore both options in parallel without reducing a nonterminal that only applies to
one or the other option. This is practically impossible the way your grammar is
written. Hence, the change will result in a reduce-reduce conflict for T_Identifier
using rules such as LValue T_Identifier and Type T_Identifier. In fact, the conflict
cannot be easily resolved, because it would require you to look ahead until after the
closing ], which can be an arbitrary number of tokens. This problem could only be
fixed by using the same set of nonterminals for variable declarations and
expressions (which would be a nightmare to typecheck). As an aside, note that that
is also the reason we have the lexer synthesize a T_Dims ([]) token distinct from [
and ].
Checking that this new type declaration is only used within a function scope
and not for globals, instance variables or formal parameters (can be done
syntactically during the parsing phase.)
Checking that the element type of any array type declaration is not a stack-
allocated array (can also be done during parsing.)
Checking that the expression within the brackets can be evaluated to an integer,
without however using variables in the scope in which they are defined, which
may require changes to your scoping mechanism as it’s different from the
default scoping rules in Decaf.
Adapting the type compatibility/equivalence functions to make sure stack-
allocated arrays can be used in any place heap-allocated arrays can be used,
except possibly as the target of a NewArray or a reassignment (depending on
your assumptions.)
e) This extension makes Decaf less typesafe, because we can now end up with
dangling pointers to objects allocated on the stack if references to such stack-
allocated arrays escape the function.
g) Unless you want to compute the frame-pointer-relative offsets for some variables at
runtime, we need to place memory for the array elements below all of the space set
aside for local variables and temporaries. The frame can be extended at runtime
by the code emitted from Mips::AllocA, after the size expression has been allocated.
h) They won’t affect them at all, provided you plant the memory for stack arrays
below everything else.
i)