Files
clang-p2996/llvm/test/Transforms/GlobalOpt/alias-resolve.ll
Fangrui Song c6692f819e [GlobalOpt] Don't replace alias with aliasee if either alias/aliasee may be preemptible
Generalize D99629 for ELF. A default visibility non-local symbol is preemptible
in a -shared link. `isInterposable` is an insufficient condition.

Moreover, a non-preemptible alias may be referenced in a sub constant expression
which intends to lower to a PC-relative relocation. Replacing the alias with a
preemptible aliasee may introduce a linker error.

Respect dso_preemptable and suppress optimization to fix the abose issues. With
the change, `alias = 345` will not be rewritten to use aliasee in a `-fpic`
compile.
```
int aliasee;
extern int alias __attribute__((alias("aliasee"), visibility("hidden")));
void foo() { alias = 345; } // intended to access the local copy
```

While here, refine the condition for the alias as well.

For some binary formats like COFF, `isInterposable` is a sufficient condition.
But I think canonicalization for the changed case has little advantage, so I
don't bother to add the `Triple(M.getTargetTriple()).isOSBinFormatELF()` or
`getPICLevel/getPIELevel` complexity.

For instrumentations, it's recommended not to create aliases that refer to
globals that have a weak linkage or is preemptible. However, the following is
supported and the IR needs to handle such cases.
```
int aliasee __attribute__((weak));
extern int alias __attribute__((alias("aliasee")));
```

There are other places where GlobalAlias isInterposable usage may need to be
fixed.

Reviewed By: rnk

Differential Revision: https://reviews.llvm.org/D107249
2022-03-18 14:17:05 -07:00

67 lines
1.8 KiB
LLVM

; RUN: opt < %s -passes=globalopt -S | FileCheck %s
@foo1 = alias void (), void ()* @foo2
;; foo2 is dso_local and non-weak. Resolved.
; CHECK: @foo1 = alias void (), void ()* @bar2
@foo2 = dso_local alias void(), void()* @bar1
;; bar1 is dso_local and non-weak. Resolved.
; CHECK: @foo2 = dso_local alias void (), void ()* @bar2
@bar1 = dso_local alias void (), void ()* @bar2
; CHECK: @bar1 = dso_local alias void (), void ()* @bar2
@weak1 = weak dso_local alias void (), void ()* @bar2
;; weak1 may be replaced with another definition in the linkage unit. Not resolved.
; CHECK: @weak1 = weak dso_local alias void (), void ()* @bar2
@bar4 = private unnamed_addr constant [2 x i8*] zeroinitializer
@foo4 = weak_odr unnamed_addr alias i8*, getelementptr inbounds ([2 x i8*], [2 x i8*]* @bar4, i32 0, i32 1)
; CHECK: @foo4 = weak_odr unnamed_addr alias i8*, getelementptr inbounds ([2 x i8*], [2 x i8*]* @bar4, i32 0, i32 1)
@priva = private alias void (), void ()* @bar5
; CHECK: @priva = private alias void (), void ()* @bar5
define dso_local void @bar2() {
ret void
}
; CHECK: define dso_local void @bar2()
define weak void @bar5() {
ret void
}
; CHECK: define weak void @bar5()
define void @baz() {
entry:
call void @foo1()
;; foo1 is dso_preemptable. Not resolved.
; CHECK: call void @foo1()
call void @foo2()
;; foo2 is dso_local and non-weak. Resolved.
; CHECK: call void @bar2()
call void @bar1()
;; bar1 is dso_local and non-weak. Resolved.
; CHECK: call void @bar2()
call void @weak1()
;; weak1 is weak. Not resolved.
; CHECK: call void @weak1()
call void @priva()
;; priva has a local linkage. Resolved.
; CHECK: call void @priva()
ret void
}
@foo3 = dso_local alias void (), void ()* @bar3
; CHECK-NOT: bar3
define internal void @bar3() {
ret void
}
;CHECK: define dso_local void @foo3