117 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			LLVM
		
	
	
	
			
		
		
	
	
			117 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			LLVM
		
	
	
	
| ; RUN: llc -verify-machineinstrs -mtriple=powerpc64le-unknown-linux-gnu \
 | |
| ; RUN:   -mcpu=pwr9 -ppc-asm-full-reg-names -ppc-vsr-nums-as-vr < %s | \
 | |
| ; RUN:   FileCheck %s
 | |
| 
 | |
| ; These tests show that for 32-bit and 64-bit scalars, combining a shift to
 | |
| ; a single multiply-high is only valid when the shift amount is the same as
 | |
| ; the width of the narrow type.
 | |
| 
 | |
| ; That is, combining a shift to mulh is only valid for 32-bit when the shift
 | |
| ; amount is 32.
 | |
| ; Likewise, combining a shift to mulh is only valid for 64-bit when the shift
 | |
| ; amount is 64.
 | |
| 
 | |
| define i32 @test_mulhw(i32 %a, i32 %b) {
 | |
| ; CHECK-LABEL: test_mulhw:
 | |
| ; CHECK:     mulld
 | |
| ; CHECK-NOT: mulhw
 | |
| ; CHECK:     blr
 | |
|   %1 = sext i32 %a to i64
 | |
|   %2 = sext i32 %b to i64
 | |
|   %mul = mul i64 %1, %2
 | |
|   %shr = lshr i64 %mul, 33
 | |
|   %tr = trunc i64 %shr to i32
 | |
|   ret i32 %tr
 | |
| }
 | |
| 
 | |
| define i32 @test_mulhu(i32 %a, i32 %b) {
 | |
| ; CHECK-LABEL: test_mulhu:
 | |
| ; CHECK:     mulld
 | |
| ; CHECK-NOT: mulhwu
 | |
| ; CHECK:     blr
 | |
|   %1 = zext i32 %a to i64
 | |
|   %2 = zext i32 %b to i64
 | |
|   %mul = mul i64 %1, %2
 | |
|   %shr = lshr i64 %mul, 33
 | |
|   %tr = trunc i64 %shr to i32
 | |
|   ret i32 %tr
 | |
| }
 | |
| 
 | |
| define i64 @test_mulhd(i64 %a, i64 %b) {
 | |
| ; CHECK-LABEL: test_mulhd:
 | |
| ; CHECK:    mulhd
 | |
| ; CHECK:    mulld
 | |
| ; CHECK:    blr
 | |
|   %1 = sext i64 %a to i128
 | |
|   %2 = sext i64 %b to i128
 | |
|   %mul = mul i128 %1, %2
 | |
|   %shr = lshr i128 %mul, 63
 | |
|   %tr = trunc i128 %shr to i64
 | |
|   ret i64 %tr
 | |
| }
 | |
| 
 | |
| define i64 @test_mulhdu(i64 %a, i64 %b) {
 | |
| ; CHECK-LABEL: test_mulhdu:
 | |
| ; CHECK:    mulhdu
 | |
| ; CHECK:    mulld
 | |
| ; CHECK:    blr
 | |
|   %1 = zext i64 %a to i128
 | |
|   %2 = zext i64 %b to i128
 | |
|   %mul = mul i128 %1, %2
 | |
|   %shr = lshr i128 %mul, 63
 | |
|   %tr = trunc i128 %shr to i64
 | |
|   ret i64 %tr
 | |
| }
 | |
| 
 | |
| define signext i32 @test_mulhw_signext(i32 %a, i32 %b) {
 | |
| ; CHECK-LABEL: test_mulhw_signext:
 | |
| ; CHECK:     mulld
 | |
| ; CHECK-NOT: mulhw
 | |
| ; CHECK:     blr
 | |
|   %1 = sext i32 %a to i64
 | |
|   %2 = sext i32 %b to i64
 | |
|   %mul = mul i64 %1, %2
 | |
|   %shr = lshr i64 %mul, 33
 | |
|   %tr = trunc i64 %shr to i32
 | |
|   ret i32 %tr
 | |
| }
 | |
| 
 | |
| define zeroext i32 @test_mulhu_zeroext(i32 %a, i32 %b) {
 | |
| ; CHECK-LABEL: test_mulhu_zeroext:
 | |
| ; CHECK:     mulld
 | |
| ; CHECK-NOT: mulhwu
 | |
| ; CHECK:     blr
 | |
|   %1 = zext i32 %a to i64
 | |
|   %2 = zext i32 %b to i64
 | |
|   %mul = mul i64 %1, %2
 | |
|   %shr = lshr i64 %mul, 33
 | |
|   %tr = trunc i64 %shr to i32
 | |
|   ret i32 %tr
 | |
| }
 | |
| 
 | |
| define signext i64 @test_mulhd_signext(i64 %a, i64 %b) {
 | |
| ; CHECK-LABEL: test_mulhd_signext:
 | |
| ; CHECK:    mulhd
 | |
| ; CHECK:    mulld
 | |
| ; CHECK:    blr
 | |
|   %1 = sext i64 %a to i128
 | |
|   %2 = sext i64 %b to i128
 | |
|   %mul = mul i128 %1, %2
 | |
|   %shr = lshr i128 %mul, 63
 | |
|   %tr = trunc i128 %shr to i64
 | |
|   ret i64 %tr
 | |
| }
 | |
| 
 | |
| define zeroext i64 @test_mulhdu_zeroext(i64 %a, i64 %b) {
 | |
| ; CHECK-LABEL: test_mulhdu_zeroext:
 | |
| ; CHECK:    mulhdu
 | |
| ; CHECK:    mulld
 | |
| ; CHECK:    blr
 | |
|   %1 = zext i64 %a to i128
 | |
|   %2 = zext i64 %b to i128
 | |
|   %mul = mul i128 %1, %2
 | |
|   %shr = lshr i128 %mul, 63
 | |
|   %tr = trunc i128 %shr to i64
 | |
|   ret i64 %tr
 | |
| }
 |