[llvm] r245575 - [Sparc]: correct the 'set' synthetic instruction

David Majnemer via llvm-commits llvm-commits at lists.llvm.org
Thu Aug 20 18:22:35 PDT 2015


On Thu, Aug 20, 2015 at 9:16 AM, Douglas Katzman via llvm-commits <
llvm-commits at lists.llvm.org> wrote:

> Author: dougk
> Date: Thu Aug 20 11:16:16 2015
> New Revision: 245575
>
> URL: http://llvm.org/viewvc/llvm-project?rev=245575&view=rev
> Log:
> [Sparc]: correct the 'set' synthetic instruction
>
> Differential Revision: http://reviews.llvm.org/D12194
>
> Added:
>     llvm/trunk/test/MC/Sparc/sparc-asm-errors.s
> Modified:
>     llvm/trunk/lib/Target/Sparc/AsmParser/SparcAsmParser.cpp
>     llvm/trunk/test/MC/Sparc/sparc-synthetic-instructions.s
>
> Modified: llvm/trunk/lib/Target/Sparc/AsmParser/SparcAsmParser.cpp
> URL:
> http://llvm.org/viewvc/llvm-project/llvm/trunk/lib/Target/Sparc/AsmParser/SparcAsmParser.cpp?rev=245575&r1=245574&r2=245575&view=diff
>
> ==============================================================================
> --- llvm/trunk/lib/Target/Sparc/AsmParser/SparcAsmParser.cpp (original)
> +++ llvm/trunk/lib/Target/Sparc/AsmParser/SparcAsmParser.cpp Thu Aug 20
> 11:16:16 2015
> @@ -437,7 +437,22 @@ void SparcAsmParser::expandSET(MCInst &I
>
>    // the imm operand can be either an expression or an immediate.
>    bool IsImm = Inst.getOperand(1).isImm();
> -  uint64_t ImmValue = IsImm ? MCValOp.getImm() : 0;
> +  int64_t RawImmValue = IsImm ? MCValOp.getImm() : 0;
> +
> +  // Allow either a signed or unsigned 32-bit immediate.
> +  if (RawImmValue < -2147483648 || RawImmValue > 4294967295) {
>

Would it be correct to write this as "isInt<32>(RawImmValue) ||
isUInt<32>(RawImmValue)" ?


> +    Error(IDLoc, "set: argument must be between -2147483648 and
> 4294967295");
> +    return;
> +  }
> +
> +  // If the value was expressed as a large unsigned number, that's ok.
> +  // We want to see if it "looks like" a small signed number.
> +  int32_t ImmValue = RawImmValue;
> +  // For 'set' you can't use 'or' with a negative operand on V9 because
> +  // that would splat the sign bit across the upper half of the
> destination
> +  // register, whereas 'set' is defined to zero the high 32 bits.
> +  bool IsEffectivelyImm13 =
> +      IsImm && ((is64Bit() ? 0 : -4096) <= ImmValue && ImmValue < 4096);
>    const MCExpr *ValExpr;
>    if (IsImm)
>      ValExpr = MCConstantExpr::create(ImmValue, getContext());
> @@ -446,7 +461,10 @@ void SparcAsmParser::expandSET(MCInst &I
>
>    MCOperand PrevReg = MCOperand::createReg(Sparc::G0);
>
> -  if (!IsImm || (ImmValue & ~0x1fff)) {
> +  // If not just a signed imm13 value, then either we use a 'sethi' with a
> +  // following 'or', or a 'sethi' by itself if there are no more 1 bits.
> +  // In either case, start with the 'sethi'.
> +  if (!IsEffectivelyImm13) {
>      MCInst TmpInst;
>      const MCExpr *Expr =
>          SparcMCExpr::create(SparcMCExpr::VK_Sparc_HI, ValExpr,
> getContext());
> @@ -458,10 +476,24 @@ void SparcAsmParser::expandSET(MCInst &I
>      PrevReg = MCRegOp;
>    }
>
> -  if (!IsImm || ((ImmValue & 0x1fff) != 0 || ImmValue == 0)) {
> +  // The low bits require touching in 3 cases:
> +  // * A non-immediate value will always require both instructions.
> +  // * An effectively imm13 value needs only an 'or' instruction.
> +  // * Otherwise, an immediate that is not effectively imm13 requires the
> +  //   'or' only if bits remain after clearing the 22 bits that 'sethi'
> set.
> +  // If the low bits are known zeros, there's nothing to do.
> +  // In the second case, and only in that case, must we NOT clear
> +  // bits of the immediate value via the %lo() assembler function.
> +  // Note also, the 'or' instruction doesn't mind a large value in the
> case
> +  // where the operand to 'set' was 0xFFFFFzzz - it does exactly what you
> mean.
> +  if (!IsImm || IsEffectivelyImm13 || (ImmValue & 0x3ff)) {
>      MCInst TmpInst;
> -    const MCExpr *Expr =
> -        SparcMCExpr::create(SparcMCExpr::VK_Sparc_LO, ValExpr,
> getContext());
> +    const MCExpr *Expr;
> +    if (IsEffectivelyImm13)
> +      Expr = ValExpr;
> +    else
> +      Expr =
> +          SparcMCExpr::create(SparcMCExpr::VK_Sparc_LO, ValExpr,
> getContext());
>      TmpInst.setLoc(IDLoc);
>      TmpInst.setOpcode(SP::ORri);
>      TmpInst.addOperand(MCRegOp);
>
> Added: llvm/trunk/test/MC/Sparc/sparc-asm-errors.s
> URL:
> http://llvm.org/viewvc/llvm-project/llvm/trunk/test/MC/Sparc/sparc-asm-errors.s?rev=245575&view=auto
>
> ==============================================================================
> --- llvm/trunk/test/MC/Sparc/sparc-asm-errors.s (added)
> +++ llvm/trunk/test/MC/Sparc/sparc-asm-errors.s Thu Aug 20 11:16:16 2015
> @@ -0,0 +1,8 @@
> +! RUN: not llvm-mc %s -arch=sparc   -show-encoding 2>&1 | FileCheck %s
> +! RUN: not llvm-mc %s -arch=sparcv9 -show-encoding 2>&1 | FileCheck %s
> +
> +! Test the lower and upper bounds of 'set'
> +        ! CHECK: argument must be between
> +        set -2147483649, %o1
> +        ! CHECK: argument must be between
> +        set 4294967296, %o1
>
> Modified: llvm/trunk/test/MC/Sparc/sparc-synthetic-instructions.s
> URL:
> http://llvm.org/viewvc/llvm-project/llvm/trunk/test/MC/Sparc/sparc-synthetic-instructions.s?rev=245575&r1=245574&r2=245575&view=diff
>
> ==============================================================================
> --- llvm/trunk/test/MC/Sparc/sparc-synthetic-instructions.s (original)
> +++ llvm/trunk/test/MC/Sparc/sparc-synthetic-instructions.s Thu Aug 20
> 11:16:16 2015
> @@ -27,13 +27,33 @@
>          ! CHECK: or %g1, %lo(40000), %g1          ! encoding:
> [0x82,0x10,0b011000AA,A]
>          ! CHECK:                                  !   fixup A - offset:
> 0, value: %lo(40000), kind: fixup_sparc_lo10
>          set 40000, %g1
> -        ! CHECK: mov      %lo(1), %g1             ! encoding:
> [0x82,0x10,0b001000AA,A]
> -        ! CHECK:                                  !   fixup A - offset:
> 0, value: %lo(1), kind: fixup_sparc_lo10
> +        ! CHECK: mov 1, %g1 ! encoding: [0x82,0x10,0x20,0x01]
>          set 1, %g1
>          ! CHECK: sethi %hi(32768), %g1            ! encoding:
> [0x03,0b00AAAAAA,A,A]
>          ! CHECK:                                  !   fixup A - offset:
> 0, value: %hi(32768), kind: fixup_sparc_hi22
>          set 32768, %g1
>
> +        ! Expect a 'sethi' without an 'or'.
> +        ! CHECK: sethi %hi(268431360), %o1       ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! CHECK:                                 !   fixup A - offset: 0,
> value: %hi(268431360), kind: fixup_sparc_hi22
> +        set 0x0ffff000, %o1
> +
> +        ! CHECK: sethi %hi(268433408), %o1       ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! CHECK:                                 !   fixup A - offset: 0,
> value: %hi(268433408), kind: fixup_sparc_hi22
> +        set 0x0ffff800, %o1
> +
> +        ! This is the boundary case that uses the lowest of the 22 bits
> in sethi.
> +        ! CHECK: sethi %hi(268434432), %o1       ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! CHECK:                                 !   fixup A - offset: 0,
> value: %hi(268434432), kind: fixup_sparc_hi22
> +        set 0x0ffffc00, %o1
> +
> +        ! Now the synthetic instruction becomes two instructions.
> +        ! CHECK: sethi %hi(2147483647), %o1      ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! CHECK:                                 !   fixup A - offset: 0,
> value: %hi(2147483647), kind: fixup_sparc_hi22
> +        ! CHECK: or %o1, %lo(2147483647), %o1    ! encoding:
> [0x92,0x12,0b011000AA,A]
> +        ! CHECK:                                 !   fixup A - offset: 0,
> value: %lo(2147483647), kind: fixup_sparc_lo10
> +        set 2147483647, %o1
> +
>          ! CHECK: xnor %g1, %g0, %g2               ! encoding:
> [0x84,0x38,0x40,0x00]
>          not %g1, %g2
>          ! CHECK: xnor %g1, %g0, %g1               ! encoding:
> [0x82,0x38,0x40,0x00]
> @@ -143,3 +163,51 @@
>          wr %i0, %tbr
>          ! CHECK: wr %g0, 5, %tbr                  ! encoding:
> [0x81,0x98,0x20,0x05]
>          wr 5, %tbr
> +
> +! The following tests exercise 'set' in such a way that its output differs
> +! depending on whether targeting V8 or V9.
> +!
> +! RUN: llvm-mc %s -arch=sparc   -show-encoding | FileCheck %s
> --check-prefix=V8
> +! RUN: llvm-mc %s -arch=sparcv9 -show-encoding | FileCheck %s
> --check-prefix=V9
> +
> +        ! V8: mov        -1, %o1              ! encoding:
> [0x92,0x10,0x3f,0xff]
> +        ! V9: sethi %hi(-1), %o1              ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %hi(-1), kind: fixup_sparc_hi22
> +        ! V9: or %o1, %lo(-1), %o1            ! encoding:
> [0x92,0x12,0b011000AA,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %lo(-1), kind: fixup_sparc_lo10
> +        set 0xffffffff, %o1
> +
> +        ! V8: mov        -2, %o1              ! encoding:
> [0x92,0x10,0x3f,0xfe]
> +        ! V9: sethi %hi(-2), %o1              ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %hi(-2), kind: fixup_sparc_hi22
> +        ! V9: or %o1, %lo(-2), %o1            ! encoding:
> [0x92,0x12,0b011000AA,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %lo(-2), kind: fixup_sparc_lo10
> +        set 0xfffffffe, %o1
> +
> +        ! V8: mov        -16, %o1             ! encoding:
> [0x92,0x10,0x3f,0xf0]
> +        ! V9: sethi %hi(-16), %o1             ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %hi(-16), kind: fixup_sparc_hi22
> +        ! V9: or %o1, %lo(-16), %o1           ! encoding:
> [0x92,0x12,0b011000AA,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %lo(-16), kind: fixup_sparc_lo10
> +        set 0xfffffff0, %o1
> +
> +        ! V8: mov        -256, %o1            ! encoding:
> [0x92,0x10,0x3f,0x00]
> +        ! V9: sethi %hi(-256), %o1            ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %hi(-256), kind: fixup_sparc_hi22
> +        ! V9: or %o1, %lo(-256), %o1          ! encoding:
> [0x92,0x12,0b011000AA,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %lo(-256), kind: fixup_sparc_lo10
> +        set 0xffffff00, %o1
> +
> +        ! V8: mov        -4096, %o1           ! encoding:
> [0x92,0x10,0x30,0x00]
> +        ! V9: sethi %hi(-4096), %o1           ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %hi(-4096), kind: fixup_sparc_hi22
> +        set 0xfffff000, %o1
> +
> +        ! These results are the same for V8 and V9, so this test could
> have
> +        ! been with the others that weren't segregated by architecture,
> +        ! but logically it belongs here as a boundary case.
> +        ! V8: sethi %hi(-8192), %o1           ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V8:                                 !   fixup A - offset: 0,
> value: %hi(-8192), kind: fixup_sparc_hi22
> +        ! V9: sethi %hi(-8192), %o1           ! encoding:
> [0x13,0b00AAAAAA,A,A]
> +        ! V9:                                 !   fixup A - offset: 0,
> value: %hi(-8192), kind: fixup_sparc_hi22
> +        set 0xffffe000, %o1
>
>
> _______________________________________________
> llvm-commits mailing list
> llvm-commits at lists.llvm.org
> http://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-commits
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20150820/828fdd20/attachment.html>


More information about the llvm-commits mailing list