mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-14 16:33:28 +00:00
Teach tablegen to support 'defm' inside multiclasses.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@105519 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
8613b18a5a
commit
270562b3d4
@ -687,6 +687,50 @@ Here is an example TableGen fragment that shows this idea:
|
|||||||
</pre>
|
</pre>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
A defm can also be used inside a multiclass providing several levels of
|
||||||
|
multiclass instanciations.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="doc_code">
|
||||||
|
<pre>
|
||||||
|
<b>class</b> Instruction<bits<4> opc, string Name> {
|
||||||
|
bits<4> opcode = opc;
|
||||||
|
string name = Name;
|
||||||
|
}
|
||||||
|
|
||||||
|
<b>multiclass</b> basic_r<bits<4> opc> {
|
||||||
|
<b>def</b> rr : Instruction<opc, "rr">;
|
||||||
|
<b>def</b> rm : Instruction<opc, "rm">;
|
||||||
|
}
|
||||||
|
|
||||||
|
<b>multiclass</b> basic_s<bits<4> opc> {
|
||||||
|
<b>defm</b> SS : basic_r<opc>;
|
||||||
|
<b>defm</b> SD : basic_r<opc>;
|
||||||
|
<b>def</b> X : Instruction<opc, "x">;
|
||||||
|
}
|
||||||
|
|
||||||
|
<b>multiclass</b> basic_p<bits<4> opc> {
|
||||||
|
<b>defm</b> PS : basic_r<opc>;
|
||||||
|
<b>defm</b> PD : basic_r<opc>;
|
||||||
|
<b>def</b> Y : Instruction<opc, "y">;
|
||||||
|
}
|
||||||
|
|
||||||
|
<b>defm</b> ADD : basic_s<0xf>, basic_p<0xf>;
|
||||||
|
...
|
||||||
|
|
||||||
|
<i>// Results</i>
|
||||||
|
<b>def</b> ADDPDrm { ...
|
||||||
|
<b>def</b> ADDPDrr { ...
|
||||||
|
<b>def</b> ADDPSrm { ...
|
||||||
|
<b>def</b> ADDPSrr { ...
|
||||||
|
<b>def</b> ADDSDrm { ...
|
||||||
|
<b>def</b> ADDSDrr { ...
|
||||||
|
<b>def</b> ADDY { ...
|
||||||
|
<b>def</b> ADDX { ...
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- ======================================================================= -->
|
<!-- ======================================================================= -->
|
||||||
|
25
test/TableGen/DefmInsideMultiClass.td
Normal file
25
test/TableGen/DefmInsideMultiClass.td
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
// RUN: tblgen %s | grep ADDPSrr | count 1
|
||||||
|
// XFAIL: vg_leak
|
||||||
|
|
||||||
|
class Instruction<bits<4> opc, string Name> {
|
||||||
|
bits<4> opcode = opc;
|
||||||
|
string name = Name;
|
||||||
|
}
|
||||||
|
|
||||||
|
multiclass basic_r<bits<4> opc> {
|
||||||
|
def rr : Instruction<opc, "rr">;
|
||||||
|
def rm : Instruction<opc, "rm">;
|
||||||
|
}
|
||||||
|
|
||||||
|
multiclass basic_s<bits<4> opc> {
|
||||||
|
defm SS : basic_r<opc>;
|
||||||
|
defm SD : basic_r<opc>;
|
||||||
|
}
|
||||||
|
|
||||||
|
multiclass basic_p<bits<4> opc> {
|
||||||
|
defm PS : basic_r<opc>;
|
||||||
|
defm PD : basic_r<opc>;
|
||||||
|
}
|
||||||
|
|
||||||
|
defm ADD : basic_s<0xf>, basic_p<0xf>;
|
||||||
|
defm SUB : basic_s<0xe>, basic_p<0xe>;
|
@ -1635,7 +1635,6 @@ bool TGParser::ParseObjectBody(Record *CurRec) {
|
|||||||
return ParseBody(CurRec);
|
return ParseBody(CurRec);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// ParseDef - Parse and return a top level or multiclass def, return the record
|
/// ParseDef - Parse and return a top level or multiclass def, return the record
|
||||||
/// corresponding to it. This returns null on error.
|
/// corresponding to it. This returns null on error.
|
||||||
///
|
///
|
||||||
@ -1807,8 +1806,6 @@ bool TGParser::ParseTopLevelLet() {
|
|||||||
/// MultiClassDef ::= DefInst
|
/// MultiClassDef ::= DefInst
|
||||||
///
|
///
|
||||||
bool TGParser::ParseMultiClassDef(MultiClass *CurMC) {
|
bool TGParser::ParseMultiClassDef(MultiClass *CurMC) {
|
||||||
if (Lex.getCode() != tgtok::Def)
|
|
||||||
return TokError("expected 'def' in multiclass body");
|
|
||||||
|
|
||||||
Record *D = ParseDef(CurMC);
|
Record *D = ParseDef(CurMC);
|
||||||
if (D == 0) return true;
|
if (D == 0) return true;
|
||||||
@ -1885,9 +1882,18 @@ bool TGParser::ParseMultiClass() {
|
|||||||
if (Lex.Lex() == tgtok::r_brace) // eat the '{'.
|
if (Lex.Lex() == tgtok::r_brace) // eat the '{'.
|
||||||
return TokError("multiclass must contain at least one def");
|
return TokError("multiclass must contain at least one def");
|
||||||
|
|
||||||
while (Lex.getCode() != tgtok::r_brace)
|
while (Lex.getCode() != tgtok::r_brace) {
|
||||||
if (ParseMultiClassDef(CurMultiClass))
|
if (Lex.getCode() != tgtok::Defm && Lex.getCode() != tgtok::Def)
|
||||||
return true;
|
return TokError("expected 'def' or 'defm' in multiclass body");
|
||||||
|
|
||||||
|
if (Lex.getCode() == tgtok::Def)
|
||||||
|
if (ParseMultiClassDef(CurMultiClass))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
if (Lex.getCode() == tgtok::Defm)
|
||||||
|
if (ParseDefm(CurMultiClass))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
Lex.Lex(); // eat the '}'.
|
Lex.Lex(); // eat the '}'.
|
||||||
}
|
}
|
||||||
@ -1900,7 +1906,7 @@ bool TGParser::ParseMultiClass() {
|
|||||||
///
|
///
|
||||||
/// DefMInst ::= DEFM ID ':' DefmSubClassRef ';'
|
/// DefMInst ::= DEFM ID ':' DefmSubClassRef ';'
|
||||||
///
|
///
|
||||||
bool TGParser::ParseDefm() {
|
bool TGParser::ParseDefm(MultiClass *CurMultiClass) {
|
||||||
assert(Lex.getCode() == tgtok::Defm && "Unexpected token!");
|
assert(Lex.getCode() == tgtok::Defm && "Unexpected token!");
|
||||||
if (Lex.Lex() != tgtok::Id) // eat the defm.
|
if (Lex.Lex() != tgtok::Id) // eat the defm.
|
||||||
return TokError("expected identifier after defm");
|
return TokError("expected identifier after defm");
|
||||||
@ -1991,8 +1997,34 @@ bool TGParser::ParseDefm() {
|
|||||||
return Error(DefmPrefixLoc, "def '" + CurRec->getName() +
|
return Error(DefmPrefixLoc, "def '" + CurRec->getName() +
|
||||||
"' already defined, instantiating defm with subdef '" +
|
"' already defined, instantiating defm with subdef '" +
|
||||||
DefProto->getName() + "'");
|
DefProto->getName() + "'");
|
||||||
Records.addDef(CurRec);
|
|
||||||
CurRec->resolveReferences();
|
// Don't create a top level definition for defm inside multiclasses,
|
||||||
|
// instead, only update the prototypes and bind the template args
|
||||||
|
// with the new created definition.
|
||||||
|
if (CurMultiClass) {
|
||||||
|
for (unsigned i = 0, e = CurMultiClass->DefPrototypes.size();
|
||||||
|
i != e; ++i) {
|
||||||
|
if (CurMultiClass->DefPrototypes[i]->getName() == CurRec->getName()) {
|
||||||
|
Error(DefmPrefixLoc, "defm '" + CurRec->getName() +
|
||||||
|
"' already defined in this multiclass!");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
CurMultiClass->DefPrototypes.push_back(CurRec);
|
||||||
|
|
||||||
|
// Copy the template arguments for the multiclass into the new def.
|
||||||
|
const std::vector<std::string> &TA =
|
||||||
|
CurMultiClass->Rec.getTemplateArgs();
|
||||||
|
|
||||||
|
for (unsigned i = 0, e = TA.size(); i != e; ++i) {
|
||||||
|
const RecordVal *RV = CurMultiClass->Rec.getValue(TA[i]);
|
||||||
|
assert(RV && "Template arg doesn't exist?");
|
||||||
|
CurRec->addValue(*RV);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Records.addDef(CurRec);
|
||||||
|
CurRec->resolveReferences();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Lex.getCode() != tgtok::comma) break;
|
if (Lex.getCode() != tgtok::comma) break;
|
||||||
|
@ -74,7 +74,7 @@ private: // Parser methods.
|
|||||||
bool ParseClass();
|
bool ParseClass();
|
||||||
bool ParseMultiClass();
|
bool ParseMultiClass();
|
||||||
bool ParseMultiClassDef(MultiClass *CurMC);
|
bool ParseMultiClassDef(MultiClass *CurMC);
|
||||||
bool ParseDefm();
|
bool ParseDefm(MultiClass *CurMultiClass = 0);
|
||||||
bool ParseTopLevelLet();
|
bool ParseTopLevelLet();
|
||||||
std::vector<LetRecord> ParseLetList();
|
std::vector<LetRecord> ParseLetList();
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user