/***** * coder.cc * Andy Hammerlindl 2004/11/06 * * Handles encoding of syntax into programs. It's methods are called by * abstract syntax objects during translation to construct the virtual machine * code. *****/ #include #include "errormsg.h" #include "coder.h" #include "genv.h" #include "entry.h" using namespace sym; using namespace types; namespace trans { namespace { function *inittype(); function *bootuptype(); } vm::lambda *newLambda(string name) { assert(!name.empty()); vm::lambda *l = new vm::lambda; #ifdef DEBUG_FRAME l->name = name; #endif return l; } // The dummy environment of the global environment. // Used purely for global variables and static code blocks of file // level modules. coder::coder(string name, modifier sord) : level(new frame(name, 0, 0)), recordLevel(0), recordType(0), isCodelet(false), l(newLambda(name)), funtype(bootuptype()), parent(0), sord(sord), perm(DEFAULT_PERM), program(new vm::program), numLabels(0), curPos(nullPos) { sord_stack.push(sord); encodeAllocInstruction(); } // Defines a new function environment. coder::coder(string name, function *t, coder *parent, modifier sord, bool reframe) : level(reframe ? new frame(name, parent->getFrame(), t->sig.getNumFormals()) : parent->getFrame()), recordLevel(parent->recordLevel), recordType(parent->recordType), isCodelet(!reframe), l(newLambda(name)), funtype(t), parent(parent), sord(sord), perm(DEFAULT_PERM), program(new vm::program), numLabels(0), curPos(nullPos) { sord_stack.push(sord); encodeAllocInstruction(); } // Start encoding the body of the record. The function being encoded // is the record's initializer. coder::coder(record *t, coder *parent, modifier sord) : level(t->getLevel()), recordLevel(t->getLevel()), recordType(t), isCodelet(false), l(t->getInit()), funtype(inittype()), parent(parent), sord(sord), perm(DEFAULT_PERM), program(new vm::program), numLabels(0), curPos(nullPos) { sord_stack.push(sord); encodeAllocInstruction(); } coder coder::newFunction(string name, function *t, modifier sord) { return coder(name, t, this, sord); } coder coder::newCodelet() { return coder("", new function(primVoid()), this, DEFAULT_DYNAMIC, false); } record *coder::newRecord(symbol *id) { frame *underlevel = getFrame(); frame *level = new frame(*id, underlevel, 0); record *r = new record(id, level); return r; } coder coder::newRecordInit(record *r, modifier sord) { return coder(r, this, sord); } #ifdef DEBUG_BLTIN void assertBltinLookup(inst::opcode op, item it) { if (op == inst::builtin) { string name=lookupBltin(vm::get(it)); assert(!name.empty()); } } #endif bool coder::encode(frame *f) { frame *toplevel = getFrame(); if (f == 0) { encode(inst::constpush,(item)0); } else if (f == toplevel) { encode(inst::pushclosure); } else { encode(inst::varpush,0); frame *level = toplevel->getParent(); while (level != f) { if (level == 0) // Frame request was in an improper scope. return false; encode(inst::fieldpush,0); level = level->getParent(); } } return true; } bool coder::encode(frame *dest, frame *top) { //cerr << "coder::encode()\n"; if (dest == 0) { encode(inst::pop); encode(inst::constpush,(item)0); } else { frame *level = top; while (level != dest) { if (level == 0) { // Frame request was in an improper scope. //cerr << "failed\n"; return false; } encode(inst::fieldpush,0); level = level->getParent(); } } //cerr << "succeeded\n"; return true; } Int coder::defLabel() { if (isStatic()) return parent->defLabel(); return defLabel(numLabels++); } Int coder::defLabel(Int label) { if (isStatic()) return parent->defLabel(label); assert(label >= 0 && label < numLabels); defs.insert(std::make_pair(label,program->end())); std::multimap::iterator p = uses.lower_bound(label); while (p != uses.upper_bound(label)) { p->second->ref = program->end(); ++p; } return label; } void coder::useLabel(inst::opcode op, Int label) { if (isStatic()) return parent->useLabel(op,label); std::map::iterator p = defs.find(label); if (p != defs.end()) { encode(op,p->second); } else { // Not yet defined uses.insert(std::make_pair(label,program->end())); encode(op); } } Int coder::fwdLabel() { if (isStatic()) return parent->fwdLabel(); // Create a new label without specifying its position. return numLabels++; } void coder::markPos(position pos) { curPos = pos; } // When translating the function is finished, this ties up loose ends // and returns the lambda. vm::lambda *coder::close() { // These steps must be done dynamically, not statically. sord = EXPLICIT_DYNAMIC; sord_stack.push(sord); // Add a return for void types; may be redundant. if (funtype->result->kind == types::ty_void) encode(inst::ret); l->code = program; l->params = level->getNumFormals(); // Now that we know how many variables the function has, allocate space for // all of them at the start of the function. finishAlloc(); sord_stack.pop(); sord = sord_stack.top(); return l; } void coder::closeRecord() { // Put record into finished state. encode(inst::pushclosure); close(); } bool coder::isRecord() { return (funtype==inittype()); } namespace { function *inittype() { static function t(types::primVoid()); return &t; } function *bootuptype() { static function t(types::primVoid()); return &t; } } // private } // namespace trans