[mips][ias] EF_MIPS_MICROMIPS should iff microMIPS code was emitted.

Summary:
This fixes PR27682. Additionally, '.set micromips' by itself is not sufficient
to raise the EF_MIPS_MICROMIPS flag. It is also necessary to emit a microMIPS
instruction. This has also been fixed.

Reviewers: sdardis, vkalintiris, rafael

Subscribers: rafael, dsanders, sdardis, llvm-commits

Differential Revision: http://reviews.llvm.org/D20214

llvm-svn: 269639
This commit is contained in:
Daniel Sanders 2016-05-16 09:10:13 +00:00
parent 367577aab9
commit cda908a0b6
5 changed files with 34 additions and 6 deletions

View File

@ -1900,6 +1900,11 @@ bool MipsAsmParser::processInstruction(MCInst &Inst, SMLoc IDLoc,
return true; return true;
} }
// We know we emitted an instruction on the MER_NotAMacro or MER_Success path.
// If we're in microMIPS mode then we must also set EF_MIPS_MICROMIPS.
if (inMicroMipsMode())
TOut.setUsesMicroMips();
// If this instruction has a delay slot and .set reorder is active, // If this instruction has a delay slot and .set reorder is active,
// emit a NOP after it. // emit a NOP after it.
if (FillDelaySlot) { if (FillDelaySlot) {
@ -5288,6 +5293,7 @@ bool MipsAsmParser::parseSetFeature(uint64_t Feature) {
getTargetStreamer().emitDirectiveSetDsp(); getTargetStreamer().emitDirectiveSetDsp();
break; break;
case Mips::FeatureMicroMips: case Mips::FeatureMicroMips:
setFeatureBits(Mips::FeatureMicroMips, "micromips");
getTargetStreamer().emitDirectiveSetMicroMips(); getTargetStreamer().emitDirectiveSetMicroMips();
break; break;
case Mips::FeatureMips1: case Mips::FeatureMips1:
@ -5586,6 +5592,7 @@ bool MipsAsmParser::parseDirectiveSet() {
} else if (Tok.getString() == "nomips16") { } else if (Tok.getString() == "nomips16") {
return parseSetNoMips16Directive(); return parseSetNoMips16Directive();
} else if (Tok.getString() == "nomicromips") { } else if (Tok.getString() == "nomicromips") {
clearFeatureBits(Mips::FeatureMicroMips, "micromips");
getTargetStreamer().emitDirectiveSetNoMicroMips(); getTargetStreamer().emitDirectiveSetNoMicroMips();
Parser.eatToEndOfStatement(); Parser.eatToEndOfStatement();
return false; return false;

View File

@ -40,6 +40,7 @@ MipsTargetStreamer::MipsTargetStreamer(MCStreamer &S)
} }
void MipsTargetStreamer::emitDirectiveSetMicroMips() {} void MipsTargetStreamer::emitDirectiveSetMicroMips() {}
void MipsTargetStreamer::emitDirectiveSetNoMicroMips() {} void MipsTargetStreamer::emitDirectiveSetNoMicroMips() {}
void MipsTargetStreamer::setUsesMicroMips() {}
void MipsTargetStreamer::emitDirectiveSetMips16() {} void MipsTargetStreamer::emitDirectiveSetMips16() {}
void MipsTargetStreamer::emitDirectiveSetNoMips16() { forbidModuleDirective(); } void MipsTargetStreamer::emitDirectiveSetNoMips16() { forbidModuleDirective(); }
void MipsTargetStreamer::emitDirectiveSetReorder() { forbidModuleDirective(); } void MipsTargetStreamer::emitDirectiveSetReorder() { forbidModuleDirective(); }
@ -830,11 +831,6 @@ MCELFStreamer &MipsTargetELFStreamer::getStreamer() {
void MipsTargetELFStreamer::emitDirectiveSetMicroMips() { void MipsTargetELFStreamer::emitDirectiveSetMicroMips() {
MicroMipsEnabled = true; MicroMipsEnabled = true;
MCAssembler &MCA = getStreamer().getAssembler();
unsigned Flags = MCA.getELFHeaderEFlags();
Flags |= ELF::EF_MIPS_MICROMIPS;
MCA.setELFHeaderEFlags(Flags);
forbidModuleDirective(); forbidModuleDirective();
} }
@ -843,6 +839,13 @@ void MipsTargetELFStreamer::emitDirectiveSetNoMicroMips() {
forbidModuleDirective(); forbidModuleDirective();
} }
void MipsTargetELFStreamer::setUsesMicroMips() {
MCAssembler &MCA = getStreamer().getAssembler();
unsigned Flags = MCA.getELFHeaderEFlags();
Flags |= ELF::EF_MIPS_MICROMIPS;
MCA.setELFHeaderEFlags(Flags);
}
void MipsTargetELFStreamer::emitDirectiveSetMips16() { void MipsTargetELFStreamer::emitDirectiveSetMips16() {
MCAssembler &MCA = getStreamer().getAssembler(); MCAssembler &MCA = getStreamer().getAssembler();
unsigned Flags = MCA.getELFHeaderEFlags(); unsigned Flags = MCA.getELFHeaderEFlags();

View File

@ -30,6 +30,7 @@ public:
virtual void emitDirectiveSetMicroMips(); virtual void emitDirectiveSetMicroMips();
virtual void emitDirectiveSetNoMicroMips(); virtual void emitDirectiveSetNoMicroMips();
virtual void setUsesMicroMips();
virtual void emitDirectiveSetMips16(); virtual void emitDirectiveSetMips16();
virtual void emitDirectiveSetNoMips16(); virtual void emitDirectiveSetNoMips16();
@ -290,6 +291,7 @@ public:
void emitDirectiveSetMicroMips() override; void emitDirectiveSetMicroMips() override;
void emitDirectiveSetNoMicroMips() override; void emitDirectiveSetNoMicroMips() override;
void setUsesMicroMips() override;
void emitDirectiveSetMips16() override; void emitDirectiveSetMips16() override;
void emitDirectiveSetNoReorder() override; void emitDirectiveSetNoReorder() override;

View File

@ -1,7 +1,12 @@
# RUN: llvm-mc -filetype=obj -triple mipsel-unknown-linux -mcpu=mips32 %s -o -| llvm-readobj -h | FileCheck %s # RUN: llvm-mc -filetype=obj -triple mipsel-unknown-linux -mcpu=mips32 \
# RUN: -mattr=+micromips < /dev/null -o -| llvm-readobj -h | FileCheck \
# RUN: -check-prefix=NO-MM %s
# RUN: llvm-mc -filetype=obj -triple mipsel-unknown-linux -mcpu=mips32 %s -o - \
# RUN: | llvm-readobj -h | FileCheck %s
# This *MUST* match the output of 'gcc -c' compiled with the same triple. # This *MUST* match the output of 'gcc -c' compiled with the same triple.
# CHECK: Flags [ (0x52001004) # CHECK: Flags [ (0x52001004)
# NO-MM: Flags [ (0x50001004)
.set micromips .set micromips
f: f:

View File

@ -0,0 +1,11 @@
# RUN: llvm-mc -filetype=obj -triple mipsel-unknown-linux -mcpu=mips32 %s -o - \
# RUN: | llvm-readobj -h | FileCheck %s
# This *MUST* match the output of 'gcc -c' compiled with the same triple.
# The microMIPS flag is not set if no microMIPS code was emitted.
# CHECK: Flags [ (0x50001004)
nop
.set micromips
.set nomicromips
nop