Files
Yunbo Ni 769ffa0133 [ForceFunctionAttrs] Fix handling of conflicts for more attributes (#186304)
Fixes #185277 

ForceFunctionAttrs currently only checks the `alwaysinline`/`noinline`
conflict when forcing function attributes. This is incomplete, because
LLVM verifier rules define additional incompatible function attribute
combinations.

Extend hasConflictingFnAttr() to reject more conflicting function
attributes, including combinations involving `optnone`, `minsize`,
`optsize`, and `optdebug`.

Also add required companion attributes when forcing function attributes:
`optnone` requires `noinline`, so forceattrs now adds `noinline`
automatically when needed.
2026-03-17 02:10:16 +08:00

180 lines
8.2 KiB
LLVM

; RUN: opt < %s -S -passes=verify,forceattrs,verify | FileCheck %s --check-prefix=CHECK-CONTROL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute foo:noinline | FileCheck %s --check-prefix=CHECK-FOO
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-remove-attribute goo:cold | FileCheck %s --check-prefix=REMOVE-COLD
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-remove-attribute goo:noinline | FileCheck %s --check-prefix=REMOVE-NOINLINE
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute goo:cold -force-remove-attribute goo:noinline | FileCheck %s --check-prefix=ADD-COLD-REMOVE-NOINLINE
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute goo:noinline -force-remove-attribute goo:noinline | FileCheck %s --check-prefix=ADD-NOINLINE-REMOVE-NOINLINE
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute optsize | FileCheck %s --check-prefix=CHECK-ADD-ALL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-remove-attribute noinline | FileCheck %s --check-prefix=CHECK-REMOVE-ALL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute alwaysinline | FileCheck %s --check-prefix=CHECK-ALWAYSINLINE-ALL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute noinline | FileCheck %s --check-prefix=CHECK-NOINLINE-ALL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute optnone | FileCheck %s --check-prefix=CHECK-OPTNONE-ALL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute minsize | FileCheck %s --check-prefix=CHECK-MINSIZE-ALL
; RUN: opt < %s -S -passes=verify,forceattrs,verify -force-attribute optdebug | FileCheck %s --check-prefix=CHECK-OPTDEBUG-ALL
; CHECK-CONTROL: define void @foo() {
; CHECK-FOO: define void @foo() #0 {
define void @foo() {
ret void
}
; Ignore `cold` which does not exist before.
; REMOVE-COLD: define void @goo() #0 {
; Remove `noinline` attribute.
; REMOVE-NOINLINE: define void @goo() {
; Add `cold` and remove `noinline` leaving `cold` only.
; ADD-COLD-REMOVE-NOINLINE: define void @goo() #0 {
; `force-remove` takes precedence over `force`.
; `noinline` is removed.
; ADD-NOINLINE-REMOVE-NOINLINE: define void @goo() {
define void @goo() #0 {
ret void
}
define void @hoo() #1 {
ret void
}
define void @zoo() #2 {
ret void
}
define void @bar() #3 {
ret void
}
define void @baz() #4 {
ret void
}
define void @qoo() #5 {
ret void
}
attributes #0 = { noinline }
attributes #1 = { alwaysinline }
attributes #2 = { optsize }
attributes #3 = { minsize }
attributes #4 = { optdebug }
attributes #5 = { noinline optnone }
; CHECK-FOO: attributes #0 = { noinline }
; REMOVE-COLD: attributes #0 = { noinline }
; ADD-COLD-REMOVE-NOINLINE: attributes #0 = { cold }
; When passing an attribute without specifying a function, the attribute
; should be added to all compatible functions in the module.
; CHECK-ADD-ALL: define void @foo() #0 {
; CHECK-ADD-ALL: define void @goo() #1 {
; CHECK-ADD-ALL: define void @hoo() #2 {
; CHECK-ADD-ALL: define void @zoo() #0 {
; CHECK-ADD-ALL: define void @bar() #3 {
; CHECK-ADD-ALL: define void @baz() #4 {
; CHECK-ADD-ALL: define void @qoo() #5 {
; CHECK-ADD-ALL-DAG: attributes #0 = { optsize }
; CHECK-ADD-ALL-DAG: attributes #1 = { noinline optsize }
; CHECK-ADD-ALL-DAG: attributes #2 = { alwaysinline optsize }
; CHECK-ADD-ALL-DAG: attributes #3 = { minsize optsize }
; CHECK-ADD-ALL-DAG: attributes #4 = { optdebug }
; CHECK-ADD-ALL-DAG: attributes #5 = { noinline optnone }
; When passing an attribute to be removed without specifying a function,
; the attribute should be removed from all functions in the module that
; have it, unless doing so would create invalid IR (e.g. `optnone` requires
; `noinline`).
; CHECK-REMOVE-ALL: define void @foo() {
; CHECK-REMOVE-ALL: define void @goo() {
; CHECK-REMOVE-ALL: define void @hoo() #0 {
; CHECK-REMOVE-ALL: define void @zoo() #1 {
; CHECK-REMOVE-ALL: define void @bar() #2 {
; CHECK-REMOVE-ALL: define void @baz() #3 {
; CHECK-REMOVE-ALL: define void @qoo() #4 {
; CHECK-REMOVE-ALL-DAG: attributes #0 = { alwaysinline }
; CHECK-REMOVE-ALL-DAG: attributes #1 = { optsize }
; CHECK-REMOVE-ALL-DAG: attributes #2 = { minsize }
; CHECK-REMOVE-ALL-DAG: attributes #3 = { optdebug }
; CHECK-REMOVE-ALL-DAG: attributes #4 = { noinline optnone }
; When forcing alwaysinline on all functions, it should not be added to
; functions that already have noinline or optnone (would produce invalid IR).
; CHECK-ALWAYSINLINE-ALL: define void @foo() #0 {
; CHECK-ALWAYSINLINE-ALL: define void @goo() #1 {
; CHECK-ALWAYSINLINE-ALL: define void @hoo() #0 {
; CHECK-ALWAYSINLINE-ALL: define void @zoo() #2 {
; CHECK-ALWAYSINLINE-ALL: define void @bar() #3 {
; CHECK-ALWAYSINLINE-ALL: define void @baz() #4 {
; CHECK-ALWAYSINLINE-ALL: define void @qoo() #5 {
; CHECK-ALWAYSINLINE-ALL-DAG: attributes #0 = { alwaysinline }
; CHECK-ALWAYSINLINE-ALL-DAG: attributes #1 = { noinline }
; CHECK-ALWAYSINLINE-ALL-DAG: attributes #2 = { alwaysinline optsize }
; CHECK-ALWAYSINLINE-ALL-DAG: attributes #3 = { alwaysinline minsize }
; CHECK-ALWAYSINLINE-ALL-DAG: attributes #4 = { alwaysinline optdebug }
; CHECK-ALWAYSINLINE-ALL-DAG: attributes #5 = { noinline optnone }
; When forcing noinline on all functions, it should not be added to
; functions that already have alwaysinline (would produce invalid IR).
; CHECK-NOINLINE-ALL: define void @foo() #0 {
; CHECK-NOINLINE-ALL: define void @goo() #0 {
; CHECK-NOINLINE-ALL: define void @hoo() #1 {
; CHECK-NOINLINE-ALL: define void @zoo() #2 {
; CHECK-NOINLINE-ALL: define void @bar() #3 {
; CHECK-NOINLINE-ALL: define void @baz() #4 {
; CHECK-NOINLINE-ALL: define void @qoo() #5 {
; CHECK-NOINLINE-ALL-DAG: attributes #0 = { noinline }
; CHECK-NOINLINE-ALL-DAG: attributes #1 = { alwaysinline }
; CHECK-NOINLINE-ALL-DAG: attributes #2 = { noinline optsize }
; CHECK-NOINLINE-ALL-DAG: attributes #3 = { minsize noinline }
; CHECK-NOINLINE-ALL-DAG: attributes #4 = { noinline optdebug }
; CHECK-NOINLINE-ALL-DAG: attributes #5 = { noinline optnone }
; When forcing optnone on all functions, it should not be added to functions
; that already have alwaysinline, optsize, minsize, or optdebug.
; CHECK-OPTNONE-ALL: define void @foo() #0 {
; CHECK-OPTNONE-ALL: define void @goo() #0 {
; CHECK-OPTNONE-ALL: define void @hoo() #1 {
; CHECK-OPTNONE-ALL: define void @zoo() #2 {
; CHECK-OPTNONE-ALL: define void @bar() #3 {
; CHECK-OPTNONE-ALL: define void @baz() #4 {
; CHECK-OPTNONE-ALL: define void @qoo() #0 {
; CHECK-OPTNONE-ALL-DAG: attributes #0 = { noinline optnone }
; CHECK-OPTNONE-ALL-DAG: attributes #1 = { alwaysinline }
; CHECK-OPTNONE-ALL-DAG: attributes #2 = { optsize }
; CHECK-OPTNONE-ALL-DAG: attributes #3 = { minsize }
; CHECK-OPTNONE-ALL-DAG: attributes #4 = { optdebug }
; When forcing minsize on all functions, it should not be added to functions
; that already have optnone or optdebug.
; CHECK-MINSIZE-ALL: define void @foo() #0 {
; CHECK-MINSIZE-ALL: define void @goo() #1 {
; CHECK-MINSIZE-ALL: define void @hoo() #2 {
; CHECK-MINSIZE-ALL: define void @zoo() #3 {
; CHECK-MINSIZE-ALL: define void @bar() #0 {
; CHECK-MINSIZE-ALL: define void @baz() #4 {
; CHECK-MINSIZE-ALL: define void @qoo() #5 {
; CHECK-MINSIZE-ALL-DAG: attributes #0 = { minsize }
; CHECK-MINSIZE-ALL-DAG: attributes #1 = { minsize noinline }
; CHECK-MINSIZE-ALL-DAG: attributes #2 = { alwaysinline minsize }
; CHECK-MINSIZE-ALL-DAG: attributes #3 = { minsize optsize }
; CHECK-MINSIZE-ALL-DAG: attributes #4 = { optdebug }
; CHECK-MINSIZE-ALL-DAG: attributes #5 = { noinline optnone }
; When forcing optdebug on all functions, it should not be added to functions
; that already have optnone, minsize, or optsize.
; CHECK-OPTDEBUG-ALL: define void @foo() #0 {
; CHECK-OPTDEBUG-ALL: define void @goo() #1 {
; CHECK-OPTDEBUG-ALL: define void @hoo() #2 {
; CHECK-OPTDEBUG-ALL: define void @zoo() #3 {
; CHECK-OPTDEBUG-ALL: define void @bar() #4 {
; CHECK-OPTDEBUG-ALL: define void @baz() #0 {
; CHECK-OPTDEBUG-ALL: define void @qoo() #5 {
; CHECK-OPTDEBUG-ALL-DAG: attributes #0 = { optdebug }
; CHECK-OPTDEBUG-ALL-DAG: attributes #1 = { noinline optdebug }
; CHECK-OPTDEBUG-ALL-DAG: attributes #2 = { alwaysinline optdebug }
; CHECK-OPTDEBUG-ALL-DAG: attributes #3 = { optsize }
; CHECK-OPTDEBUG-ALL-DAG: attributes #4 = { minsize }
; CHECK-OPTDEBUG-ALL-DAG: attributes #5 = { noinline optnone }