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:
Bruno Cardoso Lopes 2010-06-05 02:11:52 +00:00
parent 8613b18a5a
commit 270562b3d4
4 changed files with 111 additions and 10 deletions

View File

@ -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&lt;bits&lt;4&gt; opc, string Name&gt; {
bits&lt;4&gt; opcode = opc;
string name = Name;
}
<b>multiclass</b> basic_r&lt;bits&lt;4&gt; opc&gt; {
<b>def</b> rr : Instruction&lt;opc, "rr"&gt;;
<b>def</b> rm : Instruction&lt;opc, "rm"&gt;;
}
<b>multiclass</b> basic_s&lt;bits&lt;4&gt; opc&gt; {
<b>defm</b> SS : basic_r&lt;opc&gt;;
<b>defm</b> SD : basic_r&lt;opc&gt;;
<b>def</b> X : Instruction&lt;opc, "x"&gt;;
}
<b>multiclass</b> basic_p&lt;bits&lt;4&gt; opc&gt; {
<b>defm</b> PS : basic_r&lt;opc&gt;;
<b>defm</b> PD : basic_r&lt;opc&gt;;
<b>def</b> Y : Instruction&lt;opc, "y"&gt;;
}
<b>defm</b> ADD : basic_s&lt;0xf&gt;, basic_p&lt;0xf&gt;;
...
<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>
<!-- ======================================================================= --> <!-- ======================================================================= -->

View 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>;

View File

@ -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;

View File

@ -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();