#include "riscv.h" #include "backend.h" #include "ir.h" const int REG_NUMBER = 32; RiscvFunction::RiscvFunction(std::string name, int num_args, OpTy Ty) // 返回值,无返回使用void类型 : RiscvLabel(Function, name), num_args_(num_args), resType_(Ty), base_(-VARIABLE_ALIGN_BYTE) { regAlloca = new RegAlloca(); } // 输出函数对应的全部riscv语句序列 // 由于一个函数可能有若干个出口,因而恢复现场的语句根据basic block // 语句中的ret语句前面附带出现,因而不在此出现 std::string RiscvFunction::print() { // TODO: temporaily add '.global' to declare function // Don't know if '.type' is needed std::string riscvInstr = ".global " + this->name_ + "\n" + this->name_ + ":\n"; // 函数标号打印 // 对各个basic block进行拼接 for (auto x : this->blk) riscvInstr += x->print(); return riscvInstr; } std::string RiscvBasicBlock::print() { std::string riscvInstr = this->name_ + ":\n"; for (auto x : this->instruction) riscvInstr += x->print(); return riscvInstr; } // 出栈顺序和入栈相反 // 建议不使用pop语句,直接从栈中取值,最后直接修改sp的值即可 // 使用一个单独的return block以防止多出口return extern int LabelCount; RiscvOperand::OpTy RiscvOperand::getType() { return tid_; } bool RiscvOperand::isRegister() { return tid_ == FloatReg || tid_ == IntReg; } Type *findPtrType(Type *ty) { while (ty->tid_ == Type::PointerTyID) { ty = static_cast(ty)->contained_; } while (ty->tid_ == Type::ArrayTyID) { ty = static_cast(ty)->contained_; } assert(ty->tid_ == Type::IntegerTyID || ty->tid_ == Type::FloatTyID); return ty; } std::string RiscvGlobalVariable::print(bool print_name, Constant *initVal) { std::string code = ""; // 如果在调用的第一层,初始化 initVal if (print_name) { code += this->name_ + ":\n"; initVal = initValue_; } if (initVal == nullptr) return "\t.zero\t" + std::to_string(this->elementNum_ * 4) + "\n";; // 如果无初始值,或初始值为0(IR中有ConstZero类),则直接用zero命令 if (dynamic_cast(initVal) != nullptr) { code += "\t.zero\t" + std::to_string(calcTypeSize(initVal->type_)) + "\n"; return code; } // 下面是非零的处理 // 整型 if (initVal->type_->tid_ == Type::TypeID::IntegerTyID) { code += "\t.word\t" + std::to_string(dynamic_cast(initVal)->value_) + "\n"; return code; } // 浮点 else if (initVal->type_->tid_ == Type::TypeID::FloatTyID) { std::string valString = dynamic_cast(initVal)->print32(); while (valString.length() < 10) valString += "0"; code += "\t.word\t" + valString.substr(0, 10) + "\n"; return code; } else if (initVal->type_->tid_ == Type::TypeID::ArrayTyID) { ConstantArray *const_arr = dynamic_cast(initVal); assert(const_arr != nullptr); int zeroSpace = calcTypeSize(initVal->type_); for (auto elements : const_arr->const_array) { code += print(false, elements); zeroSpace -= 4; } if (zeroSpace) code += "\t.zero\t" + std::to_string(zeroSpace) + "\n"; return code; } else { std::cerr << "[Fatal Error] Unknown RiscvGlobalVariable::print() initValue type." << std::endl; std::terminate(); } } std::string RiscvGlobalVariable::print() { return print(true, nullptr); } RiscvFunction *createSyslibFunc(Function *foo) { if (foo->name_ == "__aeabi_memclr4") { auto *rfoo = createRiscvFunction(foo); // 预处理块 auto *bb1 = createRiscvBasicBlock(); bb1->addInstrBack(new MoveRiscvInst(getRegOperand("t5"), getRegOperand("a0"), bb1)); bb1->addInstrBack(new MoveRiscvInst(getRegOperand("t6"), getRegOperand("a1"), bb1)); bb1->addInstrBack(new BinaryRiscvInst(RiscvInstr::ADD, getRegOperand("a0"), getRegOperand("t6"), getRegOperand("t6"), bb1)); bb1->addInstrBack( new MoveRiscvInst(getRegOperand("a0"), new RiscvConst(0), bb1)); auto *bb2 = createRiscvBasicBlock(); // 循环块 // 默认clear为全0 bb2->addInstrBack(new StoreRiscvInst( new Type(Type::TypeID::IntegerTyID), getRegOperand("zero"), new RiscvIntPhiReg(NamefindReg("t5")), bb2)); bb2->addInstrBack(new BinaryRiscvInst(RiscvInstr::ADDI, getRegOperand("t5"), new RiscvConst(4), getRegOperand("t5"), bb1)); bb2->addInstrBack(new ICmpRiscvInstr(ICmpInst::ICMP_SLT, getRegOperand("t5"), getRegOperand("t6"), bb2, bb2)); bb2->addInstrBack(new ReturnRiscvInst(bb2)); rfoo->addBlock(bb1); rfoo->addBlock(bb2); return rfoo; } return nullptr; }