From a54df2503bbc4fee3d4d028a66891f5bf1af6bd3 Mon Sep 17 00:00:00 2001 From: Brian Gaeke Date: Fri, 19 Nov 2004 18:48:10 +0000 Subject: [PATCH] Fix bug in casting to long/ulong. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@18001 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/Target/Sparc/SparcV8ISelSimple.cpp | 38 +++++++++++++++++------- lib/Target/SparcV8/SparcV8ISelSimple.cpp | 38 +++++++++++++++++------- 2 files changed, 54 insertions(+), 22 deletions(-) diff --git a/lib/Target/Sparc/SparcV8ISelSimple.cpp b/lib/Target/Sparc/SparcV8ISelSimple.cpp index d26c77ee400..67cac56f1f5 100644 --- a/lib/Target/Sparc/SparcV8ISelSimple.cpp +++ b/lib/Target/Sparc/SparcV8ISelSimple.cpp @@ -67,9 +67,10 @@ namespace { /// emitIntegerCast, emitFPToIntegerCast - Helper methods for /// emitCastOperation. /// - void emitIntegerCast (MachineBasicBlock *BB, MachineBasicBlock::iterator IP, - const Type *oldTy, unsigned SrcReg, const Type *newTy, - unsigned DestReg); + unsigned emitIntegerCast (MachineBasicBlock *BB, + MachineBasicBlock::iterator IP, + const Type *oldTy, unsigned SrcReg, + const Type *newTy, unsigned DestReg); void emitFPToIntegerCast (MachineBasicBlock *BB, MachineBasicBlock::iterator IP, const Type *oldTy, unsigned SrcReg, const Type *newTy, @@ -563,14 +564,14 @@ void V8ISel::visitCastInst(CastInst &I) { } -void V8ISel::emitIntegerCast (MachineBasicBlock *BB, +unsigned V8ISel::emitIntegerCast (MachineBasicBlock *BB, MachineBasicBlock::iterator IP, const Type *oldTy, unsigned SrcReg, const Type *newTy, unsigned DestReg) { if (oldTy == newTy) { // No-op cast - just emit a copy; assume the reg. allocator will zap it. BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0).addReg(SrcReg); - return; + return SrcReg; } // Emit left-shift, then right-shift to sign- or zero-extend. unsigned TmpReg = makeAnotherReg (newTy); @@ -581,6 +582,8 @@ void V8ISel::emitIntegerCast (MachineBasicBlock *BB, } else { // zero-extend with SRL BuildMI(*BB, IP, V8::SRLri, 2, DestReg).addZImm (shiftWidth).addReg(TmpReg); } + // Return the temp reg. in case this is one half of a cast to long. + return TmpReg; } void V8ISel::emitFPToIntegerCast (MachineBasicBlock *BB, @@ -629,6 +632,10 @@ void V8ISel::emitCastOperation(MachineBasicBlock *BB, case cShort: case cInt: switch (oldTyClass) { + case cLong: + // Treat it like a cast from the lower half of the value. + emitIntegerCast (BB, IP, Type::IntTy, SrcReg+1, newTy, DestReg); + break; case cFloat: case cDouble: emitFPToIntegerCast (BB, IP, oldTy, SrcReg, newTy, DestReg); @@ -687,13 +694,22 @@ void V8ISel::emitCastOperation(MachineBasicBlock *BB, switch (oldTyClass) { case cByte: case cShort: - case cInt: - // Just copy it to the bottom half, and put a zero in the top half. - BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0) - .addReg (V8::G0); - BuildMI (*BB, IP, V8::ORrr, 2, DestReg+1).addReg (V8::G0) - .addReg (SrcReg); + case cInt: { + // Cast to (u)int in the bottom half, and sign(zero) extend in the top + // half. + const Type *OldHalfTy = oldTy->isSigned() ? Type::IntTy : Type::UIntTy; + const Type *NewHalfTy = newTy->isSigned() ? Type::IntTy : Type::UIntTy; + unsigned TempReg = emitIntegerCast (BB, IP, OldHalfTy, SrcReg, + NewHalfTy, DestReg+1); + if (newTy->isSigned ()) { + BuildMI (*BB, IP, V8::SRAri, 2, DestReg).addReg (TempReg) + .addZImm (31); + } else { + BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0) + .addReg (V8::G0); + } break; + } case cLong: // Just copy both halves. BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0).addReg (SrcReg); diff --git a/lib/Target/SparcV8/SparcV8ISelSimple.cpp b/lib/Target/SparcV8/SparcV8ISelSimple.cpp index d26c77ee400..67cac56f1f5 100644 --- a/lib/Target/SparcV8/SparcV8ISelSimple.cpp +++ b/lib/Target/SparcV8/SparcV8ISelSimple.cpp @@ -67,9 +67,10 @@ namespace { /// emitIntegerCast, emitFPToIntegerCast - Helper methods for /// emitCastOperation. /// - void emitIntegerCast (MachineBasicBlock *BB, MachineBasicBlock::iterator IP, - const Type *oldTy, unsigned SrcReg, const Type *newTy, - unsigned DestReg); + unsigned emitIntegerCast (MachineBasicBlock *BB, + MachineBasicBlock::iterator IP, + const Type *oldTy, unsigned SrcReg, + const Type *newTy, unsigned DestReg); void emitFPToIntegerCast (MachineBasicBlock *BB, MachineBasicBlock::iterator IP, const Type *oldTy, unsigned SrcReg, const Type *newTy, @@ -563,14 +564,14 @@ void V8ISel::visitCastInst(CastInst &I) { } -void V8ISel::emitIntegerCast (MachineBasicBlock *BB, +unsigned V8ISel::emitIntegerCast (MachineBasicBlock *BB, MachineBasicBlock::iterator IP, const Type *oldTy, unsigned SrcReg, const Type *newTy, unsigned DestReg) { if (oldTy == newTy) { // No-op cast - just emit a copy; assume the reg. allocator will zap it. BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0).addReg(SrcReg); - return; + return SrcReg; } // Emit left-shift, then right-shift to sign- or zero-extend. unsigned TmpReg = makeAnotherReg (newTy); @@ -581,6 +582,8 @@ void V8ISel::emitIntegerCast (MachineBasicBlock *BB, } else { // zero-extend with SRL BuildMI(*BB, IP, V8::SRLri, 2, DestReg).addZImm (shiftWidth).addReg(TmpReg); } + // Return the temp reg. in case this is one half of a cast to long. + return TmpReg; } void V8ISel::emitFPToIntegerCast (MachineBasicBlock *BB, @@ -629,6 +632,10 @@ void V8ISel::emitCastOperation(MachineBasicBlock *BB, case cShort: case cInt: switch (oldTyClass) { + case cLong: + // Treat it like a cast from the lower half of the value. + emitIntegerCast (BB, IP, Type::IntTy, SrcReg+1, newTy, DestReg); + break; case cFloat: case cDouble: emitFPToIntegerCast (BB, IP, oldTy, SrcReg, newTy, DestReg); @@ -687,13 +694,22 @@ void V8ISel::emitCastOperation(MachineBasicBlock *BB, switch (oldTyClass) { case cByte: case cShort: - case cInt: - // Just copy it to the bottom half, and put a zero in the top half. - BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0) - .addReg (V8::G0); - BuildMI (*BB, IP, V8::ORrr, 2, DestReg+1).addReg (V8::G0) - .addReg (SrcReg); + case cInt: { + // Cast to (u)int in the bottom half, and sign(zero) extend in the top + // half. + const Type *OldHalfTy = oldTy->isSigned() ? Type::IntTy : Type::UIntTy; + const Type *NewHalfTy = newTy->isSigned() ? Type::IntTy : Type::UIntTy; + unsigned TempReg = emitIntegerCast (BB, IP, OldHalfTy, SrcReg, + NewHalfTy, DestReg+1); + if (newTy->isSigned ()) { + BuildMI (*BB, IP, V8::SRAri, 2, DestReg).addReg (TempReg) + .addZImm (31); + } else { + BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0) + .addReg (V8::G0); + } break; + } case cLong: // Just copy both halves. BuildMI (*BB, IP, V8::ORrr, 2, DestReg).addReg (V8::G0).addReg (SrcReg);