forked from OSchip/llvm-project
134 lines
3.7 KiB
TableGen
134 lines
3.7 KiB
TableGen
//===- HexagonCallingConv.td ----------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
class CCIfArgIsVarArg<CCAction A>
|
|
: CCIf<"State.isVarArg() && "
|
|
"ValNo >= static_cast<HexagonCCState&>(State)"
|
|
".getNumNamedVarArgParams()", A>;
|
|
|
|
def CC_HexagonStack: CallingConv<[
|
|
CCIfType<[i32,v2i16,v4i8],
|
|
CCAssignToStack<4,4>>,
|
|
CCIfType<[i64,v2i32,v4i16,v8i8],
|
|
CCAssignToStack<8,8>>
|
|
]>;
|
|
|
|
def CC_Hexagon: CallingConv<[
|
|
CCIfType<[i1,i8,i16],
|
|
CCPromoteToType<i32>>,
|
|
CCIfType<[f32],
|
|
CCBitConvertToType<i32>>,
|
|
CCIfType<[f64],
|
|
CCBitConvertToType<i64>>,
|
|
|
|
CCIfByVal<
|
|
CCPassByVal<8,8>>,
|
|
CCIfArgIsVarArg<
|
|
CCDelegateTo<CC_HexagonStack>>,
|
|
|
|
// Pass split values in pairs, allocate odd register if necessary.
|
|
CCIfType<[i32],
|
|
CCIfSplit<
|
|
CCCustom<"CC_SkipOdd">>>,
|
|
|
|
CCIfType<[i32,v2i16,v4i8],
|
|
CCAssignToReg<[R0,R1,R2,R3,R4,R5]>>,
|
|
// Make sure to allocate any skipped 32-bit register, so it does not get
|
|
// allocated to a subsequent 32-bit value.
|
|
CCIfType<[i64,v2i32,v4i16,v8i8],
|
|
CCCustom<"CC_SkipOdd">>,
|
|
CCIfType<[i64,v2i32,v4i16,v8i8],
|
|
CCAssignToReg<[D0,D1,D2]>>,
|
|
|
|
CCDelegateTo<CC_HexagonStack>
|
|
]>;
|
|
|
|
def RetCC_Hexagon: CallingConv<[
|
|
CCIfType<[i1,i8,i16],
|
|
CCPromoteToType<i32>>,
|
|
CCIfType<[f32],
|
|
CCBitConvertToType<i32>>,
|
|
CCIfType<[f64],
|
|
CCBitConvertToType<i64>>,
|
|
|
|
// Small structures are returned in a pair of registers, (which is
|
|
// always r1:0). In such case, what is returned are two i32 values
|
|
// without any additional information (in ArgFlags) stating that
|
|
// they are parts of a structure. Because of that there is no way
|
|
// to differentiate that situation from an attempt to return two
|
|
// values, so always assign R0 and R1.
|
|
CCIfSplit<
|
|
CCAssignToReg<[R0,R1]>>,
|
|
CCIfType<[i32,v2i16,v4i8],
|
|
CCAssignToReg<[R0,R1]>>,
|
|
CCIfType<[i64,v2i32,v4i16,v8i8],
|
|
CCAssignToReg<[D0]>>
|
|
]>;
|
|
|
|
|
|
class CCIfHvx64<CCAction A>
|
|
: CCIf<"State.getMachineFunction().getSubtarget<HexagonSubtarget>()"
|
|
".useHVX64BOps()", A>;
|
|
|
|
class CCIfHvx128<CCAction A>
|
|
: CCIf<"State.getMachineFunction().getSubtarget<HexagonSubtarget>()"
|
|
".useHVX128BOps()", A>;
|
|
|
|
def CC_Hexagon_HVX: CallingConv<[
|
|
// HVX 64-byte mode
|
|
CCIfHvx64<
|
|
CCIfType<[v16i32,v32i16,v64i8],
|
|
CCAssignToReg<[V0,V1,V2,V3,V4,V5,V6,V7,V8,V9,V10,V11,V12,V13,V14,V15]>>>,
|
|
CCIfHvx64<
|
|
CCIfType<[v32i32,v64i16,v128i8],
|
|
CCAssignToReg<[W0,W1,W2,W3,W4,W5,W6,W7]>>>,
|
|
CCIfHvx64<
|
|
CCIfType<[v16i32,v32i16,v64i8],
|
|
CCAssignToStack<64,64>>>,
|
|
CCIfHvx64<
|
|
CCIfType<[v32i32,v64i16,v128i8],
|
|
CCAssignToStack<128,64>>>,
|
|
|
|
// HVX 128-byte mode
|
|
CCIfHvx128<
|
|
CCIfType<[v32i32,v64i16,v128i8],
|
|
CCAssignToReg<[V0,V1,V2,V3,V4,V5,V6,V7,V8,V9,V10,V11,V12,V13,V14,V15]>>>,
|
|
CCIfHvx128<
|
|
CCIfType<[v64i32,v128i16,v256i8],
|
|
CCAssignToReg<[W0,W1,W2,W3,W4,W5,W6,W7]>>>,
|
|
CCIfHvx128<
|
|
CCIfType<[v32i32,v64i16,v128i8],
|
|
CCAssignToStack<128,128>>>,
|
|
CCIfHvx128<
|
|
CCIfType<[v64i32,v128i16,v256i8],
|
|
CCAssignToStack<256,128>>>,
|
|
|
|
CCDelegateTo<CC_Hexagon>
|
|
]>;
|
|
|
|
def RetCC_Hexagon_HVX: CallingConv<[
|
|
// HVX 64-byte mode
|
|
CCIfHvx64<
|
|
CCIfType<[v16i32,v32i16,v64i8],
|
|
CCAssignToReg<[V0]>>>,
|
|
CCIfHvx64<
|
|
CCIfType<[v32i32,v64i16,v128i8],
|
|
CCAssignToReg<[W0]>>>,
|
|
|
|
// HVX 128-byte mode
|
|
CCIfHvx128<
|
|
CCIfType<[v32i32,v64i16,v128i8],
|
|
CCAssignToReg<[V0]>>>,
|
|
CCIfHvx128<
|
|
CCIfType<[v64i32,v128i16,v256i8],
|
|
CCAssignToReg<[W0]>>>,
|
|
|
|
CCDelegateTo<RetCC_Hexagon>
|
|
]>;
|
|
|