Files
clang-p2996/llvm/test/Transforms/FunctionSpecialization/global-var-constants.ll
Momchil Velikov de24d08459 [FuncSpec] Fix inconsistent treatment of global variables
There are a few inaccuracies with how FuncSpec handles global
variables.

When specialisation on non-const global variables is disabled (the
default) the pass could nevertheless perform some specializations,
e.g. on a constant GEP expression, or on a SSA variable, for which the
Solver has determined it has the value of a global variable.

When specialisation on non-const global variables is enabled, the pass
would skip non-scalars, e.g. a global array, but this should be
completely inconsequential, a pointer is a pointer.

Reviewed By: SjoerdMeijer

Differential Revision: https://reviews.llvm.org/D149476

Change-Id: Ic73051b2f8602587306760bf2ec552e5860f8d39
2023-05-05 09:56:06 +01:00

79 lines
1.9 KiB
LLVM

; RUN: opt -S --passes=ipsccp,deadargelim --force-specialization < %s | FileCheck %s --check-prefix=NO-GLOBALS
; RUN: opt -S --passes=ipsccp,deadargelim --force-specialization --funcspec-on-address < %s | FileCheck %s --check-prefix=GLOBALS
@G = global [10 x i32] zeroinitializer, align 4
define internal i32 @f(ptr %p) noinline {
entry:
%0 = load i32, ptr %p, align 4
store i32 0, ptr %p, align 4
ret i32 %0
}
define internal i32 @g(i32 %x, i32 %y, ptr %p) noinline {
entry:
%cmp = icmp sgt i32 %x, %y
br i1 %cmp, label %if.then, label %if.else
if.then:
br label %if.end
if.else:
br label %if.end
if.end:
%x.addr.0 = phi i32 [ %x, %if.then ], [ 11, %if.else ]
%p.addr.0 = phi ptr [ @G, %if.then ], [ %p, %if.else ]
%call = call i32 @f(ptr %p.addr.0)
%add = add nsw i32 %call, %x.addr.0
ret i32 %add
}
define i32 @h0(ptr %p) {
entry:
%call = call i32 @g(i32 2, i32 1, ptr %p)
ret i32 %call
}
define i32 @h1() {
entry:
%call = call i32 @f(ptr @G)
ret i32 %call
}
define i32 @h2() {
entry:
%call = call i32 @f(ptr getelementptr inbounds (i32, ptr @G, i64 1))
ret i32 %call
}
; Check if specialisation on the address of a non-const global variable
; is not allowed, then it is not performed.
; NO-GLOBALS-LABEL: define internal i32 @g()
; NO-GLOBALS: call i32 @f(ptr @G)
; NO-GLOBALS-LABEL: define i32 @h0(ptr %p)
; NO-GLOBALS:call i32 @g()
; NO-GLOBALS-LABEL: define i32 @h1()
; NO-GLOBALS: call i32 @f(ptr @G)
; NO-GLOBALS-LABEL: define i32 @h2()
; NO-GLOBALS: call i32 @f(ptr getelementptr inbounds (i32, ptr @G, i64 1))
; Check if specialisation on the address of a non-const global variable
; is allowed, then it is performed where possible.
; GLOBALS-LABEL: define internal i32 @g()
; GLOBALS: call i32 @f.2()
; GLOBALS-LABEL: define i32 @h0(ptr %p)
; GLOBALS: call i32 @g()
; GLOBALS-LABEL: define i32 @h1()
; GLOBALS: call i32 @f.2()
; GLOBALS-LABEL: define i32 @h2()
; GLOBALS: call i32 @f.1()