mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-21 03:32:21 +00:00
2e2efd9600
- CodeGenPrepare pass for identifying div/rem ops - Backend specifies the type mapping using addBypassSlowDivType - Enabled only for Intel Atom with O2 32-bit -> 8-bit - Replace IDIV with instructions which test its value and use DIVB if the value is positive and less than 256. - In the case when the quotient and remainder of a divide are used a DIV and a REM instruction will be present in the IR. In the non-Atom case they are both lowered to IDIVs and CSE removes the redundant IDIV instruction, using the quotient and remainder from the first IDIV. However, due to this optimization CSE is not able to eliminate redundant IDIV instructions because they are located in different basic blocks. This is overcome by calculating both the quotient (DIV) and remainder (REM) in each basic block that is inserted by the optimization and reusing the result values when a subsequent DIV or REM instruction uses the same operands. - Test cases check for the presents of the optimization when calculating either the quotient, remainder, or both. Patch by Tyler Nowicki! git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163150 91177308-0d34-0410-b5e6-96231b3b80d8
113 lines
2.4 KiB
LLVM
113 lines
2.4 KiB
LLVM
; RUN: llc < %s -mcpu=atom -mtriple=i686-linux | FileCheck %s
|
|
|
|
define i32 @test_get_quotient(i32 %a, i32 %b) nounwind {
|
|
; CHECK: test_get_quotient
|
|
; CHECK: orl %ecx, %edx
|
|
; CHECK-NEXT: testl $-256, %edx
|
|
; CHECK-NEXT: je
|
|
; CHECK: idivl
|
|
; CHECK: ret
|
|
; CHECK: divb
|
|
; CHECK: ret
|
|
%result = sdiv i32 %a, %b
|
|
ret i32 %result
|
|
}
|
|
|
|
define i32 @test_get_remainder(i32 %a, i32 %b) nounwind {
|
|
; CHECK: test_get_remainder
|
|
; CHECK: orl %ecx, %edx
|
|
; CHECK-NEXT: testl $-256, %edx
|
|
; CHECK-NEXT: je
|
|
; CHECK: idivl
|
|
; CHECK: ret
|
|
; CHECK: divb
|
|
; CHECK: ret
|
|
%result = srem i32 %a, %b
|
|
ret i32 %result
|
|
}
|
|
|
|
define i32 @test_get_quotient_and_remainder(i32 %a, i32 %b) nounwind {
|
|
; CHECK: test_get_quotient_and_remainder
|
|
; CHECK: orl %ecx, %edx
|
|
; CHECK-NEXT: testl $-256, %edx
|
|
; CHECK-NEXT: je
|
|
; CHECK: idivl
|
|
; CHECK: divb
|
|
; CHECK: addl
|
|
; CHECK: ret
|
|
; CEECK-NOT: idivl
|
|
; CHECK-NOT: divb
|
|
%resultdiv = sdiv i32 %a, %b
|
|
%resultrem = srem i32 %a, %b
|
|
%result = add i32 %resultdiv, %resultrem
|
|
ret i32 %result
|
|
}
|
|
|
|
define i32 @test_use_div_and_idiv(i32 %a, i32 %b) nounwind {
|
|
; CHECK: test_use_div_and_idiv
|
|
; CHECK: idivl
|
|
; CHECK: divb
|
|
; CHECK: divl
|
|
; CHECK: divb
|
|
; CHECK: addl
|
|
; CHECK: ret
|
|
%resultidiv = sdiv i32 %a, %b
|
|
%resultdiv = udiv i32 %a, %b
|
|
%result = add i32 %resultidiv, %resultdiv
|
|
ret i32 %result
|
|
}
|
|
|
|
define i32 @test_use_div_imm_imm() nounwind {
|
|
; CHECK: test_use_div_imm_imm
|
|
; CHECK: movl $64
|
|
%resultdiv = sdiv i32 256, 4
|
|
ret i32 %resultdiv
|
|
}
|
|
|
|
define i32 @test_use_div_reg_imm(i32 %a) nounwind {
|
|
; CHECK: test_use_div_reg_imm
|
|
; CEHCK-NOT: test
|
|
; CHECK-NOT: idiv
|
|
; CHECK-NOT: divb
|
|
%resultdiv = sdiv i32 %a, 33
|
|
ret i32 %resultdiv
|
|
}
|
|
|
|
define i32 @test_use_rem_reg_imm(i32 %a) nounwind {
|
|
; CHECK: test_use_rem_reg_imm
|
|
; CEHCK-NOT: test
|
|
; CHECK-NOT: idiv
|
|
; CHECK-NOT: divb
|
|
%resultrem = srem i32 %a, 33
|
|
ret i32 %resultrem
|
|
}
|
|
|
|
define i32 @test_use_divrem_reg_imm(i32 %a) nounwind {
|
|
; CHECK: test_use_divrem_reg_imm
|
|
; CEHCK-NOT: test
|
|
; CHECK-NOT: idiv
|
|
; CHECK-NOT: divb
|
|
%resultdiv = sdiv i32 %a, 33
|
|
%resultrem = srem i32 %a, 33
|
|
%result = add i32 %resultdiv, %resultrem
|
|
ret i32 %result
|
|
}
|
|
|
|
define i32 @test_use_div_imm_reg(i32 %a) nounwind {
|
|
; CHECK: test_use_div_imm_reg
|
|
; CHECK: test
|
|
; CHECK: idiv
|
|
; CHECK: divb
|
|
%resultdiv = sdiv i32 4, %a
|
|
ret i32 %resultdiv
|
|
}
|
|
|
|
define i32 @test_use_rem_imm_reg(i32 %a) nounwind {
|
|
; CHECK: test_use_rem_imm_reg
|
|
; CHECK: test
|
|
; CHECK: idiv
|
|
; CHECK: divb
|
|
%resultdiv = sdiv i32 4, %a
|
|
ret i32 %resultdiv
|
|
}
|