mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-01 00:33:09 +00:00
Support MSVC x86-32 sret convention. PR11688. Patch by Joe Groff.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@148513 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
f46fbade8a
commit
9a2478ac1a
@ -1855,7 +1855,8 @@ bool X86FastISel::DoSelectCall(const Instruction *I, const char *MemIntName) {
|
||||
// Issue CALLSEQ_END
|
||||
unsigned AdjStackUp = TII.getCallFrameDestroyOpcode();
|
||||
unsigned NumBytesCallee = 0;
|
||||
if (!Subtarget->is64Bit() && CS.paramHasAttr(1, Attribute::StructRet))
|
||||
if (!Subtarget->is64Bit() && !Subtarget->isTargetWindows() &&
|
||||
CS.paramHasAttr(1, Attribute::StructRet))
|
||||
NumBytesCallee = 4;
|
||||
BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DL, TII.get(AdjStackUp))
|
||||
.addImm(NumBytes).addImm(NumBytesCallee);
|
||||
|
@ -1811,6 +1811,7 @@ X86TargetLowering::LowerFormalArguments(SDValue Chain,
|
||||
|
||||
MachineFrameInfo *MFI = MF.getFrameInfo();
|
||||
bool Is64Bit = Subtarget->is64Bit();
|
||||
bool IsWindows = Subtarget->isTargetWindows();
|
||||
bool IsWin64 = Subtarget->isTargetWin64();
|
||||
|
||||
assert(!(isVarArg && IsTailCallConvention(CallConv)) &&
|
||||
@ -2046,7 +2047,8 @@ X86TargetLowering::LowerFormalArguments(SDValue Chain,
|
||||
} else {
|
||||
FuncInfo->setBytesToPopOnReturn(0); // Callee pops nothing.
|
||||
// If this is an sret function, the return should pop the hidden pointer.
|
||||
if (!Is64Bit && !IsTailCallConvention(CallConv) && ArgsAreStructReturn(Ins))
|
||||
if (!Is64Bit && !IsTailCallConvention(CallConv) && !IsWindows &&
|
||||
ArgsAreStructReturn(Ins))
|
||||
FuncInfo->setBytesToPopOnReturn(4);
|
||||
}
|
||||
|
||||
@ -2130,6 +2132,7 @@ X86TargetLowering::LowerCall(SDValue Chain, SDValue Callee,
|
||||
MachineFunction &MF = DAG.getMachineFunction();
|
||||
bool Is64Bit = Subtarget->is64Bit();
|
||||
bool IsWin64 = Subtarget->isTargetWin64();
|
||||
bool IsWindows = Subtarget->isTargetWindows();
|
||||
bool IsStructRet = CallIsStructReturn(Outs);
|
||||
bool IsSibcall = false;
|
||||
|
||||
@ -2543,10 +2546,12 @@ X86TargetLowering::LowerCall(SDValue Chain, SDValue Callee,
|
||||
if (X86::isCalleePop(CallConv, Is64Bit, isVarArg,
|
||||
getTargetMachine().Options.GuaranteedTailCallOpt))
|
||||
NumBytesForCalleeToPush = NumBytes; // Callee pops everything
|
||||
else if (!Is64Bit && !IsTailCallConvention(CallConv) && IsStructRet)
|
||||
else if (!Is64Bit && !IsTailCallConvention(CallConv) && !IsWindows &&
|
||||
IsStructRet)
|
||||
// If this is a call to a struct-return function, the callee
|
||||
// pops the hidden struct pointer, so we have to push it back.
|
||||
// This is common for Darwin/X86, Linux & Mingw32 targets.
|
||||
// For MSVC Win32 targets, the caller pops the hidden struct pointer.
|
||||
NumBytesForCalleeToPush = 4;
|
||||
else
|
||||
NumBytesForCalleeToPush = 0; // Callee pops nothing.
|
||||
|
28
test/CodeGen/X86/win32_sret.ll
Normal file
28
test/CodeGen/X86/win32_sret.ll
Normal file
@ -0,0 +1,28 @@
|
||||
; RUN: llc < %s -mtriple=i686-pc-win32 | FileCheck %s -check-prefix=WIN_X32
|
||||
; RUN: llc < %s -mtriple=i686-pc-mingw32 | FileCheck %s -check-prefix=MINGW_X32
|
||||
; RUN: llc < %s -mtriple=i386-pc-linux | FileCheck %s -check-prefix=LINUX
|
||||
; RUN: llc < %s -O0 -mtriple=i686-pc-win32 | FileCheck %s -check-prefix=WIN_X32
|
||||
; RUN: llc < %s -O0 -mtriple=i686-pc-mingw32 | FileCheck %s -check-prefix=MINGW_X32
|
||||
; RUN: llc < %s -O0 -mtriple=i386-pc-linux | FileCheck %s -check-prefix=LINUX
|
||||
|
||||
; The SysV ABI used by most Unixes and Mingw on x86 specifies that an sret pointer
|
||||
; is callee-cleanup. However, in MSVC's cdecl calling convention, sret pointer
|
||||
; arguments are caller-cleanup like normal arguments.
|
||||
|
||||
define void @sret1(i8* sret) nounwind {
|
||||
entry:
|
||||
; WIN_X32: {{ret$}}
|
||||
; MINGW_X32: ret $4
|
||||
; LINUX: ret $4
|
||||
ret void
|
||||
}
|
||||
|
||||
define void @sret2(i32* sret %x, i32 %y) nounwind {
|
||||
entry:
|
||||
; WIN_X32: {{ret$}}
|
||||
; MINGW_X32: ret $4
|
||||
; LINUX: ret $4
|
||||
store i32 %y, i32* %x
|
||||
ret void
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user