2016-10-09 21:07:25 +08:00
|
|
|
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
|
2018-06-07 18:13:09 +08:00
|
|
|
; RUN: llc < %s -mtriple=i686-unknown-linux | FileCheck %s --check-prefixes=X86
|
|
|
|
; RUN: llc < %s -mtriple=x86_64-unknown-linux | FileCheck %s --check-prefixes=X64
|
2016-10-09 21:07:25 +08:00
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i64 @rotl64(i64 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl64:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: pushl %ebx
|
|
|
|
; X86-NEXT: pushl %edi
|
|
|
|
; X86-NEXT: pushl %esi
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %esi
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %edi
|
|
|
|
; X86-NEXT: movl %esi, %eax
|
|
|
|
; X86-NEXT: shll %cl, %eax
|
|
|
|
; X86-NEXT: movl %edi, %edx
|
|
|
|
; X86-NEXT: shldl %cl, %esi, %edx
|
|
|
|
; X86-NEXT: testb $32, %cl
|
|
|
|
; X86-NEXT: je .LBB0_2
|
|
|
|
; X86-NEXT: # %bb.1:
|
|
|
|
; X86-NEXT: movl %eax, %edx
|
|
|
|
; X86-NEXT: xorl %eax, %eax
|
|
|
|
; X86-NEXT: .LBB0_2:
|
|
|
|
; X86-NEXT: movb $64, %ch
|
|
|
|
; X86-NEXT: subb %cl, %ch
|
|
|
|
; X86-NEXT: movl %edi, %ebx
|
|
|
|
; X86-NEXT: movb %ch, %cl
|
|
|
|
; X86-NEXT: shrl %cl, %ebx
|
|
|
|
; X86-NEXT: shrdl %cl, %edi, %esi
|
|
|
|
; X86-NEXT: testb $32, %ch
|
|
|
|
; X86-NEXT: je .LBB0_4
|
|
|
|
; X86-NEXT: # %bb.3:
|
|
|
|
; X86-NEXT: movl %ebx, %esi
|
|
|
|
; X86-NEXT: xorl %ebx, %ebx
|
|
|
|
; X86-NEXT: .LBB0_4:
|
|
|
|
; X86-NEXT: orl %ebx, %edx
|
|
|
|
; X86-NEXT: orl %esi, %eax
|
|
|
|
; X86-NEXT: popl %esi
|
|
|
|
; X86-NEXT: popl %edi
|
|
|
|
; X86-NEXT: popl %ebx
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl64:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movq %rdi, %rax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rolq %cl, %rax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%shift.upgrd.1 = zext i8 %Amt to i64
|
|
|
|
%B = shl i64 %A, %shift.upgrd.1
|
|
|
|
%Amt2 = sub i8 64, %Amt
|
|
|
|
%shift.upgrd.2 = zext i8 %Amt2 to i64
|
|
|
|
%C = lshr i64 %A, %shift.upgrd.2
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
ret i64 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i64 @rotr64(i64 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr64:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: pushl %ebx
|
|
|
|
; X86-NEXT: pushl %edi
|
|
|
|
; X86-NEXT: pushl %esi
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %edi
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %esi
|
|
|
|
; X86-NEXT: movl %esi, %edx
|
|
|
|
; X86-NEXT: shrl %cl, %edx
|
|
|
|
; X86-NEXT: movl %edi, %eax
|
|
|
|
; X86-NEXT: shrdl %cl, %esi, %eax
|
|
|
|
; X86-NEXT: testb $32, %cl
|
|
|
|
; X86-NEXT: je .LBB1_2
|
|
|
|
; X86-NEXT: # %bb.1:
|
|
|
|
; X86-NEXT: movl %edx, %eax
|
|
|
|
; X86-NEXT: xorl %edx, %edx
|
|
|
|
; X86-NEXT: .LBB1_2:
|
|
|
|
; X86-NEXT: movb $64, %ch
|
|
|
|
; X86-NEXT: subb %cl, %ch
|
|
|
|
; X86-NEXT: movl %edi, %ebx
|
|
|
|
; X86-NEXT: movb %ch, %cl
|
|
|
|
; X86-NEXT: shll %cl, %ebx
|
|
|
|
; X86-NEXT: shldl %cl, %edi, %esi
|
|
|
|
; X86-NEXT: testb $32, %ch
|
|
|
|
; X86-NEXT: je .LBB1_4
|
|
|
|
; X86-NEXT: # %bb.3:
|
|
|
|
; X86-NEXT: movl %ebx, %esi
|
|
|
|
; X86-NEXT: xorl %ebx, %ebx
|
|
|
|
; X86-NEXT: .LBB1_4:
|
|
|
|
; X86-NEXT: orl %esi, %edx
|
|
|
|
; X86-NEXT: orl %ebx, %eax
|
|
|
|
; X86-NEXT: popl %esi
|
|
|
|
; X86-NEXT: popl %edi
|
|
|
|
; X86-NEXT: popl %ebx
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr64:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movq %rdi, %rax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rorq %cl, %rax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%shift.upgrd.3 = zext i8 %Amt to i64
|
|
|
|
%B = lshr i64 %A, %shift.upgrd.3
|
|
|
|
%Amt2 = sub i8 64, %Amt
|
|
|
|
%shift.upgrd.4 = zext i8 %Amt2 to i64
|
|
|
|
%C = shl i64 %A, %shift.upgrd.4
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
ret i64 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i64 @rotli64(i64 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotli64:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %ecx
|
|
|
|
; X86-NEXT: movl %ecx, %edx
|
|
|
|
; X86-NEXT: shldl $5, %eax, %edx
|
|
|
|
; X86-NEXT: shldl $5, %ecx, %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotli64:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movq %rdi, %rax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolq $5, %rax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i64 %A, 5
|
|
|
|
%C = lshr i64 %A, 59
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
ret i64 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i64 @rotri64(i64 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotri64:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %edx
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %ecx
|
|
|
|
; X86-NEXT: movl %ecx, %eax
|
|
|
|
; X86-NEXT: shldl $27, %edx, %eax
|
|
|
|
; X86-NEXT: shldl $27, %ecx, %edx
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotri64:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movq %rdi, %rax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolq $59, %rax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i64 %A, 5
|
|
|
|
%C = shl i64 %A, 59
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
ret i64 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i64 @rotl1_64(i64 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl1_64:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %ecx
|
|
|
|
; X86-NEXT: movl %ecx, %edx
|
|
|
|
; X86-NEXT: shldl $1, %eax, %edx
|
|
|
|
; X86-NEXT: shldl $1, %ecx, %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl1_64:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movq %rdi, %rax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolq %rax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i64 %A, 1
|
|
|
|
%C = lshr i64 %A, 63
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
ret i64 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i64 @rotr1_64(i64 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_64:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %edx
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %ecx
|
|
|
|
; X86-NEXT: movl %ecx, %eax
|
|
|
|
; X86-NEXT: shldl $31, %edx, %eax
|
|
|
|
; X86-NEXT: shldl $31, %ecx, %edx
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_64:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movq %rdi, %rax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rorq %rax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i64 %A, 63
|
|
|
|
%C = lshr i64 %A, 1
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
ret i64 %D
|
|
|
|
}
|
2005-01-19 16:04:08 +08:00
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i32 @rotl32(i32 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl32:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: roll %cl, %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl32:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: roll %cl, %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%shift.upgrd.1 = zext i8 %Amt to i32
|
|
|
|
%B = shl i32 %A, %shift.upgrd.1
|
|
|
|
%Amt2 = sub i8 32, %Amt
|
|
|
|
%shift.upgrd.2 = zext i8 %Amt2 to i32
|
|
|
|
%C = lshr i32 %A, %shift.upgrd.2
|
|
|
|
%D = or i32 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i32 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i32 @rotr32(i32 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr32:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorl %cl, %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr32:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rorl %cl, %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%shift.upgrd.3 = zext i8 %Amt to i32
|
|
|
|
%B = lshr i32 %A, %shift.upgrd.3
|
|
|
|
%Amt2 = sub i8 32, %Amt
|
|
|
|
%shift.upgrd.4 = zext i8 %Amt2 to i32
|
|
|
|
%C = shl i32 %A, %shift.upgrd.4
|
|
|
|
%D = or i32 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i32 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i32 @rotli32(i32 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotli32:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: roll $5, %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotli32:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: roll $5, %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i32 %A, 5
|
|
|
|
%C = lshr i32 %A, 27
|
|
|
|
%D = or i32 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i32 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i32 @rotri32(i32 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotri32:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: roll $27, %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotri32:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: roll $27, %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i32 %A, 5
|
|
|
|
%C = shl i32 %A, 27
|
|
|
|
%D = or i32 %B, %C
|
|
|
|
ret i32 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i32 @rotl1_32(i32 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl1_32:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: roll %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl1_32:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: roll %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i32 %A, 1
|
|
|
|
%C = lshr i32 %A, 31
|
|
|
|
%D = or i32 %B, %C
|
|
|
|
ret i32 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i32 @rotr1_32(i32 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_32:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorl %eax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_32:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rorl %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i32 %A, 31
|
|
|
|
%C = lshr i32 %A, 1
|
|
|
|
%D = or i32 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i32 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i16 @rotl16(i16 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl16:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movzwl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rolw %cl, %ax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl16:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rolw %cl, %ax
|
|
|
|
; X64-NEXT: # kill: def $ax killed $ax killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%shift.upgrd.5 = zext i8 %Amt to i16
|
|
|
|
%B = shl i16 %A, %shift.upgrd.5
|
|
|
|
%Amt2 = sub i8 16, %Amt
|
|
|
|
%shift.upgrd.6 = zext i8 %Amt2 to i16
|
|
|
|
%C = lshr i16 %A, %shift.upgrd.6
|
|
|
|
%D = or i16 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i16 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i16 @rotr16(i16 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr16:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movzwl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorw %cl, %ax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr16:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rorw %cl, %ax
|
|
|
|
; X64-NEXT: # kill: def $ax killed $ax killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%shift.upgrd.7 = zext i8 %Amt to i16
|
|
|
|
%B = lshr i16 %A, %shift.upgrd.7
|
|
|
|
%Amt2 = sub i8 16, %Amt
|
|
|
|
%shift.upgrd.8 = zext i8 %Amt2 to i16
|
|
|
|
%C = shl i16 %A, %shift.upgrd.8
|
|
|
|
%D = or i16 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i16 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i16 @rotli16(i16 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotli16:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movzwl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rolw $5, %ax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotli16:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolw $5, %ax
|
|
|
|
; X64-NEXT: # kill: def $ax killed $ax killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i16 %A, 5
|
|
|
|
%C = lshr i16 %A, 11
|
|
|
|
%D = or i16 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i16 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i16 @rotri16(i16 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotri16:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movzwl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rolw $11, %ax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotri16:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolw $11, %ax
|
|
|
|
; X64-NEXT: # kill: def $ax killed $ax killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i16 %A, 5
|
|
|
|
%C = shl i16 %A, 11
|
|
|
|
%D = or i16 %B, %C
|
|
|
|
ret i16 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i16 @rotl1_16(i16 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl1_16:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movzwl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rolw %ax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl1_16:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolw %ax
|
|
|
|
; X64-NEXT: # kill: def $ax killed $ax killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i16 %A, 1
|
|
|
|
%C = lshr i16 %A, 15
|
|
|
|
%D = or i16 %B, %C
|
|
|
|
ret i16 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i16 @rotr1_16(i16 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_16:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movzwl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorw %ax
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_16:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rorw %ax
|
|
|
|
; X64-NEXT: # kill: def $ax killed $ax killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i16 %A, 1
|
|
|
|
%C = shl i16 %A, 15
|
|
|
|
%D = or i16 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i16 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i8 @rotl8(i8 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl8:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %al
|
|
|
|
; X86-NEXT: rolb %cl, %al
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl8:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rolb %cl, %al
|
|
|
|
; X64-NEXT: # kill: def $al killed $al killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i8 %A, %Amt
|
|
|
|
%Amt2 = sub i8 8, %Amt
|
|
|
|
%C = lshr i8 %A, %Amt2
|
|
|
|
%D = or i8 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i8 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i8 @rotr8(i8 %A, i8 %Amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr8:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %al
|
|
|
|
; X86-NEXT: rorb %cl, %al
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr8:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
|
|
|
; X64-NEXT: rorb %cl, %al
|
|
|
|
; X64-NEXT: # kill: def $al killed $al killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i8 %A, %Amt
|
|
|
|
%Amt2 = sub i8 8, %Amt
|
|
|
|
%C = shl i8 %A, %Amt2
|
|
|
|
%D = or i8 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i8 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i8 @rotli8(i8 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotli8:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %al
|
|
|
|
; X86-NEXT: rolb $5, %al
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotli8:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolb $5, %al
|
|
|
|
; X64-NEXT: # kill: def $al killed $al killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i8 %A, 5
|
|
|
|
%C = lshr i8 %A, 3
|
|
|
|
%D = or i8 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i8 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i8 @rotri8(i8 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotri8:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %al
|
|
|
|
; X86-NEXT: rolb $3, %al
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotri8:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolb $3, %al
|
|
|
|
; X64-NEXT: # kill: def $al killed $al killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i8 %A, 5
|
|
|
|
%C = shl i8 %A, 3
|
|
|
|
%D = or i8 %B, %C
|
|
|
|
ret i8 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i8 @rotl1_8(i8 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotl1_8:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %al
|
|
|
|
; X86-NEXT: rolb %al
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotl1_8:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rolb %al
|
|
|
|
; X64-NEXT: # kill: def $al killed $al killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = shl i8 %A, 1
|
|
|
|
%C = lshr i8 %A, 7
|
|
|
|
%D = or i8 %B, %C
|
|
|
|
ret i8 %D
|
|
|
|
}
|
|
|
|
|
2016-10-09 21:33:51 +08:00
|
|
|
define i8 @rotr1_8(i8 %A) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_8:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %al
|
|
|
|
; X86-NEXT: rorb %al
|
|
|
|
; X86-NEXT: retl
|
2016-10-09 21:07:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_8:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: rorb %al
|
|
|
|
; X64-NEXT: # kill: def $al killed $al killed $eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: retq
|
2016-10-09 21:07:25 +08:00
|
|
|
%B = lshr i8 %A, 1
|
|
|
|
%C = shl i8 %A, 7
|
|
|
|
%D = or i8 %B, %C
|
2008-02-21 15:42:26 +08:00
|
|
|
ret i8 %D
|
2005-01-19 16:04:08 +08:00
|
|
|
}
|
2017-02-20 08:37:20 +08:00
|
|
|
|
|
|
|
define void @rotr1_64_mem(i64* %Aptr) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_64_mem:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: pushl %esi
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: movl (%eax), %ecx
|
|
|
|
; X86-NEXT: movl 4(%eax), %edx
|
|
|
|
; X86-NEXT: movl %edx, %esi
|
|
|
|
; X86-NEXT: shldl $31, %ecx, %esi
|
|
|
|
; X86-NEXT: shldl $31, %edx, %ecx
|
|
|
|
; X86-NEXT: movl %esi, (%eax)
|
[DAGCombiner] If a TokenFactor would be merged into its user, consider the user later.
Summary:
A number of optimizations are inhibited by single-use TokenFactors not
being merged into the TokenFactor using it. This makes we consider if
we can do the merge immediately.
Most tests changes here are due to the change in visitation causing
minor reorderings and associated reassociation of paired memory
operations.
CodeGen tests with non-reordering changes:
X86/aligned-variadic.ll -- memory-based add folded into stored leaq
value.
X86/constant-combiners.ll -- Optimizes out overlap between stores.
X86/pr40631_deadstore_elision -- folds constant byte store into
preceding quad word constant store.
Reviewers: RKSimon, craig.topper, spatel, efriedma, courbet
Reviewed By: courbet
Subscribers: dylanmckay, sdardis, nemanjai, jvesely, nhaehnle, javed.absar, eraman, hiraditya, kbarton, jrtc27, atanasyan, jsji, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D59260
llvm-svn: 356068
2019-03-14 01:07:09 +08:00
|
|
|
; X86-NEXT: movl %ecx, 4(%eax)
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: popl %esi
|
|
|
|
; X86-NEXT: retl
|
Add LiveRangeShrink pass to shrink live range within BB.
Summary: LiveRangeShrink pass moves instruction right after the definition with the same BB if the instruction and its operands all have more than one use. This pass is inexpensive and guarantees optimal live-range within BB.
Reviewers: davidxl, wmi, hfinkel, MatzeB, andreadb
Reviewed By: MatzeB, andreadb
Subscribers: hiraditya, jyknight, sanjoy, skatkov, gberry, jholewinski, qcolombet, javed.absar, krytarowski, atrick, spatel, RKSimon, andreadb, MatzeB, mehdi_amini, mgorny, efriedma, davide, dberlin, llvm-commits
Differential Revision: https://reviews.llvm.org/D32563
llvm-svn: 304371
2017-06-01 07:25:25 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_64_mem:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: rorq (%rdi)
|
|
|
|
; X64-NEXT: retq
|
Add LiveRangeShrink pass to shrink live range within BB.
Summary: LiveRangeShrink pass moves instruction right after the definition with the same BB if the instruction and its operands all have more than one use. This pass is inexpensive and guarantees optimal live-range within BB.
Reviewers: davidxl, wmi, hfinkel, MatzeB, andreadb
Reviewed By: MatzeB, andreadb
Subscribers: hiraditya, jyknight, sanjoy, skatkov, gberry, jholewinski, qcolombet, javed.absar, krytarowski, atrick, spatel, RKSimon, andreadb, MatzeB, mehdi_amini, mgorny, efriedma, davide, dberlin, llvm-commits
Differential Revision: https://reviews.llvm.org/D32563
llvm-svn: 304371
2017-06-01 07:25:25 +08:00
|
|
|
|
2017-02-20 08:37:20 +08:00
|
|
|
%A = load i64, i64 *%Aptr
|
|
|
|
%B = shl i64 %A, 63
|
|
|
|
%C = lshr i64 %A, 1
|
|
|
|
%D = or i64 %B, %C
|
|
|
|
store i64 %D, i64* %Aptr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
define void @rotr1_32_mem(i32* %Aptr) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_32_mem:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorl (%eax)
|
|
|
|
; X86-NEXT: retl
|
2017-02-20 08:37:20 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_32_mem:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: rorl (%rdi)
|
|
|
|
; X64-NEXT: retq
|
2017-02-20 08:37:20 +08:00
|
|
|
%A = load i32, i32 *%Aptr
|
|
|
|
%B = shl i32 %A, 31
|
|
|
|
%C = lshr i32 %A, 1
|
|
|
|
%D = or i32 %B, %C
|
|
|
|
store i32 %D, i32* %Aptr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
define void @rotr1_16_mem(i16* %Aptr) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_16_mem:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorw (%eax)
|
|
|
|
; X86-NEXT: retl
|
2017-02-20 08:37:20 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_16_mem:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: rorw (%rdi)
|
|
|
|
; X64-NEXT: retq
|
2017-02-20 08:37:20 +08:00
|
|
|
%A = load i16, i16 *%Aptr
|
|
|
|
%B = shl i16 %A, 15
|
|
|
|
%C = lshr i16 %A, 1
|
|
|
|
%D = or i16 %B, %C
|
|
|
|
store i16 %D, i16* %Aptr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
define void @rotr1_8_mem(i8* %Aptr) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: rotr1_8_mem:
|
|
|
|
; X86: # %bb.0:
|
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
|
|
|
|
; X86-NEXT: rorb (%eax)
|
|
|
|
; X86-NEXT: retl
|
2017-02-20 08:37:20 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: rotr1_8_mem:
|
|
|
|
; X64: # %bb.0:
|
|
|
|
; X64-NEXT: rorb (%rdi)
|
|
|
|
; X64-NEXT: retq
|
2017-02-20 08:37:20 +08:00
|
|
|
%A = load i8, i8 *%Aptr
|
|
|
|
%B = shl i8 %A, 7
|
|
|
|
%C = lshr i8 %A, 1
|
|
|
|
%D = or i8 %B, %C
|
|
|
|
store i8 %D, i8* %Aptr
|
|
|
|
ret void
|
|
|
|
}
|
2017-12-05 04:39:57 +08:00
|
|
|
|
2017-12-18 02:20:42 +08:00
|
|
|
define i64 @truncated_rot(i64 %x, i32 %amt) nounwind {
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-LABEL: truncated_rot:
|
|
|
|
; X86: # %bb.0: # %entry
|
|
|
|
; X86-NEXT: pushl %ebx
|
|
|
|
; X86-NEXT: pushl %edi
|
|
|
|
; X86-NEXT: pushl %esi
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %esi
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: movl {{[0-9]+}}(%esp), %ebx
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: movl %esi, %eax
|
|
|
|
; X86-NEXT: shll %cl, %eax
|
|
|
|
; X86-NEXT: testb $32, %cl
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: movl $0, %edi
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: jne .LBB28_2
|
|
|
|
; X86-NEXT: # %bb.1: # %entry
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: movl %eax, %edi
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: .LBB28_2: # %entry
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: movb $64, %dl
|
|
|
|
; X86-NEXT: subb %cl, %dl
|
|
|
|
; X86-NEXT: movl %ebx, %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: movl %edx, %ecx
|
|
|
|
; X86-NEXT: shrl %cl, %eax
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: shrdl %cl, %ebx, %esi
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: testb $32, %dl
|
|
|
|
; X86-NEXT: jne .LBB28_4
|
|
|
|
; X86-NEXT: # %bb.3: # %entry
|
|
|
|
; X86-NEXT: movl %esi, %eax
|
|
|
|
; X86-NEXT: .LBB28_4: # %entry
|
2018-12-09 00:07:38 +08:00
|
|
|
; X86-NEXT: orl %edi, %eax
|
2018-06-07 18:13:09 +08:00
|
|
|
; X86-NEXT: xorl %edx, %edx
|
|
|
|
; X86-NEXT: popl %esi
|
|
|
|
; X86-NEXT: popl %edi
|
|
|
|
; X86-NEXT: popl %ebx
|
|
|
|
; X86-NEXT: retl
|
2017-12-18 02:20:42 +08:00
|
|
|
;
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-LABEL: truncated_rot:
|
|
|
|
; X64: # %bb.0: # %entry
|
|
|
|
; X64-NEXT: movl %esi, %ecx
|
2018-09-20 02:59:08 +08:00
|
|
|
; X64-NEXT: # kill: def $cl killed $cl killed $ecx
|
2018-06-07 18:13:09 +08:00
|
|
|
; X64-NEXT: rolq %cl, %rdi
|
|
|
|
; X64-NEXT: movl %edi, %eax
|
|
|
|
; X64-NEXT: retq
|
2017-12-05 04:39:57 +08:00
|
|
|
entry:
|
|
|
|
%sh_prom = zext i32 %amt to i64
|
|
|
|
%shl = shl i64 %x, %sh_prom
|
|
|
|
%sub = sub nsw i32 64, %amt
|
|
|
|
%sh_prom1 = zext i32 %sub to i64
|
|
|
|
%shr = lshr i64 %x, %sh_prom1
|
|
|
|
%or = or i64 %shr, %shl
|
|
|
|
%and = and i64 %or, 4294967295
|
|
|
|
ret i64 %and
|
|
|
|
}
|