mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-10-25 10:27:04 +00:00 
			
		
		
		
	Fix PR3486. Fix a bug in code that manually patch physical register live interval after its sub-register is coalesced with a virtual register.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@64082 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
		| @@ -283,6 +283,10 @@ namespace llvm { | |||||||
|       return *I->second; |       return *I->second; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// dupInterval - Duplicate a live interval. The caller is responsible for | ||||||
|  |     /// managing the allocated memory. | ||||||
|  |     LiveInterval *dupInterval(LiveInterval *li); | ||||||
|  |      | ||||||
|     /// addLiveRangeToEndOfBlock - Given a register and an instruction, |     /// addLiveRangeToEndOfBlock - Given a register and an instruction, | ||||||
|     /// adds a live range from that instruction to the end of its MBB. |     /// adds a live range from that instruction to the end of its MBB. | ||||||
|     LiveRange addLiveRangeToEndOfBlock(unsigned reg, |     LiveRange addLiveRangeToEndOfBlock(unsigned reg, | ||||||
|   | |||||||
| @@ -824,11 +824,18 @@ bool LiveIntervals::findReachableMBBs(unsigned Start, unsigned End, | |||||||
| } | } | ||||||
|  |  | ||||||
| LiveInterval* LiveIntervals::createInterval(unsigned reg) { | LiveInterval* LiveIntervals::createInterval(unsigned reg) { | ||||||
|   float Weight = TargetRegisterInfo::isPhysicalRegister(reg) ? |   float Weight = TargetRegisterInfo::isPhysicalRegister(reg) ? HUGE_VALF : 0.0F; | ||||||
|                        HUGE_VALF : 0.0F; |  | ||||||
|   return new LiveInterval(reg, Weight); |   return new LiveInterval(reg, Weight); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// dupInterval - Duplicate a live interval. The caller is responsible for | ||||||
|  | /// managing the allocated memory. | ||||||
|  | LiveInterval* LiveIntervals::dupInterval(LiveInterval *li) { | ||||||
|  |   LiveInterval *NewLI = createInterval(li->reg); | ||||||
|  |   NewLI->Copy(*li, getVNInfoAllocator()); | ||||||
|  |   return NewLI; | ||||||
|  | } | ||||||
|  |  | ||||||
| /// getVNInfoSourceReg - Helper function that parses the specified VNInfo | /// getVNInfoSourceReg - Helper function that parses the specified VNInfo | ||||||
| /// copy field and returns the source register that defines it. | /// copy field and returns the source register that defines it. | ||||||
| unsigned LiveIntervals::getVNInfoSourceReg(const VNInfo *VNI) const { | unsigned LiveIntervals::getVNInfoSourceReg(const VNInfo *VNI) const { | ||||||
|   | |||||||
| @@ -1354,6 +1354,15 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) { | |||||||
|   DOUT << " and "; DstInt.print(DOUT, tri_); |   DOUT << " and "; DstInt.print(DOUT, tri_); | ||||||
|   DOUT << ": "; |   DOUT << ": "; | ||||||
|  |  | ||||||
|  |   // Save a copy of the virtual register live interval. We'll manually | ||||||
|  |   // merge this into the "real" physical register live interval this is | ||||||
|  |   // coalesced with. | ||||||
|  |   LiveInterval *SavedLI = 0; | ||||||
|  |   if (RealDstReg) | ||||||
|  |     SavedLI = li_->dupInterval(&SrcInt); | ||||||
|  |   else if (RealSrcReg) | ||||||
|  |     SavedLI = li_->dupInterval(&DstInt); | ||||||
|  |  | ||||||
|   // Check if it is necessary to propagate "isDead" property. |   // Check if it is necessary to propagate "isDead" property. | ||||||
|   if (!isExtSubReg && !isInsSubReg) { |   if (!isExtSubReg && !isInsSubReg) { | ||||||
|     MachineOperand *mopd = CopyMI->findRegisterDefOperand(DstReg, false); |     MachineOperand *mopd = CopyMI->findRegisterDefOperand(DstReg, false); | ||||||
| @@ -1445,21 +1454,17 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) { | |||||||
|     if (RealDstReg || RealSrcReg) { |     if (RealDstReg || RealSrcReg) { | ||||||
|       LiveInterval &RealInt = |       LiveInterval &RealInt = | ||||||
|         li_->getOrCreateInterval(RealDstReg ? RealDstReg : RealSrcReg); |         li_->getOrCreateInterval(RealDstReg ? RealDstReg : RealSrcReg); | ||||||
|       SmallSet<const VNInfo*, 4> CopiedValNos; |       for (LiveInterval::const_vni_iterator I = SavedLI->vni_begin(), | ||||||
|       for (LiveInterval::Ranges::const_iterator I = ResSrcInt->ranges.begin(), |              E = SavedLI->vni_end(); I != E; ++I) { | ||||||
|              E = ResSrcInt->ranges.end(); I != E; ++I) { |         const VNInfo *ValNo = *I; | ||||||
|         const LiveRange *DstLR = ResDstInt->getLiveRangeContaining(I->start); |         VNInfo *NewValNo = RealInt.getNextValue(ValNo->def, ValNo->copy, | ||||||
|         assert(DstLR  && "Invalid joined interval!"); |                                                 li_->getVNInfoAllocator()); | ||||||
|         const VNInfo *DstValNo = DstLR->valno; |         NewValNo->hasPHIKill = ValNo->hasPHIKill; | ||||||
|         if (CopiedValNos.insert(DstValNo)) { |         NewValNo->redefByEC = ValNo->redefByEC; | ||||||
|           VNInfo *ValNo = RealInt.getNextValue(DstValNo->def, DstValNo->copy, |         RealInt.addKills(NewValNo, ValNo->kills); | ||||||
|                                                li_->getVNInfoAllocator()); |         RealInt.MergeValueInAsValue(*SavedLI, ValNo, NewValNo); | ||||||
|           ValNo->hasPHIKill = DstValNo->hasPHIKill; |  | ||||||
|           RealInt.addKills(ValNo, DstValNo->kills); |  | ||||||
|           RealInt.MergeValueInAsValue(*ResDstInt, DstValNo, ValNo); |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|        |       RealInt.weight += SavedLI->weight;       | ||||||
|       DstReg = RealDstReg ? RealDstReg : RealSrcReg; |       DstReg = RealDstReg ? RealDstReg : RealSrcReg; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1529,6 +1534,12 @@ bool SimpleRegisterCoalescing::JoinCopy(CopyRec &TheCopy, bool &Again) { | |||||||
|   // being merged. |   // being merged. | ||||||
|   li_->removeInterval(SrcReg); |   li_->removeInterval(SrcReg); | ||||||
|  |  | ||||||
|  |   // Manually deleted the live interval copy. | ||||||
|  |   if (SavedLI) { | ||||||
|  |     SavedLI->clear(); | ||||||
|  |     delete SavedLI; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   if (isEmpty) { |   if (isEmpty) { | ||||||
|     // Now the copy is being coalesced away, the val# previously defined |     // Now the copy is being coalesced away, the val# previously defined | ||||||
|     // by the copy is being defined by an IMPLICIT_DEF which defines a zero |     // by the copy is being defined by an IMPLICIT_DEF which defines a zero | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								test/CodeGen/X86/2009-02-08-CoalescerBug.ll
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								test/CodeGen/X86/2009-02-08-CoalescerBug.ll
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | ; RUN: llvm-as < %s | llc -march=x86 | ||||||
|  | ; PR3486 | ||||||
|  |  | ||||||
|  | define i32 @foo(i8 signext %p_26) nounwind { | ||||||
|  | entry: | ||||||
|  | 	%0 = icmp eq i8 %p_26, 0		; <i1> [#uses=2] | ||||||
|  | 	%or.cond = or i1 false, %0		; <i1> [#uses=2] | ||||||
|  | 	%iftmp.1.0 = zext i1 %or.cond to i16		; <i16> [#uses=1] | ||||||
|  | 	br i1 %0, label %bb.i, label %bar.exit | ||||||
|  |  | ||||||
|  | bb.i:		; preds = %entry | ||||||
|  | 	%1 = zext i1 %or.cond to i32		; <i32> [#uses=1] | ||||||
|  | 	%2 = sdiv i32 %1, 0		; <i32> [#uses=1] | ||||||
|  | 	%3 = trunc i32 %2 to i16		; <i16> [#uses=1] | ||||||
|  | 	br label %bar.exit | ||||||
|  |  | ||||||
|  | bar.exit:		; preds = %bb.i, %entry | ||||||
|  | 	%4 = phi i16 [ %3, %bb.i ], [ %iftmp.1.0, %entry ]		; <i16> [#uses=1] | ||||||
|  | 	%5 = trunc i16 %4 to i8		; <i8> [#uses=1] | ||||||
|  | 	%6 = sext i8 %5 to i32		; <i32> [#uses=1] | ||||||
|  | 	ret i32 %6 | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user