mirror of
https://src.fedoraproject.org/rpms/llvm.git
synced 2024-11-24 09:32:42 +00:00
Backport patch from LLVM 18 to fix a miscompliation related to coroutine
Signed-off-by: Kefu Chai <tchaikov@gmail.com>
This commit is contained in:
parent
1968652869
commit
2f5dbfd993
2 changed files with 121 additions and 1 deletions
114
PR59723.diff
Normal file
114
PR59723.diff
Normal file
|
@ -0,0 +1,114 @@
|
||||||
|
diff --git a/llvm/lib/Transforms/Coroutines/CoroElide.cpp b/llvm/lib/Transforms/Coroutines/CoroElide.cpp
|
||||||
|
index f032c568449b..3e31d95586c5 100644
|
||||||
|
--- a/llvm/lib/Transforms/Coroutines/CoroElide.cpp
|
||||||
|
+++ b/llvm/lib/Transforms/Coroutines/CoroElide.cpp
|
||||||
|
@@ -192,12 +192,49 @@ bool Lowerer::hasEscapePath(const CoroBeginInst *CB,
|
||||||
|
for (auto *DA : It->second)
|
||||||
|
Visited.insert(DA->getParent());
|
||||||
|
|
||||||
|
+ SmallPtrSet<const BasicBlock *, 32> EscapingBBs;
|
||||||
|
+ for (auto *U : CB->users()) {
|
||||||
|
+ // The use from coroutine intrinsics are not a problem.
|
||||||
|
+ if (isa<CoroFreeInst, CoroSubFnInst, CoroSaveInst>(U))
|
||||||
|
+ continue;
|
||||||
|
+
|
||||||
|
+ // Think all other usages may be an escaping candidate conservatively.
|
||||||
|
+ //
|
||||||
|
+ // Note that the major user of switch ABI coroutine (the C++) will store
|
||||||
|
+ // resume.fn, destroy.fn and the index to the coroutine frame immediately.
|
||||||
|
+ // So the parent of the coro.begin in C++ will be always escaping.
|
||||||
|
+ // Then we can't get any performance benefits for C++ by improving the
|
||||||
|
+ // precision of the method.
|
||||||
|
+ //
|
||||||
|
+ // The reason why we still judge it is we want to make LLVM Coroutine in
|
||||||
|
+ // switch ABIs to be self contained as much as possible instead of a
|
||||||
|
+ // by-product of C++20 Coroutines.
|
||||||
|
+ EscapingBBs.insert(cast<Instruction>(U)->getParent());
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
+ bool PotentiallyEscaped = false;
|
||||||
|
+
|
||||||
|
do {
|
||||||
|
const auto *BB = Worklist.pop_back_val();
|
||||||
|
if (!Visited.insert(BB).second)
|
||||||
|
continue;
|
||||||
|
- if (TIs.count(BB))
|
||||||
|
- return true;
|
||||||
|
+
|
||||||
|
+ // A Path insensitive marker to test whether the coro.begin escapes.
|
||||||
|
+ // It is intentional to make it path insensitive while it may not be
|
||||||
|
+ // precise since we don't want the process to be too slow.
|
||||||
|
+ PotentiallyEscaped |= EscapingBBs.count(BB);
|
||||||
|
+
|
||||||
|
+ if (TIs.count(BB)) {
|
||||||
|
+ if (!BB->getTerminator()->isExceptionalTerminator() || PotentiallyEscaped)
|
||||||
|
+ return true;
|
||||||
|
+
|
||||||
|
+ // If the function ends with the exceptional terminator, the memory used
|
||||||
|
+ // by the coroutine frame can be released by stack unwinding
|
||||||
|
+ // automatically. So we can think the coro.begin doesn't escape if it
|
||||||
|
+ // exits the function by exceptional terminator.
|
||||||
|
+
|
||||||
|
+ continue;
|
||||||
|
+ }
|
||||||
|
|
||||||
|
// Conservatively say that there is potentially a path.
|
||||||
|
if (!--Limit)
|
||||||
|
@@ -234,36 +271,36 @@ bool Lowerer::shouldElide(Function *F, DominatorTree &DT) const {
|
||||||
|
// memory location storing that value and not the virtual register.
|
||||||
|
|
||||||
|
SmallPtrSet<BasicBlock *, 8> Terminators;
|
||||||
|
- // First gather all of the non-exceptional terminators for the function.
|
||||||
|
+ // First gather all of the terminators for the function.
|
||||||
|
// Consider the final coro.suspend as the real terminator when the current
|
||||||
|
// function is a coroutine.
|
||||||
|
- for (BasicBlock &B : *F) {
|
||||||
|
- auto *TI = B.getTerminator();
|
||||||
|
- if (TI->getNumSuccessors() == 0 && !TI->isExceptionalTerminator() &&
|
||||||
|
- !isa<UnreachableInst>(TI))
|
||||||
|
- Terminators.insert(&B);
|
||||||
|
- }
|
||||||
|
+ for (BasicBlock &B : *F) {
|
||||||
|
+ auto *TI = B.getTerminator();
|
||||||
|
+
|
||||||
|
+ if (TI->getNumSuccessors() != 0 || isa<UnreachableInst>(TI))
|
||||||
|
+ continue;
|
||||||
|
+
|
||||||
|
+ Terminators.insert(&B);
|
||||||
|
+ }
|
||||||
|
|
||||||
|
// Filter out the coro.destroy that lie along exceptional paths.
|
||||||
|
SmallPtrSet<CoroBeginInst *, 8> ReferencedCoroBegins;
|
||||||
|
for (const auto &It : DestroyAddr) {
|
||||||
|
- // If there is any coro.destroy dominates all of the terminators for the
|
||||||
|
- // coro.begin, we could know the corresponding coro.begin wouldn't escape.
|
||||||
|
- for (Instruction *DA : It.second) {
|
||||||
|
- if (llvm::all_of(Terminators, [&](auto *TI) {
|
||||||
|
- return DT.dominates(DA, TI->getTerminator());
|
||||||
|
- })) {
|
||||||
|
- ReferencedCoroBegins.insert(It.first);
|
||||||
|
- break;
|
||||||
|
- }
|
||||||
|
- }
|
||||||
|
-
|
||||||
|
- // Whether there is any paths from coro.begin to Terminators which not pass
|
||||||
|
- // through any of the coro.destroys.
|
||||||
|
+ // If every terminators is dominated by coro.destroy, we could know the
|
||||||
|
+ // corresponding coro.begin wouldn't escape.
|
||||||
|
+ //
|
||||||
|
+ // Otherwise hasEscapePath would decide whether there is any paths from
|
||||||
|
+ // coro.begin to Terminators which not pass through any of the
|
||||||
|
+ // coro.destroys.
|
||||||
|
//
|
||||||
|
// hasEscapePath is relatively slow, so we avoid to run it as much as
|
||||||
|
// possible.
|
||||||
|
- if (!ReferencedCoroBegins.count(It.first) &&
|
||||||
|
+ if (llvm::all_of(Terminators,
|
||||||
|
+ [&](auto *TI) {
|
||||||
|
+ return llvm::any_of(It.second, [&](auto *DA) {
|
||||||
|
+ return DT.dominates(DA, TI->getTerminator());
|
||||||
|
+ });
|
||||||
|
+ }) ||
|
||||||
|
!hasEscapePath(It.first, Terminators))
|
||||||
|
ReferencedCoroBegins.insert(It.first);
|
||||||
|
}
|
|
@ -75,7 +75,7 @@
|
||||||
|
|
||||||
Name: %{pkg_name}
|
Name: %{pkg_name}
|
||||||
Version: %{maj_ver}.%{min_ver}.%{patch_ver}%{?rc_ver:~rc%{rc_ver}}
|
Version: %{maj_ver}.%{min_ver}.%{patch_ver}%{?rc_ver:~rc%{rc_ver}}
|
||||||
Release: 2%{?dist}
|
Release: 3%{?dist}
|
||||||
Summary: The Low Level Virtual Machine
|
Summary: The Low Level Virtual Machine
|
||||||
|
|
||||||
License: Apache-2.0 WITH LLVM-exception OR NCSA
|
License: Apache-2.0 WITH LLVM-exception OR NCSA
|
||||||
|
@ -92,6 +92,8 @@ Source6: release-keys.asc
|
||||||
Patch2: 0001-llvm-Add-install-targets-for-gtest.patch
|
Patch2: 0001-llvm-Add-install-targets-for-gtest.patch
|
||||||
# Backport of https://reviews.llvm.org/D156379 from LLVM 18.
|
# Backport of https://reviews.llvm.org/D156379 from LLVM 18.
|
||||||
Patch3: D156379.diff
|
Patch3: D156379.diff
|
||||||
|
# Backport of https://reviews.llvm.org/rG7037331a2f05990cd59f35a7c0f6ce87c0f3cb5f from LLVM 18
|
||||||
|
Patch4: PR59723.diff
|
||||||
|
|
||||||
# RHEL-specific patch to avoid unwanted recommonmark dep
|
# RHEL-specific patch to avoid unwanted recommonmark dep
|
||||||
Patch101: 0101-Deactivate-markdown-doc.patch
|
Patch101: 0101-Deactivate-markdown-doc.patch
|
||||||
|
@ -566,6 +568,10 @@ fi
|
||||||
%endif
|
%endif
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
|
* Thu Aug 24 2023 Kefu Chai <kefu.chai@scylladb.com> - 16.0.6-3
|
||||||
|
- Fix the stack-use-after-return when using coroutine
|
||||||
|
See https://github.com/llvm/llvm-project/issues/59723
|
||||||
|
|
||||||
* Thu Aug 03 2023 Tulio Magno Quites Machado Filho <tuliom@redhat.com> - 16.0.6-2
|
* Thu Aug 03 2023 Tulio Magno Quites Machado Filho <tuliom@redhat.com> - 16.0.6-2
|
||||||
- Fix rhbz #2224885
|
- Fix rhbz #2224885
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue