Index: utils/TableGen/CMakeLists.txt =================================================================== --- utils/TableGen/CMakeLists.txt +++ utils/TableGen/CMakeLists.txt @@ -6,6 +6,7 @@ AsmWriterInst.cpp Attributes.cpp CallingConvEmitter.cpp + CodeBeadsGen.cpp CodeEmitterGen.cpp CodeGenDAGPatterns.cpp CodeGenHwModes.cpp Index: utils/TableGen/CodeBeadsGen.cpp =================================================================== --- /dev/null +++ utils/TableGen/CodeBeadsGen.cpp @@ -0,0 +1,135 @@ +//===-- CodeBeadsGen.cpp - Code Beads Generator -*- C++ -*-----------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +/// \file +/// This file contains tablegen implementation of "code beads", binary strings +/// defined for an instruction. They are used to pass very target specific +/// binary information directly from a tablegen file, e.g. instruction +/// encoding. +/// +//===----------------------------------------------------------------------===// + +#include "CodeGenTarget.h" + +#include "llvm/ADT/StringExtras.h" +#include "llvm/Support/Debug.h" +#include "llvm/TableGen/Error.h" +#include "llvm/TableGen/Record.h" +#include "llvm/TableGen/TableGenBackend.h" + +#include +#include +#include + +using namespace llvm; + +namespace { + +class CodeBeadsGen { + RecordKeeper &Records; + +public: + CodeBeadsGen(RecordKeeper &R) : Records(R) {} + void run(raw_ostream &o); +}; + +void CodeBeadsGen::run(raw_ostream &o) { + CodeGenTarget Target(Records); + + // For little-endian instruction bit encodings, reverse the bit order + Target.reverseBitsForLittleEndianEncoding(); + + auto Instructions = Target.getInstructionsByEnumValue(); + + // Every bead is assumed to be 8 bits long + constexpr unsigned BeadSize = 8; + // Current limit of beads is 192 bits + constexpr unsigned BeadsLength = 192; + constexpr unsigned BeadsNumber = BeadsLength / BeadSize; + + // Emit function declaration + o << "const uint" << BeadSize << "_t * " << Target.getName(); + o << "MCCodeEmitter::getGenInstrBeads(const MCInst &MI) const {\n"; + + // Emit instruction base values + o << " static const uint" << BeadSize << "_t InstBits[][" << BeadsNumber + << "] = {\n"; + + for (const auto *CGI : Instructions) { + Record *R = CGI->TheDef; + + // Pseudos don't have encoding + if (R->getValueAsString("Namespace") == "TargetOpcode" || + R->getValueAsBit("isPseudo")) { + o << "\t{ 0x0 },"; + o << '\t' << "// (Pseudo) " << R->getName() << "\n"; + continue; + } + + BitsInit *Beads = R->getValueAsBitsInit("Beads"); + + if (!Beads->isComplete()) { + PrintFatalError(R->getLoc(), "Record `" + R->getName() + + "', bit field 'Beads' is not complete"); + } + + if (Beads->getNumBits() > BeadsLength) { + PrintFatalError(R->getLoc(), + "Record `" + R->getName() + + "', bit field 'Beads' is too long(maximum: " + + std::to_string(BeadsLength) + ")"); + } + + // Convert to byte array: + // [dcba] -> [a][b][c][d] + o << "\t{"; + for (unsigned p = 0; p < BeadsNumber; ++p) { + unsigned Right = BeadSize * p; + unsigned Left = Right + BeadSize; + unsigned Value = 0; + + for (unsigned i = Right; i != Left; ++i) { + unsigned Shift = i % BeadSize; + + if (BitInit *B = dyn_cast(Beads->getBit(i))) { + Value |= (static_cast(B->getValue()) << Shift); + } else { + PrintFatalError(R->getLoc(), + "Record `" + R->getName() + "', bit 'Beads[" + + std::to_string(i) + "]' is not defined"); + } + } + + if (p != 0) { + o << ','; + } + + o << " 0x"; + o.write_hex(Value); + o << ""; + } + o << " }," << '\t' << "// " << R->getName() << "\n"; + } + o << "\t{ 0x0 }\n };\n"; + + // Emit initial function code + o << " return InstBits[MI.getOpcode()];\n" + << "}\n\n"; +} + +} // anonymous namespace + +namespace llvm { + +void EmitCodeBeads(RecordKeeper &RK, raw_ostream &OS) { + emitSourceFileHeader("Machine Code Beads", OS); + CodeBeadsGen(RK).run(OS); +} + +} // namespace llvm Index: utils/TableGen/TableGen.cpp =================================================================== --- utils/TableGen/TableGen.cpp +++ utils/TableGen/TableGen.cpp @@ -25,6 +25,7 @@ enum ActionType { PrintRecords, DumpJSON, + GenCodeBeads, GenEmitter, GenRegisterInfo, GenInstrInfo, @@ -62,6 +63,8 @@ "Print all records to stdout (default)"), clEnumValN(DumpJSON, "dump-json", "Dump all records as machine-readable JSON"), + clEnumValN(GenCodeBeads, "gen-code-beads", + "Generate machine code beads"), clEnumValN(GenEmitter, "gen-emitter", "Generate machine code emitter"), clEnumValN(GenRegisterInfo, "gen-register-info", @@ -132,6 +135,9 @@ case DumpJSON: EmitJSON(Records, OS); break; + case GenCodeBeads: + EmitCodeBeads(Records, OS); + break; case GenEmitter: EmitCodeEmitter(Records, OS); break; Index: utils/TableGen/TableGenBackends.h =================================================================== --- utils/TableGen/TableGenBackends.h +++ utils/TableGen/TableGenBackends.h @@ -69,6 +69,7 @@ void EmitAsmMatcher(RecordKeeper &RK, raw_ostream &OS); void EmitAsmWriter(RecordKeeper &RK, raw_ostream &OS); void EmitCallingConv(RecordKeeper &RK, raw_ostream &OS); +void EmitCodeBeads(RecordKeeper &RK, raw_ostream &OS); void EmitCodeEmitter(RecordKeeper &RK, raw_ostream &OS); void EmitDAGISel(RecordKeeper &RK, raw_ostream &OS); void EmitDFAPacketizer(RecordKeeper &RK, raw_ostream &OS);