mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-20 12:31:40 +00:00
ArgPromotion: Don't touch variadic functions
Adding, removing, or changing non-pack parameters can change the ABI classification of pack parameters. Clang and other frontends encode the classification in the IR of the call site, but the callee side determines it dynamically based on the number of registers consumed so far. Changing the prototype affects the number of registers consumed would break such code. Dead argument elimination performs a similar task and already has a similar check to avoid this problem. Patch by Thomas Jablin! git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@216421 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
9fe621a69e
commit
9d1f8b1b21
@ -156,6 +156,13 @@ CallGraphNode *ArgPromotion::PromoteArguments(CallGraphNode *CGN) {
|
||||
isSelfRecursive = true;
|
||||
}
|
||||
|
||||
// Don't promote arguments for variadic functions. Adding, removing, or
|
||||
// changing non-pack parameters can change the classification of pack
|
||||
// parameters. Frontends encode that classification at the call site in the
|
||||
// IR, while in the callee the classification is determined dynamically based
|
||||
// on the number of registers consumed so far.
|
||||
if (F->isVarArg()) return nullptr;
|
||||
|
||||
// Check to see which arguments are promotable. If an argument is promotable,
|
||||
// add it to ArgsToPromote.
|
||||
SmallPtrSet<Argument*, 8> ArgsToPromote;
|
||||
|
28
test/Transforms/ArgumentPromotion/variadic.ll
Normal file
28
test/Transforms/ArgumentPromotion/variadic.ll
Normal file
@ -0,0 +1,28 @@
|
||||
; RUN: opt < %s -argpromotion -S | FileCheck %s
|
||||
|
||||
; Unused arguments from variadic functions cannot be eliminated as that changes
|
||||
; their classiciation according to the SysV amd64 ABI. Clang and other frontends
|
||||
; bake in the classification when they use things like byval, as in this test.
|
||||
|
||||
target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
|
||||
target triple = "x86_64-unknown-linux-gnu"
|
||||
|
||||
%struct.tt0 = type { i64, i64 }
|
||||
%struct.__va_list_tag = type { i32, i32, i8*, i8* }
|
||||
|
||||
@t45 = internal global %struct.tt0 { i64 1335139741, i64 438042995 }, align 8
|
||||
|
||||
; Function Attrs: nounwind uwtable
|
||||
define i32 @main(i32 %argc, i8** nocapture readnone %argv) #0 {
|
||||
entry:
|
||||
tail call void (i8*, i8*, i8*, i8*, i8*, ...)* @callee_t0f(i8* undef, i8* undef, i8* undef, i8* undef, i8* undef, %struct.tt0* byval align 8 @t45)
|
||||
ret i32 0
|
||||
}
|
||||
|
||||
; Function Attrs: nounwind uwtable
|
||||
define internal void @callee_t0f(i8* nocapture readnone %tp13, i8* nocapture readnone %tp14, i8* nocapture readnone %tp15, i8* nocapture readnone %tp16, i8* nocapture readnone %tp17, ...) {
|
||||
entry:
|
||||
ret void
|
||||
}
|
||||
|
||||
; CHECK-LABEL: define internal void @callee_t0f(i8* nocapture readnone %tp13, i8* nocapture readnone %tp14, i8* nocapture readnone %tp15, i8* nocapture readnone %tp16, i8* nocapture readnone %tp17, ...)
|
Loading…
x
Reference in New Issue
Block a user