forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			265 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			LLVM
		
	
	
	
			
		
		
	
	
			265 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			LLVM
		
	
	
	
; RUN: opt -S -rewrite-statepoints-for-gc < %s | FileCheck %s
 | 
						|
; RUN: opt -S -passes=rewrite-statepoints-for-gc < %s | FileCheck %s
 | 
						|
 | 
						|
; constants don't get relocated.
 | 
						|
@G = addrspace(1) global i8 5
 | 
						|
 | 
						|
declare void @foo()
 | 
						|
 | 
						|
define i8 @test() gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test
 | 
						|
; CHECK: gc.statepoint
 | 
						|
; CHECK-NEXT: load i8, i8 addrspace(1)* inttoptr (i64 15 to i8 addrspace(1)*)
 | 
						|
; Mostly just here to show reasonable code test can come from.  
 | 
						|
entry:
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  %res = load i8, i8 addrspace(1)* inttoptr (i64 15 to i8 addrspace(1)*)
 | 
						|
  ret i8 %res
 | 
						|
}
 | 
						|
 | 
						|
define i8 @test2(i8 addrspace(1)* %p) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test2
 | 
						|
; CHECK: gc.statepoint
 | 
						|
; CHECK-NEXT: gc.relocate
 | 
						|
; CHECK-NEXT: icmp
 | 
						|
; Globals don't move and thus don't get relocated
 | 
						|
entry:
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  %cmp = icmp eq i8 addrspace(1)* %p, null
 | 
						|
  br i1 %cmp, label %taken, label %not_taken
 | 
						|
 | 
						|
taken:                                            ; preds = %not_taken, %entry
 | 
						|
  ret i8 0
 | 
						|
 | 
						|
not_taken:                                        ; preds = %entry
 | 
						|
  %cmp2 = icmp ne i8 addrspace(1)* %p, null
 | 
						|
  br i1 %cmp2, label %taken, label %dead
 | 
						|
 | 
						|
dead:                                             ; preds = %not_taken
 | 
						|
  %addr = getelementptr i8, i8 addrspace(1)* %p, i32 15
 | 
						|
  %res = load i8, i8 addrspace(1)* %addr
 | 
						|
  ret i8 %res
 | 
						|
}
 | 
						|
 | 
						|
define i8 @test3(i1 %always_true) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test3
 | 
						|
; CHECK: gc.statepoint
 | 
						|
; CHECK-NEXT: load i8, i8 addrspace(1)* @G
 | 
						|
entry:
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  %res = load i8, i8 addrspace(1)* @G, align 1
 | 
						|
  ret i8 %res
 | 
						|
}
 | 
						|
 | 
						|
; Even for source languages without constant references, we can
 | 
						|
; see constants can show up along paths where the value is dead.
 | 
						|
; This is particular relevant when computing bases of PHIs.  
 | 
						|
define i8 addrspace(1)* @test4(i8 addrspace(1)* %p) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test4
 | 
						|
entry:
 | 
						|
  %is_null = icmp eq i8 addrspace(1)* %p, null
 | 
						|
  br i1 %is_null, label %split, label %join
 | 
						|
 | 
						|
split:
 | 
						|
  call void @foo()
 | 
						|
  %arg_value_addr.i = getelementptr inbounds i8, i8 addrspace(1)* %p, i64 8
 | 
						|
  %arg_value_addr_casted.i = bitcast i8 addrspace(1)* %arg_value_addr.i to i8 addrspace(1)* addrspace(1)*
 | 
						|
  br label %join
 | 
						|
 | 
						|
join:
 | 
						|
; CHECK-LABEL: join
 | 
						|
; CHECK: %addr2.base =
 | 
						|
  %addr2 = phi i8 addrspace(1)* addrspace(1)* [ %arg_value_addr_casted.i, %split ], [ inttoptr (i64 8 to i8 addrspace(1)* addrspace(1)*), %entry ]
 | 
						|
  ;; NOTE: This particular example can be jump-threaded, but in general,
 | 
						|
  ;; we can't, and have to deal with the resulting IR.
 | 
						|
  br i1 %is_null, label %early-exit, label %use
 | 
						|
 | 
						|
early-exit:
 | 
						|
  ret i8 addrspace(1)* null
 | 
						|
 | 
						|
use:
 | 
						|
; CHECK-LABEL: use:
 | 
						|
; CHECK: gc.statepoint
 | 
						|
; CHECK: gc.relocate
 | 
						|
  call void @foo()
 | 
						|
  %res = load i8 addrspace(1)*, i8 addrspace(1)* addrspace(1)* %addr2, align 1
 | 
						|
  ret i8 addrspace(1)* %res
 | 
						|
}
 | 
						|
 | 
						|
; Globals don't move and thus don't get relocated
 | 
						|
define i8 addrspace(1)* @test5(i1 %always_true) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test5
 | 
						|
; CHECK: gc.statepoint
 | 
						|
; CHECK-NEXT: %res = extractelement <2 x i8 addrspace(1)*> <i8 addrspace(1)* @G, i8 addrspace(1)* @G>, i32 0
 | 
						|
entry:
 | 
						|
  call void @foo()
 | 
						|
  %res = extractelement <2 x i8 addrspace(1)*> <i8 addrspace(1)* @G, i8 addrspace(1)* @G>, i32 0
 | 
						|
  ret i8 addrspace(1)* %res
 | 
						|
}
 | 
						|
 | 
						|
define i8 addrspace(1)* @test6(i64 %arg) gc "statepoint-example" {
 | 
						|
entry:
 | 
						|
  ; Don't fail any assertions and don't record null as a live value
 | 
						|
  ; CHECK-LABEL: test6
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  ; CHECK-NOT: call {{.*}}gc.relocate
 | 
						|
  %load_addr = getelementptr i8, i8 addrspace(1)* null, i64 %arg
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ret i8 addrspace(1)* %load_addr
 | 
						|
}
 | 
						|
 | 
						|
define i8 addrspace(1)* @test7(i64 %arg) gc "statepoint-example" {
 | 
						|
entry:
 | 
						|
  ; Same as test7 but use regular constant instead of a null
 | 
						|
  ; CHECK-LABEL: test7
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  ; CHECK-NOT: call {{.*}}gc.relocate
 | 
						|
  %load_addr = getelementptr i8, i8 addrspace(1)* inttoptr (i64 15 to i8 addrspace(1)*), i64 %arg
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ret i8 addrspace(1)* %load_addr
 | 
						|
}
 | 
						|
 | 
						|
define i8 @test8(i8 addrspace(1)* %p) gc "statepoint-example" {
 | 
						|
; Checks that base( phi(gep null, oop) ) = phi(null, base(oop)) and that we
 | 
						|
; correctly relocate this value
 | 
						|
; CHECK-LABEL: @test8
 | 
						|
entry:
 | 
						|
  %is_null = icmp eq i8 addrspace(1)* %p, null
 | 
						|
  br i1 %is_null, label %null.crit-edge, label %not-null
 | 
						|
 | 
						|
not-null:
 | 
						|
  %load_addr = getelementptr inbounds i8, i8 addrspace(1)* %p, i64 8
 | 
						|
  br label %join
 | 
						|
 | 
						|
null.crit-edge:
 | 
						|
  %load_addr.const = getelementptr inbounds i8, i8 addrspace(1)* null, i64 8
 | 
						|
  br label %join
 | 
						|
 | 
						|
join:
 | 
						|
  %addr = phi i8 addrspace(1)* [ %load_addr, %not-null ], [%load_addr.const, %null.crit-edge]
 | 
						|
  ; CHECK: %addr.base = phi i8 addrspace(1)*
 | 
						|
  ; CHECK-DAG: [ %p, %not-null ]
 | 
						|
  ; CHECK-DAG: [ null, %null.crit-edge ]
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%addr.base, %addr.base)
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%addr.base, %addr)
 | 
						|
  br i1 %is_null, label %early-exit, label %use
 | 
						|
 | 
						|
early-exit:
 | 
						|
  ret i8 0
 | 
						|
 | 
						|
use:
 | 
						|
  %res = load i8, i8 addrspace(1)* %addr, align 1
 | 
						|
  ret i8 %res
 | 
						|
}
 | 
						|
 | 
						|
define i8 @test9(i8 addrspace(1)* %p) gc "statepoint-example" {
 | 
						|
; Checks that base( phi(inttoptr, oop) ) = phi(null, base(oop)) and that we
 | 
						|
; correctly relocate this value
 | 
						|
; CHECK-LABEL: @test9
 | 
						|
entry:
 | 
						|
  %is_null = icmp eq i8 addrspace(1)* %p, null
 | 
						|
  br i1 %is_null, label %null.crit-edge, label %not-null
 | 
						|
 | 
						|
not-null:
 | 
						|
  %load_addr = getelementptr inbounds i8, i8 addrspace(1)* %p, i64 8
 | 
						|
  br label %join
 | 
						|
 | 
						|
null.crit-edge:
 | 
						|
  br label %join
 | 
						|
 | 
						|
join:
 | 
						|
  %addr = phi i8 addrspace(1)* [ %load_addr, %not-null ], [inttoptr (i64 8 to i8 addrspace(1)*), %null.crit-edge]
 | 
						|
  ; CHECK: %addr.base = phi i8 addrspace(1)*
 | 
						|
  ; CHECK-DAG: [ %p, %not-null ]
 | 
						|
  ; CHECK-DAG: [ null, %null.crit-edge ]
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%addr.base, %addr.base)
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%addr.base, %addr)
 | 
						|
  br i1 %is_null, label %early-exit, label %use
 | 
						|
 | 
						|
early-exit:
 | 
						|
  ret i8 0
 | 
						|
 | 
						|
use:
 | 
						|
  %res = load i8, i8 addrspace(1)* %addr, align 1
 | 
						|
  ret i8 %res
 | 
						|
}
 | 
						|
 | 
						|
define i8 @test10(i8 addrspace(1)* %p) gc "statepoint-example" {
 | 
						|
; Checks that base( phi(const gep, oop) ) = phi(null, base(oop)) and that we
 | 
						|
; correctly relocate this value
 | 
						|
; CHECK-LABEL: @test10
 | 
						|
entry:
 | 
						|
  %is_null = icmp eq i8 addrspace(1)* %p, null
 | 
						|
  br i1 %is_null, label %null.crit-edge, label %not-null
 | 
						|
 | 
						|
not-null:
 | 
						|
  %load_addr = getelementptr inbounds i8, i8 addrspace(1)* %p, i64 8
 | 
						|
  br label %join
 | 
						|
 | 
						|
null.crit-edge:
 | 
						|
  br label %join
 | 
						|
 | 
						|
join:
 | 
						|
  %addr = phi i8 addrspace(1)* [ %load_addr, %not-null ], [getelementptr (i8, i8 addrspace(1)* null, i64 8), %null.crit-edge]
 | 
						|
  ; CHECK: %addr.base = phi i8 addrspace(1)*
 | 
						|
  ; CHECK-DAG: [ %p, %not-null ]
 | 
						|
  ; CHECK-DAG: [ null, %null.crit-edge ]
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%addr.base, %addr.base)
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%addr.base, %addr)
 | 
						|
  br i1 %is_null, label %early-exit, label %use
 | 
						|
 | 
						|
early-exit:
 | 
						|
  ret i8 0
 | 
						|
 | 
						|
use:
 | 
						|
  %res = load i8, i8 addrspace(1)* %addr, align 1
 | 
						|
  ret i8 %res
 | 
						|
}
 | 
						|
 | 
						|
define i32 addrspace(1)* @test11(i1 %c) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test11
 | 
						|
; Checks that base( select(const1, const2) ) == null and that we don't record
 | 
						|
; such value in the oop map
 | 
						|
entry:
 | 
						|
  %val = select i1 %c, i32 addrspace(1)* inttoptr (i64 8 to i32 addrspace(1)*), i32 addrspace(1)* inttoptr (i64 15 to i32 addrspace(1)*)
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  ; CHECK-NOT: call {{.*}}gc.relocate
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ret i32 addrspace(1)* %val
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
define <2 x i32 addrspace(1)*> @test12(i1 %c) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test12
 | 
						|
; Same as test11 but with vectors
 | 
						|
entry:
 | 
						|
  %val = select i1 %c, <2 x i32 addrspace(1)*> <i32 addrspace(1)* inttoptr (i64 5 to i32 addrspace(1)*), 
 | 
						|
                                                i32 addrspace(1)* inttoptr (i64 15 to i32 addrspace(1)*)>, 
 | 
						|
                       <2 x i32 addrspace(1)*> <i32 addrspace(1)* inttoptr (i64 30 to i32 addrspace(1)*), 
 | 
						|
                                                i32 addrspace(1)* inttoptr (i64 60 to i32 addrspace(1)*)>
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  ; CHECK-NOT: call {{.*}}gc.relocate
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ret <2 x i32 addrspace(1)*> %val
 | 
						|
}
 | 
						|
 | 
						|
define <2 x i32 addrspace(1)*> @test13(i1 %c, <2 x i32 addrspace(1)*> %ptr) gc "statepoint-example" {
 | 
						|
; CHECK-LABEL: @test13
 | 
						|
; Similar to test8, test9 and test10 but with vectors
 | 
						|
entry:
 | 
						|
  %val = select i1 %c, <2 x i32 addrspace(1)*> %ptr, 
 | 
						|
                       <2 x i32 addrspace(1)*> <i32 addrspace(1)* inttoptr (i64 30 to i32 addrspace(1)*), i32 addrspace(1)* inttoptr (i64 60 to i32 addrspace(1)*)>
 | 
						|
  ; CHECK: %val.base = select i1 %c, <2 x i32 addrspace(1)*> %ptr, <2 x i32 addrspace(1)*> zeroinitializer, !is_base_value !0
 | 
						|
  ; CHECK: gc.statepoint
 | 
						|
  call void @foo() [ "deopt"() ]
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%val.base, %val.base)
 | 
						|
  ; CHECK-DAG: call {{.*}}gc.relocate{{.*}}(%val.base, %val)
 | 
						|
  ret <2 x i32 addrspace(1)*> %val
 | 
						|
}
 |