summaryrefslogtreecommitdiff
path: root/test/Analysis/DSGraph/GlobalsGraphFuncPtr.ll
blob: 6cc06c4b01ec69b534e1a7292685c06a326ea8c9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
; Test resolvable and unresolvable calls through function pointers:
; -- both should be retained in function graphs until resolved or until main
; -- former should get resolved in or before main() and never appear in GG
; -- latter should remain unresolved in main() and copied to GG
; -- globals in GG pointed to by latter should be marked I, but not other nodes
;
; RUN: llvm-as < %s | opt -analyze -datastructure-gc -dsgc-check-flags=KnownPtr:S,UnknownPtr:SI -dsgc-dspass=bu

%Z = internal global int 0
%X = internal global int 0
%M = internal global int 0
%.str_1 = internal constant [9 x sbyte] c"&Z = %p\0A\00"

implementation

declare int %printf(sbyte*, ...)
declare void %exit_dummy(int*)

internal void %makeCalls(void (int*)* %GpKnown.1, void (int*)* %GpUnknown.1,
                         int* %GpKnownPtr, int* %GpUnknownPtr) {
	%tmp.0 = load int* %Z
	%tmp.1.not = setne int %tmp.0, 0
	br bool %tmp.1.not, label %else, label %then

then:
	; pass to exit_dummy: never resolved
	call void %GpUnknown.1( int* %GpUnknownPtr )
	%tmp.61 = load int* %Z
	%inc1 = add int %tmp.61, 1
	store int %inc1, int* %Z
	%tmp.71 = call int (sbyte*, ...)* %printf( sbyte* getelementptr ([9 x sbyte]* %.str_1, long 0, long 0), int* %Z )
	ret void

else:
	; pass to knownF: resolved in main
	call void %GpKnown.1( int* %GpKnownPtr )
	%tmp.6 = load int* %Z
	%inc = add int %tmp.6, 1
	store int %inc, int* %Z

	; "known external": resolved here
	%tmp.7 = call int (sbyte*, ...)* %printf( sbyte* getelementptr ([9 x sbyte]* %.str_1, long 0, long 0), int* %Z )
	ret void
}

internal void %knownF(int* %Y.1) {
	%tmp.1 = seteq int* %Y.1, null
	br bool %tmp.1, label %then, label %UnifiedExitNode

then:
	call void %knownF( int* %Y.1 )   ; direct call to self: resolved here
	br label %UnifiedExitNode

UnifiedExitNode:
	ret void
}

int %main(int %argc.1) {
        %KnownPtr = alloca int
        %UnknownPtr = alloca int
	store int 1, int* %Z
	call void %makeCalls( void (int*)* %knownF, void (int*)* %exit_dummy,
                              int* %KnownPtr, int* %UnknownPtr )
	ret int 0
}