You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			353 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			LLVM
		
	
	
	
	
	
		
		
			
		
	
	
			353 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			LLVM
		
	
	
	
	
	
|   | ; Test transactional-execution intrinsics.
 | ||
|  | ;
 | ||
|  | ; RUN: llc < %s -mtriple=s390x-linux-gnu -mcpu=zEC12 | FileCheck %s
 | ||
|  | 
 | ||
|  | declare i32 @llvm.s390.tbegin(i8 *, i32) | ||
|  | declare i32 @llvm.s390.tbegin.nofloat(i8 *, i32) | ||
|  | declare void @llvm.s390.tbeginc(i8 *, i32) | ||
|  | declare i32 @llvm.s390.tend() | ||
|  | declare void @llvm.s390.tabort(i64) | ||
|  | declare void @llvm.s390.ntstg(i64, i64 *) | ||
|  | declare i32 @llvm.s390.etnd() | ||
|  | declare void @llvm.s390.ppa.txassist(i32) | ||
|  | 
 | ||
|  | ; TBEGIN.
 | ||
|  | define void @test_tbegin() { | ||
|  | ; CHECK-LABEL: test_tbegin:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK: std %f8,
 | ||
|  | ; CHECK: std %f9,
 | ||
|  | ; CHECK: std %f10,
 | ||
|  | ; CHECK: std %f11,
 | ||
|  | ; CHECK: std %f12,
 | ||
|  | ; CHECK: std %f13,
 | ||
|  | ; CHECK: std %f14,
 | ||
|  | ; CHECK: std %f15,
 | ||
|  | ; CHECK: tbegin 0, 65292
 | ||
|  | ; CHECK: ld %f8,
 | ||
|  | ; CHECK: ld %f9,
 | ||
|  | ; CHECK: ld %f10,
 | ||
|  | ; CHECK: ld %f11,
 | ||
|  | ; CHECK: ld %f12,
 | ||
|  | ; CHECK: ld %f13,
 | ||
|  | ; CHECK: ld %f14,
 | ||
|  | ; CHECK: ld %f15,
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin(i8 *null, i32 65292) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat).
 | ||
|  | define void @test_tbegin_nofloat1() { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat1:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 65292
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 65292) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with integer CC return value.
 | ||
|  | define i32 @test_tbegin_nofloat2() { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat2:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 65292
 | ||
|  | ; CHECK: ipm %r2
 | ||
|  | ; CHECK: srl %r2, 28
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 65292) | ||
|  |   ret i32 %res | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with implicit CC check.
 | ||
|  | define void @test_tbegin_nofloat3(i32 *%ptr) { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat3:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 65292
 | ||
|  | ; CHECK: bnhr %r14
 | ||
|  | ; CHECK: mvhi 0(%r2), 0
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 65292) | ||
|  |   %cmp = icmp eq i32 %res, 2 | ||
|  |   br i1 %cmp, label %if.then, label %if.end | ||
|  | 
 | ||
|  | if.then:                                          ; preds = %entry
 | ||
|  |   store i32 0, i32* %ptr, align 4 | ||
|  |   br label %if.end | ||
|  | 
 | ||
|  | if.end:                                           ; preds = %if.then, %entry
 | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with dual CC use.
 | ||
|  | define i32 @test_tbegin_nofloat4(i32 %pad, i32 *%ptr) { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat4:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 65292
 | ||
|  | ; CHECK: ipm %r2
 | ||
|  | ; CHECK: srl %r2, 28
 | ||
|  | ; CHECK: ciblh %r2, 2, 0(%r14)
 | ||
|  | ; CHECK: mvhi 0(%r3), 0
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 65292) | ||
|  |   %cmp = icmp eq i32 %res, 2 | ||
|  |   br i1 %cmp, label %if.then, label %if.end | ||
|  | 
 | ||
|  | if.then:                                          ; preds = %entry
 | ||
|  |   store i32 0, i32* %ptr, align 4 | ||
|  |   br label %if.end | ||
|  | 
 | ||
|  | if.end:                                           ; preds = %if.then, %entry
 | ||
|  |   ret i32 %res | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with register.
 | ||
|  | define void @test_tbegin_nofloat5(i8 *%ptr) { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat5:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0(%r2), 65292
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *%ptr, i32 65292) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with GRSM 0x0f00.
 | ||
|  | define void @test_tbegin_nofloat6() { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat6:
 | ||
|  | ; CHECK: stmg %r6, %r15,
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 3840
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 3840) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with GRSM 0xf100.
 | ||
|  | define void @test_tbegin_nofloat7() { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat7:
 | ||
|  | ; CHECK: stmg %r8, %r15,
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 61696
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 61696) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with GRSM 0xfe00 -- stack pointer added automatically.
 | ||
|  | define void @test_tbegin_nofloat8() { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat8:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 65280
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 65024) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with GRSM 0xfb00 -- no frame pointer needed.
 | ||
|  | define void @test_tbegin_nofloat9() { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat9:
 | ||
|  | ; CHECK: stmg %r10, %r15,
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 64256
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 64256) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGIN (nofloat) with GRSM 0xfb00 -- frame pointer added automatically.
 | ||
|  | define void @test_tbegin_nofloat10(i64 %n) { | ||
|  | ; CHECK-LABEL: test_tbegin_nofloat10:
 | ||
|  | ; CHECK: stmg %r11, %r15,
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbegin 0, 65280
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %buf = alloca i8, i64 %n | ||
|  |   call i32 @llvm.s390.tbegin.nofloat(i8 *null, i32 64256) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TBEGINC.
 | ||
|  | define void @test_tbeginc() { | ||
|  | ; CHECK-LABEL: test_tbeginc:
 | ||
|  | ; CHECK-NOT: stmg
 | ||
|  | ; CHECK-NOT: std
 | ||
|  | ; CHECK: tbeginc 0, 65288
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call void @llvm.s390.tbeginc(i8 *null, i32 65288) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TEND with integer CC return value.
 | ||
|  | define i32 @test_tend1() { | ||
|  | ; CHECK-LABEL: test_tend1:
 | ||
|  | ; CHECK: tend
 | ||
|  | ; CHECK: ipm %r2
 | ||
|  | ; CHECK: srl %r2, 28
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.tend() | ||
|  |   ret i32 %res | ||
|  | } | ||
|  | 
 | ||
|  | ; TEND with implicit CC check.
 | ||
|  | define void @test_tend3(i32 *%ptr) { | ||
|  | ; CHECK-LABEL: test_tend3:
 | ||
|  | ; CHECK: tend
 | ||
|  | ; CHECK: ber %r14
 | ||
|  | ; CHECK: mvhi 0(%r2), 0
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.tend() | ||
|  |   %cmp = icmp eq i32 %res, 2 | ||
|  |   br i1 %cmp, label %if.then, label %if.end | ||
|  | 
 | ||
|  | if.then:                                          ; preds = %entry
 | ||
|  |   store i32 0, i32* %ptr, align 4 | ||
|  |   br label %if.end | ||
|  | 
 | ||
|  | if.end:                                           ; preds = %if.then, %entry
 | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TEND with dual CC use.
 | ||
|  | define i32 @test_tend2(i32 %pad, i32 *%ptr) { | ||
|  | ; CHECK-LABEL: test_tend2:
 | ||
|  | ; CHECK: tend
 | ||
|  | ; CHECK: ipm %r2
 | ||
|  | ; CHECK: srl %r2, 28
 | ||
|  | ; CHECK: ciblh %r2, 2, 0(%r14)
 | ||
|  | ; CHECK: mvhi 0(%r3), 0
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.tend() | ||
|  |   %cmp = icmp eq i32 %res, 2 | ||
|  |   br i1 %cmp, label %if.then, label %if.end | ||
|  | 
 | ||
|  | if.then:                                          ; preds = %entry
 | ||
|  |   store i32 0, i32* %ptr, align 4 | ||
|  |   br label %if.end | ||
|  | 
 | ||
|  | if.end:                                           ; preds = %if.then, %entry
 | ||
|  |   ret i32 %res | ||
|  | } | ||
|  | 
 | ||
|  | ; TABORT with register only.
 | ||
|  | define void @test_tabort1(i64 %val) { | ||
|  | ; CHECK-LABEL: test_tabort1:
 | ||
|  | ; CHECK: tabort 0(%r2)
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call void @llvm.s390.tabort(i64 %val) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TABORT with immediate only.
 | ||
|  | define void @test_tabort2(i64 %val) { | ||
|  | ; CHECK-LABEL: test_tabort2:
 | ||
|  | ; CHECK: tabort 1234
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call void @llvm.s390.tabort(i64 1234) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TABORT with register + immediate.
 | ||
|  | define void @test_tabort3(i64 %val) { | ||
|  | ; CHECK-LABEL: test_tabort3:
 | ||
|  | ; CHECK: tabort 1234(%r2)
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %sum = add i64 %val, 1234 | ||
|  |   call void @llvm.s390.tabort(i64 %sum) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; TABORT with out-of-range immediate.
 | ||
|  | define void @test_tabort4(i64 %val) { | ||
|  | ; CHECK-LABEL: test_tabort4:
 | ||
|  | ; CHECK: tabort 0({{%r[1-5]}})
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call void @llvm.s390.tabort(i64 4096) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; NTSTG with base pointer only.
 | ||
|  | define void @test_ntstg1(i64 *%ptr, i64 %val) { | ||
|  | ; CHECK-LABEL: test_ntstg1:
 | ||
|  | ; CHECK: ntstg %r3, 0(%r2)
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call void @llvm.s390.ntstg(i64 %val, i64 *%ptr) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; NTSTG with base and index.
 | ||
|  | ; Check that VSTL doesn't allow an index.
 | ||
|  | define void @test_ntstg2(i64 *%base, i64 %index, i64 %val) { | ||
|  | ; CHECK-LABEL: test_ntstg2:
 | ||
|  | ; CHECK: sllg [[REG:%r[1-5]]], %r3, 3
 | ||
|  | ; CHECK: ntstg %r4, 0([[REG]],%r2)
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %ptr = getelementptr i64, i64 *%base, i64 %index | ||
|  |   call void @llvm.s390.ntstg(i64 %val, i64 *%ptr) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; NTSTG with the highest in-range displacement.
 | ||
|  | define void @test_ntstg3(i64 *%base, i64 %val) { | ||
|  | ; CHECK-LABEL: test_ntstg3:
 | ||
|  | ; CHECK: ntstg %r3, 524280(%r2)
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %ptr = getelementptr i64, i64 *%base, i64 65535 | ||
|  |   call void @llvm.s390.ntstg(i64 %val, i64 *%ptr) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; NTSTG with an out-of-range positive displacement.
 | ||
|  | define void @test_ntstg4(i64 *%base, i64 %val) { | ||
|  | ; CHECK-LABEL: test_ntstg4:
 | ||
|  | ; CHECK: ntstg %r3, 0({{%r[1-5]}})
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %ptr = getelementptr i64, i64 *%base, i64 65536 | ||
|  |   call void @llvm.s390.ntstg(i64 %val, i64 *%ptr) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; NTSTG with the lowest in-range displacement.
 | ||
|  | define void @test_ntstg5(i64 *%base, i64 %val) { | ||
|  | ; CHECK-LABEL: test_ntstg5:
 | ||
|  | ; CHECK: ntstg %r3, -524288(%r2)
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %ptr = getelementptr i64, i64 *%base, i64 -65536 | ||
|  |   call void @llvm.s390.ntstg(i64 %val, i64 *%ptr) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; NTSTG with an out-of-range negative displacement.
 | ||
|  | define void @test_ntstg6(i64 *%base, i64 %val) { | ||
|  | ; CHECK-LABEL: test_ntstg6:
 | ||
|  | ; CHECK: ntstg %r3, 0({{%r[1-5]}})
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %ptr = getelementptr i64, i64 *%base, i64 -65537 | ||
|  |   call void @llvm.s390.ntstg(i64 %val, i64 *%ptr) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | ; ETND.
 | ||
|  | define i32 @test_etnd() { | ||
|  | ; CHECK-LABEL: test_etnd:
 | ||
|  | ; CHECK: etnd %r2
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   %res = call i32 @llvm.s390.etnd() | ||
|  |   ret i32 %res | ||
|  | } | ||
|  | 
 | ||
|  | ; PPA (Transaction-Abort Assist)
 | ||
|  | define void @test_ppa_txassist(i32 %val) { | ||
|  | ; CHECK-LABEL: test_ppa_txassist:
 | ||
|  | ; CHECK: ppa %r2, 0, 1
 | ||
|  | ; CHECK: br %r14
 | ||
|  |   call void @llvm.s390.ppa.txassist(i32 %val) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 |