From dde19c01441be3aec285967d6a9aa279332597f1 Mon Sep 17 00:00:00 2001 From: "J. Gareth \"Curious Kit\" Moreton" Date: Thu, 27 Jul 2023 19:24:03 +0100 Subject: [PATCH] * Improvement to TEST/JNE/TEST/JNE code to be more accurate where register deallocations are concerned --- compiler/x86/aoptx86.pas | 24 +++++++++++++++++++----- 1 file changed, 19 insertions(+), 5 deletions(-) diff --git a/compiler/x86/aoptx86.pas b/compiler/x86/aoptx86.pas index fa745959ae..c0b87c1f39 100644 --- a/compiler/x86/aoptx86.pas +++ b/compiler/x86/aoptx86.pas @@ -5501,15 +5501,29 @@ unit aoptx86; if IsJumpToLabel(taicpu(hp1_dist)) then TAsmLabel(taicpu(hp1_dist).oper[0]^.ref^.symbol).DecRefs; - DebugMsg(SPeepholeOptimization + 'TEST/JNE/TEST/JNE merged', p); - RemoveInstruction(hp1_dist); - { Only remove the second test if no jumps or other conditional instructions follow } TransferUsedRegs(TmpUsedRegs); UpdateUsedRegs(TmpUsedRegs, tai(p.Next)); UpdateUsedRegs(TmpUsedRegs, tai(hp1.Next)); - if not RegUsedAfterInstruction(NR_DEFAULTFLAGS, p_dist, TmpUsedRegs) then - RemoveInstruction(p_dist); + UpdateUsedRegs(TmpUsedRegs, tai(p_dist.Next)); + if not RegUsedAfterInstruction(NR_DEFAULTFLAGS, hp1_dist, TmpUsedRegs) then + begin + DebugMsg(SPeepholeOptimization + 'TEST/JNE/TEST/JNE merged', p); + RemoveInstruction(p_dist); + + { Remove the first jump, not the second, to keep + any register deallocations between the second + TEST/JNE pair in the same place. Aids future + optimisation. } + RemoveInstruction(hp1); + end + else + begin + DebugMsg(SPeepholeOptimization + 'TEST/JNE/TEST/JNE merged (second TEST preserved)', p); + + { Remove second jump in this instance } + RemoveInstruction(hp1_dist); + end; Result := True; Exit;