From dcd4896f123331a9c76ce6570845a831b4373a11 Mon Sep 17 00:00:00 2001 From: Dan Gohman Date: Wed, 3 Dec 2008 01:28:04 +0000 Subject: [PATCH] Fix byval arguments in the fastcc calling convention. The fastcc convention delegates to the regular x86-32 convention which handles byval, but only after it handles a few cases, and it's necessary to handle byval before handling those cases. This fixes PR3122 (and rdar://6400815), llvm-gcc miscompiling LLVM. llvm-svn: 60453 --- lib/Target/X86/X86CallingConv.td | 5 +++++ test/CodeGen/X86/fastcc-byval.ll | 20 ++++++++++++++++++++ test/CodeGen/X86/tailcallbyval.ll | 3 +-- 3 files changed, 26 insertions(+), 2 deletions(-) create mode 100644 test/CodeGen/X86/fastcc-byval.ll diff --git a/lib/Target/X86/X86CallingConv.td b/lib/Target/X86/X86CallingConv.td index 385bd916183..0bda72d9198 100644 --- a/lib/Target/X86/X86CallingConv.td +++ b/lib/Target/X86/X86CallingConv.td @@ -321,6 +321,11 @@ def CC_X86_32_FastCall : CallingConv<[ ]>; def CC_X86_32_FastCC : CallingConv<[ + // Handles byval parameters. Note that we can't rely on the delegation + // to CC_X86_32_Common for this because that happens after code that + // handles i32 arguments. + CCIfByVal>, + // Promote i8/i16 arguments to i32. CCIfType<[i8, i16], CCPromoteToType>, diff --git a/test/CodeGen/X86/fastcc-byval.ll b/test/CodeGen/X86/fastcc-byval.ll new file mode 100644 index 00000000000..9f6649c3b1a --- /dev/null +++ b/test/CodeGen/X86/fastcc-byval.ll @@ -0,0 +1,20 @@ +; RUN: llvm-as < %s | llc | grep {movl\[\[:space:\]\]*8(%esp), %eax} | count 2 +; PR3122 +; rdar://6400815 + +; byval requires a copy, even with fastcc. + +target datalayout = "e-p:32:32:32-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:32:64-f32:32:32-f64:32:64-v64:64:64-v128:128:128-a0:0:64-f80:128:128" +target triple = "i386-apple-darwin9.5" + %struct.MVT = type { i32 } + +define fastcc i32 @bar() nounwind { + %V = alloca %struct.MVT + %a = getelementptr %struct.MVT* %V, i32 0, i32 0 + store i32 1, i32* %a + call fastcc void @foo(%struct.MVT* byval %V) nounwind + %t = load i32* %a + ret i32 %t +} + +declare fastcc void @foo(%struct.MVT* byval) diff --git a/test/CodeGen/X86/tailcallbyval.ll b/test/CodeGen/X86/tailcallbyval.ll index 2861bb12582..916be566a14 100644 --- a/test/CodeGen/X86/tailcallbyval.ll +++ b/test/CodeGen/X86/tailcallbyval.ll @@ -1,6 +1,5 @@ ; RUN: llvm-as < %s | llc -march=x86 -tailcallopt | grep TAILCALL -; check for the 2 byval moves -; RUN: llvm-as < %s | llc -march=x86 -tailcallopt | grep movl | grep ecx | grep eax | wc -l | grep 1 +; RUN: llvm-as < %s | llc -march=x86 -tailcallopt | grep {movl\[\[:space:\]\]*4(%esp), %eax} | count 1 %struct.s = type {i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32, i32 }