mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-28 06:32:09 +00:00
llvm-mc/AsmParser: Separate instruction ordering for ambiguity detection.
- We want the ordering operation to be simple, since we run it on every match. The old ordering is also not a strict weak ordering when there are ambiguities, which makes MSVC unhappy. - While we are at it, detect all ambiguities instead of just the adjacent ones. There are actually 655, for X86. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@78526 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
1ff446fef6
commit
2b54481a77
@ -263,7 +263,7 @@ static bool IsAssemblerInstruction(const StringRef &Name,
|
|||||||
DEBUG({
|
DEBUG({
|
||||||
errs() << "warning: '" << Name << "': "
|
errs() << "warning: '" << Name << "': "
|
||||||
<< "ignoring instruction; tied operand '"
|
<< "ignoring instruction; tied operand '"
|
||||||
<< Tokens[i] << "', \n";
|
<< Tokens[i] << "'\n";
|
||||||
});
|
});
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -368,29 +368,9 @@ struct InstructionInfo {
|
|||||||
|
|
||||||
/// operator< - Compare two instructions.
|
/// operator< - Compare two instructions.
|
||||||
bool operator<(const InstructionInfo &RHS) const {
|
bool operator<(const InstructionInfo &RHS) const {
|
||||||
// Order first by the number of operands (which is unambiguous).
|
|
||||||
if (Operands.size() != RHS.Operands.size())
|
if (Operands.size() != RHS.Operands.size())
|
||||||
return Operands.size() < RHS.Operands.size();
|
return Operands.size() < RHS.Operands.size();
|
||||||
|
|
||||||
// Otherwise, order by lexicographic comparison of tokens and operand kinds
|
|
||||||
// (these can never be ambiguous).
|
|
||||||
for (unsigned i = 0, e = Operands.size(); i != e; ++i)
|
|
||||||
if (Operands[i].Class->Kind != RHS.Operands[i].Class->Kind ||
|
|
||||||
Operands[i].Class->Kind == ClassInfo::Token)
|
|
||||||
if (*Operands[i].Class < *RHS.Operands[i].Class)
|
|
||||||
return true;
|
|
||||||
|
|
||||||
// Finally, order by the component wise comparison of operand classes. We
|
|
||||||
// don't want to rely on the lexigraphic ordering of elements, so we define
|
|
||||||
// only define the ordering when it is unambiguous. That is, when some pair
|
|
||||||
// compares less than and no pair compares greater than.
|
|
||||||
|
|
||||||
// Check that no pair compares greater than.
|
|
||||||
for (unsigned i = 0, e = Operands.size(); i != e; ++i)
|
|
||||||
if (*RHS.Operands[i].Class < *Operands[i].Class)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
// Otherwise, return true if some pair compares less than.
|
|
||||||
for (unsigned i = 0, e = Operands.size(); i != e; ++i)
|
for (unsigned i = 0, e = Operands.size(); i != e; ++i)
|
||||||
if (*Operands[i].Class < *RHS.Operands[i].Class)
|
if (*Operands[i].Class < *RHS.Operands[i].Class)
|
||||||
return true;
|
return true;
|
||||||
@ -398,6 +378,37 @@ struct InstructionInfo {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// CouldMatchAmiguouslyWith - Check whether this instruction could
|
||||||
|
/// ambiguously match the same set of operands as \arg RHS (without being a
|
||||||
|
/// strictly superior match).
|
||||||
|
bool CouldMatchAmiguouslyWith(const InstructionInfo &RHS) {
|
||||||
|
// The number of operands is unambiguous.
|
||||||
|
if (Operands.size() != RHS.Operands.size())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Tokens and operand kinds are unambiguous (assuming a correct target
|
||||||
|
// specific parser).
|
||||||
|
for (unsigned i = 0, e = Operands.size(); i != e; ++i)
|
||||||
|
if (Operands[i].Class->Kind != RHS.Operands[i].Class->Kind ||
|
||||||
|
Operands[i].Class->Kind == ClassInfo::Token)
|
||||||
|
if (*Operands[i].Class < *RHS.Operands[i].Class ||
|
||||||
|
*RHS.Operands[i].Class < *Operands[i].Class)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Otherwise, this operand could commute if all operands are equivalent, or
|
||||||
|
// there is a pair of operands that compare less than and a pair that
|
||||||
|
// compare greater than.
|
||||||
|
bool HasLT = false, HasGT = false;
|
||||||
|
for (unsigned i = 0, e = Operands.size(); i != e; ++i) {
|
||||||
|
if (*Operands[i].Class < *RHS.Operands[i].Class)
|
||||||
|
HasLT = true;
|
||||||
|
if (*RHS.Operands[i].Class < *Operands[i].Class)
|
||||||
|
HasGT = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return !(HasLT ^ HasGT);
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void dump();
|
void dump();
|
||||||
};
|
};
|
||||||
@ -991,23 +1002,21 @@ void AsmMatcherEmitter::run(raw_ostream &OS) {
|
|||||||
|
|
||||||
// Check for ambiguous instructions.
|
// Check for ambiguous instructions.
|
||||||
unsigned NumAmbiguous = 0;
|
unsigned NumAmbiguous = 0;
|
||||||
for (std::vector<InstructionInfo*>::const_iterator it =
|
for (unsigned i = 0, e = Info.Instructions.size(); i != e; ++i) {
|
||||||
Info.Instructions.begin(), ie = Info.Instructions.end() - 1;
|
for (unsigned j = i + 1; j != e; ++j) {
|
||||||
it != ie;) {
|
InstructionInfo &A = *Info.Instructions[i];
|
||||||
InstructionInfo &II = **it;
|
InstructionInfo &B = *Info.Instructions[j];
|
||||||
++it;
|
|
||||||
|
|
||||||
InstructionInfo &Next = **it;
|
if (A.CouldMatchAmiguouslyWith(B)) {
|
||||||
|
DEBUG_WITH_TYPE("ambiguous_instrs", {
|
||||||
if (!(II < Next)){
|
errs() << "warning: ambiguous instruction match:\n";
|
||||||
DEBUG_WITH_TYPE("ambiguous_instrs", {
|
A.dump();
|
||||||
errs() << "warning: ambiguous instruction match:\n";
|
errs() << "\nis incomparable with:\n";
|
||||||
II.dump();
|
B.dump();
|
||||||
errs() << "\nis incomparable with:\n";
|
errs() << "\n\n";
|
||||||
Next.dump();
|
});
|
||||||
errs() << "\n\n";
|
++NumAmbiguous;
|
||||||
});
|
}
|
||||||
++NumAmbiguous;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (NumAmbiguous)
|
if (NumAmbiguous)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user