2009-05-03 20:57:15 +08:00
|
|
|
//===-- MSP430ISelLowering.cpp - MSP430 DAG Lowering Implementation ------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the MSP430TargetLowering class.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "MSP430ISelLowering.h"
|
|
|
|
#include "MSP430.h"
|
2009-12-07 10:28:10 +08:00
|
|
|
#include "MSP430MachineFunctionInfo.h"
|
2009-05-03 20:57:15 +08:00
|
|
|
#include "MSP430Subtarget.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "MSP430TargetMachine.h"
|
2009-05-03 20:57:15 +08:00
|
|
|
#include "llvm/CodeGen/CallingConvLower.h"
|
|
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
|
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
|
|
|
#include "llvm/CodeGen/SelectionDAGISel.h"
|
2010-02-16 06:37:53 +08:00
|
|
|
#include "llvm/CodeGen/TargetLoweringObjectFileImpl.h"
|
2009-05-03 20:57:15 +08:00
|
|
|
#include "llvm/CodeGen/ValueTypes.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/CallingConv.h"
|
|
|
|
#include "llvm/IR/DerivedTypes.h"
|
|
|
|
#include "llvm/IR/Function.h"
|
|
|
|
#include "llvm/IR/GlobalAlias.h"
|
|
|
|
#include "llvm/IR/GlobalVariable.h"
|
|
|
|
#include "llvm/IR/Intrinsics.h"
|
2009-12-07 10:27:08 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2009-05-03 20:57:15 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2009-07-09 03:04:27 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2009-08-23 15:05:07 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2009-05-03 20:57:15 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 10:41:26 +08:00
|
|
|
#define DEBUG_TYPE "msp430-lower"
|
|
|
|
|
2015-02-26 08:00:24 +08:00
|
|
|
MSP430TargetLowering::MSP430TargetLowering(const TargetMachine &TM,
|
|
|
|
const MSP430Subtarget &STI)
|
2014-11-14 05:29:21 +08:00
|
|
|
: TargetLowering(TM) {
|
2009-12-07 10:28:10 +08:00
|
|
|
|
2009-05-03 20:57:15 +08:00
|
|
|
// Set up the register classes.
|
2012-04-20 15:30:17 +08:00
|
|
|
addRegisterClass(MVT::i8, &MSP430::GR8RegClass);
|
|
|
|
addRegisterClass(MVT::i16, &MSP430::GR16RegClass);
|
2009-05-03 20:57:15 +08:00
|
|
|
|
|
|
|
// Compute derived properties from the register classes
|
2015-02-26 08:00:24 +08:00
|
|
|
computeRegisterProperties(STI.getRegisterInfo());
|
2009-05-03 20:59:50 +08:00
|
|
|
|
2009-05-03 21:03:14 +08:00
|
|
|
// Provide all sorts of operation actions
|
2014-09-10 14:58:14 +08:00
|
|
|
setStackPointerRegisterToSaveRestore(MSP430::SP);
|
2009-05-03 21:11:35 +08:00
|
|
|
setBooleanContents(ZeroOrOneBooleanContent);
|
2011-09-07 03:07:46 +08:00
|
|
|
setBooleanVectorContents(ZeroOrOneBooleanContent); // FIXME: Is this correct?
|
2009-05-03 21:11:35 +08:00
|
|
|
|
2009-11-08 01:15:25 +08:00
|
|
|
// We have post-incremented loads / stores.
|
2009-11-08 01:15:06 +08:00
|
|
|
setIndexedLoadAction(ISD::POST_INC, MVT::i8, Legal);
|
|
|
|
setIndexedLoadAction(ISD::POST_INC, MVT::i16, Legal);
|
|
|
|
|
[SelectionDAG] Allow targets to specify legality of extloads' result
type (in addition to the memory type).
The *LoadExt* legalization handling used to only have one type, the
memory type. This forced users to assume that as long as the extload
for the memory type was declared legal, and the result type was legal,
the whole extload was legal.
However, this isn't always the case. For instance, on X86, with AVX,
this is legal:
v4i32 load, zext from v4i8
but this isn't:
v4i64 load, zext from v4i8
Whereas v4i64 is (arguably) legal, even without AVX2.
Note that the same thing was done a while ago for truncstores (r46140),
but I assume no one needed it yet for extloads, so here we go.
Calls to getLoadExtAction were changed to add the value type, found
manually in the surrounding code.
Calls to setLoadExtAction were mechanically changed, by wrapping the
call in a loop, to match previous behavior. The loop iterates over
the MVT subrange corresponding to the memory type (FP vectors, etc...).
I also pulled neighboring setTruncStoreActions into some of the loops;
those shouldn't make a difference, as the additional types are illegal.
(e.g., i128->i1 truncstores on PPC.)
No functional change intended.
Differential Revision: http://reviews.llvm.org/D6532
llvm-svn: 225421
2015-01-08 08:51:32 +08:00
|
|
|
for (MVT VT : MVT::integer_valuetypes()) {
|
|
|
|
setLoadExtAction(ISD::EXTLOAD, VT, MVT::i1, Promote);
|
|
|
|
setLoadExtAction(ISD::SEXTLOAD, VT, MVT::i1, Promote);
|
|
|
|
setLoadExtAction(ISD::ZEXTLOAD, VT, MVT::i1, Promote);
|
|
|
|
setLoadExtAction(ISD::SEXTLOAD, VT, MVT::i8, Expand);
|
|
|
|
setLoadExtAction(ISD::SEXTLOAD, VT, MVT::i16, Expand);
|
|
|
|
}
|
2009-05-03 21:06:03 +08:00
|
|
|
|
2009-05-03 21:06:26 +08:00
|
|
|
// We don't have any truncstores
|
2009-08-12 04:47:22 +08:00
|
|
|
setTruncStoreAction(MVT::i16, MVT::i8, Expand);
|
|
|
|
|
|
|
|
setOperationAction(ISD::SRA, MVT::i8, Custom);
|
|
|
|
setOperationAction(ISD::SHL, MVT::i8, Custom);
|
|
|
|
setOperationAction(ISD::SRL, MVT::i8, Custom);
|
|
|
|
setOperationAction(ISD::SRA, MVT::i16, Custom);
|
|
|
|
setOperationAction(ISD::SHL, MVT::i16, Custom);
|
|
|
|
setOperationAction(ISD::SRL, MVT::i16, Custom);
|
|
|
|
setOperationAction(ISD::ROTL, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::ROTR, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::ROTL, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::ROTR, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::GlobalAddress, MVT::i16, Custom);
|
|
|
|
setOperationAction(ISD::ExternalSymbol, MVT::i16, Custom);
|
2010-05-01 20:04:32 +08:00
|
|
|
setOperationAction(ISD::BlockAddress, MVT::i16, Custom);
|
2009-08-12 04:47:22 +08:00
|
|
|
setOperationAction(ISD::BR_JT, MVT::Other, Expand);
|
|
|
|
setOperationAction(ISD::BR_CC, MVT::i8, Custom);
|
|
|
|
setOperationAction(ISD::BR_CC, MVT::i16, Custom);
|
|
|
|
setOperationAction(ISD::BRCOND, MVT::Other, Expand);
|
2009-12-12 07:01:29 +08:00
|
|
|
setOperationAction(ISD::SETCC, MVT::i8, Custom);
|
|
|
|
setOperationAction(ISD::SETCC, MVT::i16, Custom);
|
2009-08-12 04:47:22 +08:00
|
|
|
setOperationAction(ISD::SELECT, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::SELECT, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::SELECT_CC, MVT::i8, Custom);
|
|
|
|
setOperationAction(ISD::SELECT_CC, MVT::i16, Custom);
|
|
|
|
setOperationAction(ISD::SIGN_EXTEND, MVT::i16, Custom);
|
2009-08-26 01:00:23 +08:00
|
|
|
setOperationAction(ISD::DYNAMIC_STACKALLOC, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::DYNAMIC_STACKALLOC, MVT::i16, Expand);
|
2009-08-12 04:47:22 +08:00
|
|
|
|
|
|
|
setOperationAction(ISD::CTTZ, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::CTTZ, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::CTLZ, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::CTLZ, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::CTPOP, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::CTPOP, MVT::i16, Expand);
|
|
|
|
|
|
|
|
setOperationAction(ISD::SHL_PARTS, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::SHL_PARTS, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::SRL_PARTS, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::SRL_PARTS, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::SRA_PARTS, MVT::i8, Expand);
|
|
|
|
setOperationAction(ISD::SRA_PARTS, MVT::i16, Expand);
|
|
|
|
|
|
|
|
setOperationAction(ISD::SIGN_EXTEND_INREG, MVT::i1, Expand);
|
2009-07-17 15:28:06 +08:00
|
|
|
|
2009-05-03 21:14:25 +08:00
|
|
|
// FIXME: Implement efficiently multiplication by a constant
|
2017-05-12 03:56:14 +08:00
|
|
|
setOperationAction(ISD::MUL, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::MULHS, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::MULHU, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::SMUL_LOHI, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::UMUL_LOHI, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::MUL, MVT::i16, LibCall);
|
2009-08-12 04:47:22 +08:00
|
|
|
setOperationAction(ISD::MULHS, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::MULHU, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::SMUL_LOHI, MVT::i16, Expand);
|
|
|
|
setOperationAction(ISD::UMUL_LOHI, MVT::i16, Expand);
|
|
|
|
|
2017-05-12 03:56:14 +08:00
|
|
|
setOperationAction(ISD::UDIV, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::UDIVREM, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::UREM, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::SDIV, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::SDIVREM, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::SREM, MVT::i8, Promote);
|
|
|
|
setOperationAction(ISD::UDIV, MVT::i16, LibCall);
|
2009-08-12 04:47:22 +08:00
|
|
|
setOperationAction(ISD::UDIVREM, MVT::i16, Expand);
|
2017-05-12 03:56:14 +08:00
|
|
|
setOperationAction(ISD::UREM, MVT::i16, LibCall);
|
|
|
|
setOperationAction(ISD::SDIV, MVT::i16, LibCall);
|
2009-08-12 04:47:22 +08:00
|
|
|
setOperationAction(ISD::SDIVREM, MVT::i16, Expand);
|
2017-05-12 03:56:14 +08:00
|
|
|
setOperationAction(ISD::SREM, MVT::i16, LibCall);
|
2009-12-07 10:27:08 +08:00
|
|
|
|
2012-11-22 01:28:27 +08:00
|
|
|
// varargs support
|
|
|
|
setOperationAction(ISD::VASTART, MVT::Other, Custom);
|
|
|
|
setOperationAction(ISD::VAARG, MVT::Other, Expand);
|
|
|
|
setOperationAction(ISD::VAEND, MVT::Other, Expand);
|
|
|
|
setOperationAction(ISD::VACOPY, MVT::Other, Expand);
|
2013-07-02 03:44:44 +08:00
|
|
|
setOperationAction(ISD::JumpTable, MVT::i16, Custom);
|
2012-11-22 01:28:27 +08:00
|
|
|
|
2017-05-12 03:56:14 +08:00
|
|
|
// EABI Libcalls - EABI Section 6.2
|
|
|
|
const struct {
|
|
|
|
const RTLIB::Libcall Op;
|
|
|
|
const char * const Name;
|
|
|
|
const ISD::CondCode Cond;
|
|
|
|
} LibraryCalls[] = {
|
|
|
|
// Floating point conversions - EABI Table 6
|
|
|
|
{ RTLIB::FPROUND_F64_F32, "__mspabi_cvtdf", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPEXT_F32_F64, "__mspabi_cvtfd", ISD::SETCC_INVALID },
|
|
|
|
// The following is NOT implemented in libgcc
|
|
|
|
//{ RTLIB::FPTOSINT_F64_I16, "__mspabi_fixdi", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOSINT_F64_I32, "__mspabi_fixdli", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOSINT_F64_I64, "__mspabi_fixdlli", ISD::SETCC_INVALID },
|
|
|
|
// The following is NOT implemented in libgcc
|
|
|
|
//{ RTLIB::FPTOUINT_F64_I16, "__mspabi_fixdu", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOUINT_F64_I32, "__mspabi_fixdul", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOUINT_F64_I64, "__mspabi_fixdull", ISD::SETCC_INVALID },
|
|
|
|
// The following is NOT implemented in libgcc
|
|
|
|
//{ RTLIB::FPTOSINT_F32_I16, "__mspabi_fixfi", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOSINT_F32_I32, "__mspabi_fixfli", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOSINT_F32_I64, "__mspabi_fixflli", ISD::SETCC_INVALID },
|
|
|
|
// The following is NOT implemented in libgcc
|
|
|
|
//{ RTLIB::FPTOUINT_F32_I16, "__mspabi_fixfu", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOUINT_F32_I32, "__mspabi_fixful", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::FPTOUINT_F32_I64, "__mspabi_fixfull", ISD::SETCC_INVALID },
|
|
|
|
// TODO The following IS implemented in libgcc
|
|
|
|
//{ RTLIB::SINTTOFP_I16_F64, "__mspabi_fltid", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SINTTOFP_I32_F64, "__mspabi_fltlid", ISD::SETCC_INVALID },
|
|
|
|
// TODO The following IS implemented in libgcc but is not in the EABI
|
|
|
|
{ RTLIB::SINTTOFP_I64_F64, "__mspabi_fltllid", ISD::SETCC_INVALID },
|
|
|
|
// TODO The following IS implemented in libgcc
|
|
|
|
//{ RTLIB::UINTTOFP_I16_F64, "__mspabi_fltud", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UINTTOFP_I32_F64, "__mspabi_fltuld", ISD::SETCC_INVALID },
|
|
|
|
// The following IS implemented in libgcc but is not in the EABI
|
|
|
|
{ RTLIB::UINTTOFP_I64_F64, "__mspabi_fltulld", ISD::SETCC_INVALID },
|
|
|
|
// TODO The following IS implemented in libgcc
|
|
|
|
//{ RTLIB::SINTTOFP_I16_F32, "__mspabi_fltif", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SINTTOFP_I32_F32, "__mspabi_fltlif", ISD::SETCC_INVALID },
|
|
|
|
// TODO The following IS implemented in libgcc but is not in the EABI
|
|
|
|
{ RTLIB::SINTTOFP_I64_F32, "__mspabi_fltllif", ISD::SETCC_INVALID },
|
|
|
|
// TODO The following IS implemented in libgcc
|
|
|
|
//{ RTLIB::UINTTOFP_I16_F32, "__mspabi_fltuf", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UINTTOFP_I32_F32, "__mspabi_fltulf", ISD::SETCC_INVALID },
|
|
|
|
// The following IS implemented in libgcc but is not in the EABI
|
|
|
|
{ RTLIB::UINTTOFP_I64_F32, "__mspabi_fltullf", ISD::SETCC_INVALID },
|
|
|
|
|
|
|
|
// Floating point comparisons - EABI Table 7
|
|
|
|
{ RTLIB::OEQ_F64, "__mspabi_cmpd", ISD::SETEQ },
|
|
|
|
{ RTLIB::UNE_F64, "__mspabi_cmpd", ISD::SETNE },
|
|
|
|
{ RTLIB::OGE_F64, "__mspabi_cmpd", ISD::SETGE },
|
|
|
|
{ RTLIB::OLT_F64, "__mspabi_cmpd", ISD::SETLT },
|
|
|
|
{ RTLIB::OLE_F64, "__mspabi_cmpd", ISD::SETLE },
|
|
|
|
{ RTLIB::OGT_F64, "__mspabi_cmpd", ISD::SETGT },
|
|
|
|
{ RTLIB::OEQ_F32, "__mspabi_cmpf", ISD::SETEQ },
|
|
|
|
{ RTLIB::UNE_F32, "__mspabi_cmpf", ISD::SETNE },
|
|
|
|
{ RTLIB::OGE_F32, "__mspabi_cmpf", ISD::SETGE },
|
|
|
|
{ RTLIB::OLT_F32, "__mspabi_cmpf", ISD::SETLT },
|
|
|
|
{ RTLIB::OLE_F32, "__mspabi_cmpf", ISD::SETLE },
|
|
|
|
{ RTLIB::OGT_F32, "__mspabi_cmpf", ISD::SETGT },
|
|
|
|
|
|
|
|
// Floating point arithmetic - EABI Table 8
|
|
|
|
{ RTLIB::ADD_F64, "__mspabi_addd", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::ADD_F32, "__mspabi_addf", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::DIV_F64, "__mspabi_divd", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::DIV_F32, "__mspabi_divf", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::MUL_F64, "__mspabi_mpyd", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::MUL_F32, "__mspabi_mpyf", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SUB_F64, "__mspabi_subd", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SUB_F32, "__mspabi_subf", ISD::SETCC_INVALID },
|
|
|
|
// The following are NOT implemented in libgcc
|
|
|
|
// { RTLIB::NEG_F64, "__mspabi_negd", ISD::SETCC_INVALID },
|
|
|
|
// { RTLIB::NEG_F32, "__mspabi_negf", ISD::SETCC_INVALID },
|
|
|
|
|
|
|
|
// TODO: SLL/SRA/SRL are in libgcc, RLL isn't
|
|
|
|
|
|
|
|
// Universal Integer Operations - EABI Table 9
|
|
|
|
{ RTLIB::SDIV_I16, "__mspabi_divi", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SDIV_I32, "__mspabi_divli", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SDIV_I64, "__mspabi_divlli", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UDIV_I16, "__mspabi_divu", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UDIV_I32, "__mspabi_divul", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UDIV_I64, "__mspabi_divull", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SREM_I16, "__mspabi_remi", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SREM_I32, "__mspabi_remli", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::SREM_I64, "__mspabi_remlli", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UREM_I16, "__mspabi_remu", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UREM_I32, "__mspabi_remul", ISD::SETCC_INVALID },
|
|
|
|
{ RTLIB::UREM_I64, "__mspabi_remull", ISD::SETCC_INVALID },
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const auto &LC : LibraryCalls) {
|
|
|
|
setLibcallName(LC.Op, LC.Name);
|
|
|
|
if (LC.Cond != ISD::SETCC_INVALID)
|
|
|
|
setCmpLibcallCC(LC.Op, LC.Cond);
|
|
|
|
}
|
|
|
|
|
2017-05-24 05:49:42 +08:00
|
|
|
if (STI.hasHWMult16()) {
|
2017-05-12 03:56:14 +08:00
|
|
|
const struct {
|
|
|
|
const RTLIB::Libcall Op;
|
|
|
|
const char * const Name;
|
|
|
|
} LibraryCalls[] = {
|
|
|
|
// Integer Multiply - EABI Table 9
|
|
|
|
{ RTLIB::MUL_I16, "__mspabi_mpyi_hw" },
|
|
|
|
{ RTLIB::MUL_I32, "__mspabi_mpyl_hw" },
|
|
|
|
{ RTLIB::MUL_I64, "__mspabi_mpyll_hw" },
|
|
|
|
// TODO The __mspabi_mpysl*_hw functions ARE implemented in libgcc
|
|
|
|
// TODO The __mspabi_mpyul*_hw functions ARE implemented in libgcc
|
|
|
|
};
|
|
|
|
for (const auto &LC : LibraryCalls) {
|
|
|
|
setLibcallName(LC.Op, LC.Name);
|
|
|
|
}
|
2017-05-24 05:49:42 +08:00
|
|
|
} else if (STI.hasHWMult32()) {
|
2017-05-12 03:56:14 +08:00
|
|
|
const struct {
|
|
|
|
const RTLIB::Libcall Op;
|
|
|
|
const char * const Name;
|
|
|
|
} LibraryCalls[] = {
|
|
|
|
// Integer Multiply - EABI Table 9
|
|
|
|
{ RTLIB::MUL_I16, "__mspabi_mpyi_hw" },
|
|
|
|
{ RTLIB::MUL_I32, "__mspabi_mpyl_hw32" },
|
|
|
|
{ RTLIB::MUL_I64, "__mspabi_mpyll_hw32" },
|
|
|
|
// TODO The __mspabi_mpysl*_hw32 functions ARE implemented in libgcc
|
|
|
|
// TODO The __mspabi_mpyul*_hw32 functions ARE implemented in libgcc
|
|
|
|
};
|
|
|
|
for (const auto &LC : LibraryCalls) {
|
|
|
|
setLibcallName(LC.Op, LC.Name);
|
|
|
|
}
|
2017-05-24 05:49:42 +08:00
|
|
|
} else if (STI.hasHWMultF5()) {
|
2017-05-12 03:56:14 +08:00
|
|
|
const struct {
|
|
|
|
const RTLIB::Libcall Op;
|
|
|
|
const char * const Name;
|
|
|
|
} LibraryCalls[] = {
|
|
|
|
// Integer Multiply - EABI Table 9
|
|
|
|
{ RTLIB::MUL_I16, "__mspabi_mpyi_f5hw" },
|
|
|
|
{ RTLIB::MUL_I32, "__mspabi_mpyl_f5hw" },
|
|
|
|
{ RTLIB::MUL_I64, "__mspabi_mpyll_f5hw" },
|
|
|
|
// TODO The __mspabi_mpysl*_f5hw functions ARE implemented in libgcc
|
|
|
|
// TODO The __mspabi_mpyul*_f5hw functions ARE implemented in libgcc
|
|
|
|
};
|
|
|
|
for (const auto &LC : LibraryCalls) {
|
|
|
|
setLibcallName(LC.Op, LC.Name);
|
|
|
|
}
|
|
|
|
} else { // NoHWMult
|
|
|
|
const struct {
|
|
|
|
const RTLIB::Libcall Op;
|
|
|
|
const char * const Name;
|
|
|
|
} LibraryCalls[] = {
|
|
|
|
// Integer Multiply - EABI Table 9
|
|
|
|
{ RTLIB::MUL_I16, "__mspabi_mpyi" },
|
|
|
|
{ RTLIB::MUL_I32, "__mspabi_mpyl" },
|
|
|
|
{ RTLIB::MUL_I64, "__mspabi_mpyll" },
|
|
|
|
// The __mspabi_mpysl* functions are NOT implemented in libgcc
|
|
|
|
// The __mspabi_mpyul* functions are NOT implemented in libgcc
|
|
|
|
};
|
|
|
|
for (const auto &LC : LibraryCalls) {
|
|
|
|
setLibcallName(LC.Op, LC.Name);
|
|
|
|
}
|
|
|
|
setLibcallCallingConv(RTLIB::MUL_I64, CallingConv::MSP430_BUILTIN);
|
2009-12-07 10:27:08 +08:00
|
|
|
}
|
2011-05-07 04:34:06 +08:00
|
|
|
|
2017-05-12 03:56:14 +08:00
|
|
|
// Several of the runtime library functions use a special calling conv
|
|
|
|
setLibcallCallingConv(RTLIB::UDIV_I64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::UREM_I64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::SDIV_I64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::SREM_I64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::ADD_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::SUB_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::MUL_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::DIV_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::OEQ_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::UNE_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::OGE_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::OLT_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::OLE_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
setLibcallCallingConv(RTLIB::OGT_F64, CallingConv::MSP430_BUILTIN);
|
|
|
|
// TODO: __mspabi_srall, __mspabi_srlll, __mspabi_sllll
|
|
|
|
|
2011-05-07 04:34:06 +08:00
|
|
|
setMinFunctionAlignment(1);
|
2017-09-20 05:05:20 +08:00
|
|
|
setPrefFunctionAlignment(1);
|
2009-05-03 20:57:15 +08:00
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerOperation(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
2009-05-03 20:57:15 +08:00
|
|
|
switch (Op.getOpcode()) {
|
2009-05-03 21:13:17 +08:00
|
|
|
case ISD::SHL: // FALLTHROUGH
|
2009-05-03 21:16:17 +08:00
|
|
|
case ISD::SRL:
|
2009-05-03 21:07:31 +08:00
|
|
|
case ISD::SRA: return LowerShifts(Op, DAG);
|
2009-05-03 21:08:33 +08:00
|
|
|
case ISD::GlobalAddress: return LowerGlobalAddress(Op, DAG);
|
2010-05-01 20:04:32 +08:00
|
|
|
case ISD::BlockAddress: return LowerBlockAddress(Op, DAG);
|
2009-05-03 21:14:46 +08:00
|
|
|
case ISD::ExternalSymbol: return LowerExternalSymbol(Op, DAG);
|
2009-12-12 07:01:29 +08:00
|
|
|
case ISD::SETCC: return LowerSETCC(Op, DAG);
|
2009-05-03 21:19:09 +08:00
|
|
|
case ISD::BR_CC: return LowerBR_CC(Op, DAG);
|
|
|
|
case ISD::SELECT_CC: return LowerSELECT_CC(Op, DAG);
|
2009-05-03 21:17:49 +08:00
|
|
|
case ISD::SIGN_EXTEND: return LowerSIGN_EXTEND(Op, DAG);
|
2009-12-07 10:28:10 +08:00
|
|
|
case ISD::RETURNADDR: return LowerRETURNADDR(Op, DAG);
|
|
|
|
case ISD::FRAMEADDR: return LowerFRAMEADDR(Op, DAG);
|
2012-11-22 01:28:27 +08:00
|
|
|
case ISD::VASTART: return LowerVASTART(Op, DAG);
|
2013-07-02 03:44:44 +08:00
|
|
|
case ISD::JumpTable: return LowerJumpTable(Op, DAG);
|
2009-05-03 20:57:15 +08:00
|
|
|
default:
|
2009-07-15 00:55:14 +08:00
|
|
|
llvm_unreachable("unimplemented operand");
|
2009-05-03 20:57:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-26 21:44:29 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// MSP430 Inline Assembly Support
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// getConstraintType - Given a constraint letter, return the type of
|
|
|
|
/// constraint it is for this target.
|
|
|
|
TargetLowering::ConstraintType
|
2015-07-06 03:29:18 +08:00
|
|
|
MSP430TargetLowering::getConstraintType(StringRef Constraint) const {
|
2009-08-26 21:44:29 +08:00
|
|
|
if (Constraint.size() == 1) {
|
|
|
|
switch (Constraint[0]) {
|
|
|
|
case 'r':
|
|
|
|
return C_RegisterClass;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return TargetLowering::getConstraintType(Constraint);
|
|
|
|
}
|
|
|
|
|
2015-02-27 06:38:43 +08:00
|
|
|
std::pair<unsigned, const TargetRegisterClass *>
|
|
|
|
MSP430TargetLowering::getRegForInlineAsmConstraint(
|
2015-07-06 03:29:18 +08:00
|
|
|
const TargetRegisterInfo *TRI, StringRef Constraint, MVT VT) const {
|
2009-08-26 21:44:29 +08:00
|
|
|
if (Constraint.size() == 1) {
|
|
|
|
// GCC Constraint Letters
|
|
|
|
switch (Constraint[0]) {
|
|
|
|
default: break;
|
|
|
|
case 'r': // GENERAL_REGS
|
|
|
|
if (VT == MVT::i8)
|
2012-04-20 15:30:17 +08:00
|
|
|
return std::make_pair(0U, &MSP430::GR8RegClass);
|
2009-08-26 21:44:29 +08:00
|
|
|
|
2012-04-20 15:30:17 +08:00
|
|
|
return std::make_pair(0U, &MSP430::GR16RegClass);
|
2009-08-26 21:44:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-27 06:38:43 +08:00
|
|
|
return TargetLowering::getRegForInlineAsmConstraint(TRI, Constraint, VT);
|
2009-08-26 21:44:29 +08:00
|
|
|
}
|
|
|
|
|
2009-05-03 20:59:33 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Calling Convention Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2009-05-03 20:57:15 +08:00
|
|
|
#include "MSP430GenCallingConv.inc"
|
2009-05-03 20:59:33 +08:00
|
|
|
|
2013-10-15 16:19:39 +08:00
|
|
|
/// For each argument in a function store the number of pieces it is composed
|
|
|
|
/// of.
|
|
|
|
template<typename ArgT>
|
|
|
|
static void ParseFunctionArgs(const SmallVectorImpl<ArgT> &Args,
|
|
|
|
SmallVectorImpl<unsigned> &Out) {
|
2017-03-03 04:25:10 +08:00
|
|
|
unsigned CurrentArgIndex;
|
|
|
|
|
|
|
|
if (Args.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
CurrentArgIndex = Args[0].OrigArgIndex;
|
|
|
|
Out.push_back(0);
|
|
|
|
|
|
|
|
for (auto &Arg : Args) {
|
|
|
|
if (CurrentArgIndex == Arg.OrigArgIndex) {
|
|
|
|
Out.back() += 1;
|
2013-10-15 16:19:39 +08:00
|
|
|
} else {
|
|
|
|
Out.push_back(1);
|
2017-03-03 04:25:10 +08:00
|
|
|
CurrentArgIndex = Arg.OrigArgIndex;
|
2013-10-15 16:19:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void AnalyzeVarArgs(CCState &State,
|
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs) {
|
|
|
|
State.AnalyzeCallOperands(Outs, CC_MSP430_AssignStack);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void AnalyzeVarArgs(CCState &State,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins) {
|
|
|
|
State.AnalyzeFormalArguments(Ins, CC_MSP430_AssignStack);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Analyze incoming and outgoing function arguments. We need custom C++ code
|
|
|
|
/// to handle special constraints in the ABI like reversing the order of the
|
|
|
|
/// pieces of splitted arguments. In addition, all pieces of a certain argument
|
|
|
|
/// have to be passed either using registers or the stack but never mixing both.
|
|
|
|
template<typename ArgT>
|
|
|
|
static void AnalyzeArguments(CCState &State,
|
|
|
|
SmallVectorImpl<CCValAssign> &ArgLocs,
|
|
|
|
const SmallVectorImpl<ArgT> &Args) {
|
2017-05-12 03:56:14 +08:00
|
|
|
static const MCPhysReg CRegList[] = {
|
|
|
|
MSP430::R12, MSP430::R13, MSP430::R14, MSP430::R15
|
|
|
|
};
|
|
|
|
static const unsigned CNbRegs = array_lengthof(CRegList);
|
|
|
|
static const MCPhysReg BuiltinRegList[] = {
|
|
|
|
MSP430::R8, MSP430::R9, MSP430::R10, MSP430::R11,
|
2017-03-03 04:25:10 +08:00
|
|
|
MSP430::R12, MSP430::R13, MSP430::R14, MSP430::R15
|
2013-10-15 16:19:39 +08:00
|
|
|
};
|
2017-05-12 03:56:14 +08:00
|
|
|
static const unsigned BuiltinNbRegs = array_lengthof(BuiltinRegList);
|
|
|
|
|
|
|
|
ArrayRef<MCPhysReg> RegList;
|
|
|
|
unsigned NbRegs;
|
|
|
|
|
|
|
|
bool Builtin = (State.getCallingConv() == CallingConv::MSP430_BUILTIN);
|
|
|
|
if (Builtin) {
|
|
|
|
RegList = BuiltinRegList;
|
|
|
|
NbRegs = BuiltinNbRegs;
|
|
|
|
} else {
|
|
|
|
RegList = CRegList;
|
|
|
|
NbRegs = CNbRegs;
|
|
|
|
}
|
2013-10-15 16:19:39 +08:00
|
|
|
|
|
|
|
if (State.isVarArg()) {
|
|
|
|
AnalyzeVarArgs(State, Args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SmallVector<unsigned, 4> ArgsParts;
|
|
|
|
ParseFunctionArgs(Args, ArgsParts);
|
|
|
|
|
2017-05-12 03:56:14 +08:00
|
|
|
if (Builtin) {
|
|
|
|
assert(ArgsParts.size() == 2 &&
|
|
|
|
"Builtin calling convention requires two arguments");
|
|
|
|
}
|
|
|
|
|
2013-10-15 16:19:39 +08:00
|
|
|
unsigned RegsLeft = NbRegs;
|
2017-03-03 04:25:10 +08:00
|
|
|
bool UsedStack = false;
|
2013-10-15 16:19:39 +08:00
|
|
|
unsigned ValNo = 0;
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = ArgsParts.size(); i != e; i++) {
|
|
|
|
MVT ArgVT = Args[ValNo].VT;
|
|
|
|
ISD::ArgFlagsTy ArgFlags = Args[ValNo].Flags;
|
|
|
|
MVT LocVT = ArgVT;
|
|
|
|
CCValAssign::LocInfo LocInfo = CCValAssign::Full;
|
|
|
|
|
|
|
|
// Promote i8 to i16
|
|
|
|
if (LocVT == MVT::i8) {
|
|
|
|
LocVT = MVT::i16;
|
|
|
|
if (ArgFlags.isSExt())
|
|
|
|
LocInfo = CCValAssign::SExt;
|
|
|
|
else if (ArgFlags.isZExt())
|
|
|
|
LocInfo = CCValAssign::ZExt;
|
|
|
|
else
|
|
|
|
LocInfo = CCValAssign::AExt;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle byval arguments
|
|
|
|
if (ArgFlags.isByVal()) {
|
|
|
|
State.HandleByVal(ValNo++, ArgVT, LocVT, LocInfo, 2, 2, ArgFlags);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned Parts = ArgsParts[i];
|
|
|
|
|
2017-05-12 03:56:14 +08:00
|
|
|
if (Builtin) {
|
|
|
|
assert(Parts == 4 &&
|
|
|
|
"Builtin calling convention requires 64-bit arguments");
|
|
|
|
}
|
|
|
|
|
2017-03-03 04:25:10 +08:00
|
|
|
if (!UsedStack && Parts == 2 && RegsLeft == 1) {
|
|
|
|
// Special case for 32-bit register split, see EABI section 3.3.3
|
|
|
|
unsigned Reg = State.AllocateReg(RegList);
|
|
|
|
State.addLoc(CCValAssign::getReg(ValNo++, ArgVT, Reg, LocVT, LocInfo));
|
|
|
|
RegsLeft -= 1;
|
|
|
|
|
|
|
|
UsedStack = true;
|
|
|
|
CC_MSP430_AssignStack(ValNo++, ArgVT, LocVT, LocInfo, ArgFlags, State);
|
|
|
|
} else if (Parts <= RegsLeft) {
|
2013-10-15 16:19:39 +08:00
|
|
|
for (unsigned j = 0; j < Parts; j++) {
|
2015-02-21 10:11:17 +08:00
|
|
|
unsigned Reg = State.AllocateReg(RegList);
|
2013-10-15 16:19:39 +08:00
|
|
|
State.addLoc(CCValAssign::getReg(ValNo++, ArgVT, Reg, LocVT, LocInfo));
|
|
|
|
RegsLeft--;
|
|
|
|
}
|
|
|
|
} else {
|
2017-03-03 04:25:10 +08:00
|
|
|
UsedStack = true;
|
2013-10-15 16:19:39 +08:00
|
|
|
for (unsigned j = 0; j < Parts; j++)
|
|
|
|
CC_MSP430_AssignStack(ValNo++, ArgVT, LocVT, LocInfo, ArgFlags, State);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void AnalyzeRetResult(CCState &State,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins) {
|
|
|
|
State.AnalyzeCallResult(Ins, RetCC_MSP430);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void AnalyzeRetResult(CCState &State,
|
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs) {
|
|
|
|
State.AnalyzeReturn(Outs, RetCC_MSP430);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename ArgT>
|
|
|
|
static void AnalyzeReturnValues(CCState &State,
|
|
|
|
SmallVectorImpl<CCValAssign> &RVLocs,
|
|
|
|
const SmallVectorImpl<ArgT> &Args) {
|
|
|
|
AnalyzeRetResult(State, Args);
|
|
|
|
}
|
|
|
|
|
2016-06-12 23:39:02 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerFormalArguments(
|
|
|
|
SDValue Chain, CallingConv::ID CallConv, bool isVarArg,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins, const SDLoc &dl,
|
|
|
|
SelectionDAG &DAG, SmallVectorImpl<SDValue> &InVals) const {
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
|
|
|
|
switch (CallConv) {
|
2009-05-03 20:59:33 +08:00
|
|
|
default:
|
2017-08-22 17:11:41 +08:00
|
|
|
report_fatal_error("Unsupported calling convention");
|
2009-05-03 20:59:33 +08:00
|
|
|
case CallingConv::C:
|
|
|
|
case CallingConv::Fast:
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
return LowerCCCArguments(Chain, CallConv, isVarArg, Ins, dl, DAG, InVals);
|
2009-12-07 10:27:53 +08:00
|
|
|
case CallingConv::MSP430_INTR:
|
2012-01-21 05:51:11 +08:00
|
|
|
if (Ins.empty())
|
|
|
|
return Chain;
|
2010-04-08 06:58:41 +08:00
|
|
|
report_fatal_error("ISRs cannot have arguments");
|
2009-05-03 20:59:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
SDValue
|
2012-05-26 00:35:28 +08:00
|
|
|
MSP430TargetLowering::LowerCall(TargetLowering::CallLoweringInfo &CLI,
|
2010-04-17 23:26:15 +08:00
|
|
|
SmallVectorImpl<SDValue> &InVals) const {
|
2012-05-26 00:35:28 +08:00
|
|
|
SelectionDAG &DAG = CLI.DAG;
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc &dl = CLI.DL;
|
2013-07-14 12:42:23 +08:00
|
|
|
SmallVectorImpl<ISD::OutputArg> &Outs = CLI.Outs;
|
|
|
|
SmallVectorImpl<SDValue> &OutVals = CLI.OutVals;
|
|
|
|
SmallVectorImpl<ISD::InputArg> &Ins = CLI.Ins;
|
2012-05-26 00:35:28 +08:00
|
|
|
SDValue Chain = CLI.Chain;
|
|
|
|
SDValue Callee = CLI.Callee;
|
|
|
|
bool &isTailCall = CLI.IsTailCall;
|
|
|
|
CallingConv::ID CallConv = CLI.CallConv;
|
|
|
|
bool isVarArg = CLI.IsVarArg;
|
|
|
|
|
2010-01-27 08:07:07 +08:00
|
|
|
// MSP430 target does not yet support tail call optimization.
|
|
|
|
isTailCall = false;
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
|
|
|
|
switch (CallConv) {
|
2009-05-03 21:07:31 +08:00
|
|
|
default:
|
2017-08-22 17:11:41 +08:00
|
|
|
report_fatal_error("Unsupported calling convention");
|
2017-05-12 03:56:14 +08:00
|
|
|
case CallingConv::MSP430_BUILTIN:
|
2009-05-03 21:07:31 +08:00
|
|
|
case CallingConv::Fast:
|
|
|
|
case CallingConv::C:
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
return LowerCCCCallTo(Chain, Callee, CallConv, isVarArg, isTailCall,
|
2010-07-07 23:54:55 +08:00
|
|
|
Outs, OutVals, Ins, dl, DAG, InVals);
|
2009-12-07 10:27:53 +08:00
|
|
|
case CallingConv::MSP430_INTR:
|
2010-04-08 06:58:41 +08:00
|
|
|
report_fatal_error("ISRs cannot be called directly");
|
2009-05-03 21:07:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-03 20:59:33 +08:00
|
|
|
/// LowerCCCArguments - transform physical registers into virtual registers and
|
|
|
|
/// generate load operations for arguments places on the stack.
|
|
|
|
// FIXME: struct return stuff
|
2016-06-12 23:39:02 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerCCCArguments(
|
|
|
|
SDValue Chain, CallingConv::ID CallConv, bool isVarArg,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins, const SDLoc &dl,
|
|
|
|
SelectionDAG &DAG, SmallVectorImpl<SDValue> &InVals) const {
|
2009-05-03 20:59:33 +08:00
|
|
|
MachineFunction &MF = DAG.getMachineFunction();
|
2016-07-29 02:40:00 +08:00
|
|
|
MachineFrameInfo &MFI = MF.getFrameInfo();
|
2009-05-03 20:59:33 +08:00
|
|
|
MachineRegisterInfo &RegInfo = MF.getRegInfo();
|
2012-11-22 01:28:27 +08:00
|
|
|
MSP430MachineFunctionInfo *FuncInfo = MF.getInfo<MSP430MachineFunctionInfo>();
|
2009-05-03 20:59:33 +08:00
|
|
|
|
|
|
|
// Assign locations to all of the incoming arguments.
|
|
|
|
SmallVector<CCValAssign, 16> ArgLocs;
|
2014-08-07 02:45:26 +08:00
|
|
|
CCState CCInfo(CallConv, isVarArg, DAG.getMachineFunction(), ArgLocs,
|
|
|
|
*DAG.getContext());
|
2013-10-15 16:19:39 +08:00
|
|
|
AnalyzeArguments(CCInfo, ArgLocs, Ins);
|
2009-05-03 20:59:33 +08:00
|
|
|
|
2012-11-22 01:28:27 +08:00
|
|
|
// Create frame index for the start of the first vararg value
|
|
|
|
if (isVarArg) {
|
|
|
|
unsigned Offset = CCInfo.getNextStackOffset();
|
2016-07-29 02:40:00 +08:00
|
|
|
FuncInfo->setVarArgsFrameIndex(MFI.CreateFixedObject(1, Offset, true));
|
2012-11-22 01:28:27 +08:00
|
|
|
}
|
2009-05-03 20:59:33 +08:00
|
|
|
|
|
|
|
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
|
|
|
CCValAssign &VA = ArgLocs[i];
|
|
|
|
if (VA.isRegLoc()) {
|
|
|
|
// Arguments passed in registers
|
2009-08-11 06:56:29 +08:00
|
|
|
EVT RegVT = VA.getLocVT();
|
2009-08-12 04:47:22 +08:00
|
|
|
switch (RegVT.getSimpleVT().SimpleTy) {
|
2011-02-26 05:41:48 +08:00
|
|
|
default:
|
2009-07-09 03:04:27 +08:00
|
|
|
{
|
2009-07-09 04:53:28 +08:00
|
|
|
#ifndef NDEBUG
|
2009-08-23 15:05:07 +08:00
|
|
|
errs() << "LowerFormalArguments Unhandled argument type: "
|
2016-04-18 01:37:33 +08:00
|
|
|
<< RegVT.getEVTString() << "\n";
|
2009-07-09 04:53:28 +08:00
|
|
|
#endif
|
2014-04-28 12:05:08 +08:00
|
|
|
llvm_unreachable(nullptr);
|
2009-07-09 03:04:27 +08:00
|
|
|
}
|
2009-08-12 04:47:22 +08:00
|
|
|
case MVT::i16:
|
2012-04-20 15:30:17 +08:00
|
|
|
unsigned VReg = RegInfo.createVirtualRegister(&MSP430::GR16RegClass);
|
2009-05-03 20:59:33 +08:00
|
|
|
RegInfo.addLiveIn(VA.getLocReg(), VReg);
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
SDValue ArgValue = DAG.getCopyFromReg(Chain, dl, VReg, RegVT);
|
2009-05-03 20:59:33 +08:00
|
|
|
|
|
|
|
// If this is an 8-bit value, it is really passed promoted to 16
|
|
|
|
// bits. Insert an assert[sz]ext to capture this, then truncate to the
|
|
|
|
// right size.
|
|
|
|
if (VA.getLocInfo() == CCValAssign::SExt)
|
|
|
|
ArgValue = DAG.getNode(ISD::AssertSext, dl, RegVT, ArgValue,
|
|
|
|
DAG.getValueType(VA.getValVT()));
|
|
|
|
else if (VA.getLocInfo() == CCValAssign::ZExt)
|
|
|
|
ArgValue = DAG.getNode(ISD::AssertZext, dl, RegVT, ArgValue,
|
|
|
|
DAG.getValueType(VA.getValVT()));
|
|
|
|
|
|
|
|
if (VA.getLocInfo() != CCValAssign::Full)
|
|
|
|
ArgValue = DAG.getNode(ISD::TRUNCATE, dl, VA.getValVT(), ArgValue);
|
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
InVals.push_back(ArgValue);
|
2009-05-03 20:59:33 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Sanity check
|
|
|
|
assert(VA.isMemLoc());
|
2012-11-22 01:23:03 +08:00
|
|
|
|
|
|
|
SDValue InVal;
|
|
|
|
ISD::ArgFlagsTy Flags = Ins[i].Flags;
|
|
|
|
|
|
|
|
if (Flags.isByVal()) {
|
2016-07-29 02:40:00 +08:00
|
|
|
int FI = MFI.CreateFixedObject(Flags.getByValSize(),
|
|
|
|
VA.getLocMemOffset(), true);
|
2015-07-09 10:09:04 +08:00
|
|
|
InVal = DAG.getFrameIndex(FI, getPointerTy(DAG.getDataLayout()));
|
2012-11-22 01:23:03 +08:00
|
|
|
} else {
|
|
|
|
// Load the argument to a virtual register
|
|
|
|
unsigned ObjSize = VA.getLocVT().getSizeInBits()/8;
|
|
|
|
if (ObjSize > 2) {
|
|
|
|
errs() << "LowerFormalArguments Unhandled argument type: "
|
|
|
|
<< EVT(VA.getLocVT()).getEVTString()
|
|
|
|
<< "\n";
|
|
|
|
}
|
|
|
|
// Create the frame index object for this incoming parameter...
|
2016-07-29 02:40:00 +08:00
|
|
|
int FI = MFI.CreateFixedObject(ObjSize, VA.getLocMemOffset(), true);
|
2012-11-22 01:23:03 +08:00
|
|
|
|
|
|
|
// Create the SelectionDAG nodes corresponding to a load
|
|
|
|
//from this parameter
|
|
|
|
SDValue FIN = DAG.getFrameIndex(FI, MVT::i16);
|
2015-08-12 07:09:45 +08:00
|
|
|
InVal = DAG.getLoad(
|
|
|
|
VA.getLocVT(), dl, Chain, FIN,
|
[SelectionDAG] Get rid of bool parameters in SelectionDAG::getLoad, getStore, and friends.
Summary:
Instead, we take a single flags arg (a bitset).
Also add a default 0 alignment, and change the order of arguments so the
alignment comes before the flags.
This greatly simplifies many callsites, and fixes a bug in
AMDGPUISelLowering, wherein the order of the args to getLoad was
inverted. It also greatly simplifies the process of adding another flag
to getLoad.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, jyknight, dsanders, nemanjai, llvm-commits
Differential Revision: http://reviews.llvm.org/D22249
llvm-svn: 275592
2016-07-16 02:27:10 +08:00
|
|
|
MachinePointerInfo::getFixedStack(DAG.getMachineFunction(), FI));
|
2009-05-03 20:59:33 +08:00
|
|
|
}
|
2012-11-22 01:23:03 +08:00
|
|
|
|
|
|
|
InVals.push_back(InVal);
|
2009-05-03 20:59:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-03 04:25:10 +08:00
|
|
|
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
|
|
|
if (Ins[i].Flags.isSRet()) {
|
|
|
|
unsigned Reg = FuncInfo->getSRetReturnReg();
|
|
|
|
if (!Reg) {
|
|
|
|
Reg = MF.getRegInfo().createVirtualRegister(
|
|
|
|
getRegClassFor(MVT::i16));
|
|
|
|
FuncInfo->setSRetReturnReg(Reg);
|
|
|
|
}
|
|
|
|
SDValue Copy = DAG.getCopyToReg(DAG.getEntryNode(), dl, Reg, InVals[i]);
|
|
|
|
Chain = DAG.getNode(ISD::TokenFactor, dl, MVT::Other, Copy, Chain);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
return Chain;
|
2009-05-03 20:59:33 +08:00
|
|
|
}
|
2009-05-03 20:59:50 +08:00
|
|
|
|
2017-03-03 04:25:10 +08:00
|
|
|
bool
|
|
|
|
MSP430TargetLowering::CanLowerReturn(CallingConv::ID CallConv,
|
|
|
|
MachineFunction &MF,
|
|
|
|
bool IsVarArg,
|
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
|
|
LLVMContext &Context) const {
|
|
|
|
SmallVector<CCValAssign, 16> RVLocs;
|
|
|
|
CCState CCInfo(CallConv, IsVarArg, MF, RVLocs, Context);
|
|
|
|
return CCInfo.CheckReturn(Outs, RetCC_MSP430);
|
|
|
|
}
|
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
SDValue
|
2016-06-12 23:39:02 +08:00
|
|
|
MSP430TargetLowering::LowerReturn(SDValue Chain, CallingConv::ID CallConv,
|
|
|
|
bool isVarArg,
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
const SmallVectorImpl<ISD::OutputArg> &Outs,
|
2010-07-07 23:54:55 +08:00
|
|
|
const SmallVectorImpl<SDValue> &OutVals,
|
2016-06-12 23:39:02 +08:00
|
|
|
const SDLoc &dl, SelectionDAG &DAG) const {
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
|
2017-03-03 04:25:10 +08:00
|
|
|
MachineFunction &MF = DAG.getMachineFunction();
|
|
|
|
|
2009-05-03 20:59:50 +08:00
|
|
|
// CCValAssign - represent the assignment of the return value to a location
|
|
|
|
SmallVector<CCValAssign, 16> RVLocs;
|
|
|
|
|
2009-12-07 10:27:53 +08:00
|
|
|
// ISRs cannot return any value.
|
2012-01-21 05:51:11 +08:00
|
|
|
if (CallConv == CallingConv::MSP430_INTR && !Outs.empty())
|
2010-04-08 06:58:41 +08:00
|
|
|
report_fatal_error("ISRs cannot return any value");
|
2009-12-07 10:27:53 +08:00
|
|
|
|
2009-05-03 20:59:50 +08:00
|
|
|
// CCState - Info about the registers and stack slot.
|
2014-08-07 02:45:26 +08:00
|
|
|
CCState CCInfo(CallConv, isVarArg, DAG.getMachineFunction(), RVLocs,
|
|
|
|
*DAG.getContext());
|
2009-05-03 20:59:50 +08:00
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
// Analize return values.
|
2013-10-15 16:19:39 +08:00
|
|
|
AnalyzeReturnValues(CCInfo, RVLocs, Outs);
|
2009-05-03 20:59:50 +08:00
|
|
|
|
|
|
|
SDValue Flag;
|
2013-02-06 02:12:06 +08:00
|
|
|
SmallVector<SDValue, 4> RetOps(1, Chain);
|
2009-05-03 20:59:50 +08:00
|
|
|
|
|
|
|
// Copy the result values into the output registers.
|
|
|
|
for (unsigned i = 0; i != RVLocs.size(); ++i) {
|
|
|
|
CCValAssign &VA = RVLocs[i];
|
|
|
|
assert(VA.isRegLoc() && "Can only return in registers!");
|
|
|
|
|
|
|
|
Chain = DAG.getCopyToReg(Chain, dl, VA.getLocReg(),
|
2010-07-07 23:54:55 +08:00
|
|
|
OutVals[i], Flag);
|
2009-05-03 20:59:50 +08:00
|
|
|
|
2009-05-03 21:00:11 +08:00
|
|
|
// Guarantee that all emitted copies are stuck together,
|
|
|
|
// avoiding something bad.
|
2009-05-03 20:59:50 +08:00
|
|
|
Flag = Chain.getValue(1);
|
2013-02-06 02:12:06 +08:00
|
|
|
RetOps.push_back(DAG.getRegister(VA.getLocReg(), VA.getLocVT()));
|
2009-05-03 20:59:50 +08:00
|
|
|
}
|
|
|
|
|
2017-12-16 06:22:58 +08:00
|
|
|
if (MF.getFunction().hasStructRetAttr()) {
|
2017-03-03 04:25:10 +08:00
|
|
|
MSP430MachineFunctionInfo *FuncInfo = MF.getInfo<MSP430MachineFunctionInfo>();
|
|
|
|
unsigned Reg = FuncInfo->getSRetReturnReg();
|
|
|
|
|
|
|
|
if (!Reg)
|
|
|
|
llvm_unreachable("sret virtual register not created in entry block");
|
|
|
|
|
|
|
|
SDValue Val =
|
|
|
|
DAG.getCopyFromReg(Chain, dl, Reg, getPointerTy(DAG.getDataLayout()));
|
|
|
|
unsigned R12 = MSP430::R12;
|
|
|
|
|
|
|
|
Chain = DAG.getCopyToReg(Chain, dl, R12, Val, Flag);
|
|
|
|
Flag = Chain.getValue(1);
|
|
|
|
RetOps.push_back(DAG.getRegister(R12, getPointerTy(DAG.getDataLayout())));
|
|
|
|
}
|
|
|
|
|
2009-12-07 10:27:53 +08:00
|
|
|
unsigned Opc = (CallConv == CallingConv::MSP430_INTR ?
|
|
|
|
MSP430ISD::RETI_FLAG : MSP430ISD::RET_FLAG);
|
|
|
|
|
2013-02-06 02:12:06 +08:00
|
|
|
RetOps[0] = Chain; // Update chain.
|
|
|
|
|
|
|
|
// Add the flag if we have it.
|
2009-05-03 20:59:50 +08:00
|
|
|
if (Flag.getNode())
|
2013-02-06 02:12:06 +08:00
|
|
|
RetOps.push_back(Flag);
|
2009-05-03 20:59:50 +08:00
|
|
|
|
2014-04-27 02:35:24 +08:00
|
|
|
return DAG.getNode(Opc, dl, MVT::Other, RetOps);
|
2009-05-03 20:59:50 +08:00
|
|
|
}
|
|
|
|
|
2009-05-03 21:07:31 +08:00
|
|
|
/// LowerCCCCallTo - functions arguments are copied from virtual regs to
|
|
|
|
/// (physical regs)/(stack frame), CALLSEQ_START and CALLSEQ_END are emitted.
|
2016-06-12 23:39:02 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerCCCCallTo(
|
|
|
|
SDValue Chain, SDValue Callee, CallingConv::ID CallConv, bool isVarArg,
|
|
|
|
bool isTailCall, const SmallVectorImpl<ISD::OutputArg> &Outs,
|
|
|
|
const SmallVectorImpl<SDValue> &OutVals,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins, const SDLoc &dl,
|
|
|
|
SelectionDAG &DAG, SmallVectorImpl<SDValue> &InVals) const {
|
2009-05-03 21:07:31 +08:00
|
|
|
// Analyze operands of the call, assigning locations to each operand.
|
|
|
|
SmallVector<CCValAssign, 16> ArgLocs;
|
2014-08-07 02:45:26 +08:00
|
|
|
CCState CCInfo(CallConv, isVarArg, DAG.getMachineFunction(), ArgLocs,
|
|
|
|
*DAG.getContext());
|
2013-10-15 16:19:39 +08:00
|
|
|
AnalyzeArguments(CCInfo, ArgLocs, Outs);
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
// Get a count of how many bytes are to be pushed on the stack.
|
|
|
|
unsigned NumBytes = CCInfo.getNextStackOffset();
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
2009-05-03 21:07:31 +08:00
|
|
|
|
2017-05-09 21:35:13 +08:00
|
|
|
Chain = DAG.getCALLSEQ_START(Chain, NumBytes, 0, dl);
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
SmallVector<std::pair<unsigned, SDValue>, 4> RegsToPass;
|
|
|
|
SmallVector<SDValue, 12> MemOpChains;
|
|
|
|
SDValue StackPtr;
|
|
|
|
|
|
|
|
// Walk the register/memloc assignments, inserting copies/loads.
|
|
|
|
for (unsigned i = 0, e = ArgLocs.size(); i != e; ++i) {
|
|
|
|
CCValAssign &VA = ArgLocs[i];
|
|
|
|
|
2010-07-07 23:54:55 +08:00
|
|
|
SDValue Arg = OutVals[i];
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
// Promote the value if needed.
|
|
|
|
switch (VA.getLocInfo()) {
|
2009-07-15 00:55:14 +08:00
|
|
|
default: llvm_unreachable("Unknown loc info!");
|
2009-05-03 21:07:31 +08:00
|
|
|
case CCValAssign::Full: break;
|
|
|
|
case CCValAssign::SExt:
|
|
|
|
Arg = DAG.getNode(ISD::SIGN_EXTEND, dl, VA.getLocVT(), Arg);
|
|
|
|
break;
|
|
|
|
case CCValAssign::ZExt:
|
|
|
|
Arg = DAG.getNode(ISD::ZERO_EXTEND, dl, VA.getLocVT(), Arg);
|
|
|
|
break;
|
|
|
|
case CCValAssign::AExt:
|
|
|
|
Arg = DAG.getNode(ISD::ANY_EXTEND, dl, VA.getLocVT(), Arg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Arguments that can be passed on register must be kept at RegsToPass
|
|
|
|
// vector
|
|
|
|
if (VA.isRegLoc()) {
|
|
|
|
RegsToPass.push_back(std::make_pair(VA.getLocReg(), Arg));
|
|
|
|
} else {
|
|
|
|
assert(VA.isMemLoc());
|
|
|
|
|
2014-04-25 13:30:21 +08:00
|
|
|
if (!StackPtr.getNode())
|
2015-07-09 10:09:04 +08:00
|
|
|
StackPtr = DAG.getCopyFromReg(Chain, dl, MSP430::SP, PtrVT);
|
2009-05-03 21:07:31 +08:00
|
|
|
|
2015-07-09 10:09:04 +08:00
|
|
|
SDValue PtrOff =
|
|
|
|
DAG.getNode(ISD::ADD, dl, PtrVT, StackPtr,
|
|
|
|
DAG.getIntPtrConstant(VA.getLocMemOffset(), dl));
|
2009-05-03 21:07:31 +08:00
|
|
|
|
2012-11-22 01:23:03 +08:00
|
|
|
SDValue MemOp;
|
|
|
|
ISD::ArgFlagsTy Flags = Outs[i].Flags;
|
|
|
|
|
|
|
|
if (Flags.isByVal()) {
|
2015-04-28 22:05:47 +08:00
|
|
|
SDValue SizeNode = DAG.getConstant(Flags.getByValSize(), dl, MVT::i16);
|
2012-11-22 01:23:03 +08:00
|
|
|
MemOp = DAG.getMemcpy(Chain, dl, PtrOff, Arg, SizeNode,
|
|
|
|
Flags.getByValAlign(),
|
|
|
|
/*isVolatile*/false,
|
|
|
|
/*AlwaysInline=*/true,
|
2015-04-14 01:16:45 +08:00
|
|
|
/*isTailCall=*/false,
|
2012-11-22 01:23:03 +08:00
|
|
|
MachinePointerInfo(),
|
|
|
|
MachinePointerInfo());
|
|
|
|
} else {
|
[SelectionDAG] Get rid of bool parameters in SelectionDAG::getLoad, getStore, and friends.
Summary:
Instead, we take a single flags arg (a bitset).
Also add a default 0 alignment, and change the order of arguments so the
alignment comes before the flags.
This greatly simplifies many callsites, and fixes a bug in
AMDGPUISelLowering, wherein the order of the args to getLoad was
inverted. It also greatly simplifies the process of adding another flag
to getLoad.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, jyknight, dsanders, nemanjai, llvm-commits
Differential Revision: http://reviews.llvm.org/D22249
llvm-svn: 275592
2016-07-16 02:27:10 +08:00
|
|
|
MemOp = DAG.getStore(Chain, dl, Arg, PtrOff, MachinePointerInfo());
|
2012-11-22 01:23:03 +08:00
|
|
|
}
|
2009-05-03 21:07:31 +08:00
|
|
|
|
2012-11-22 01:23:03 +08:00
|
|
|
MemOpChains.push_back(MemOp);
|
2009-05-03 21:07:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Transform all store nodes into one single node because all store nodes are
|
|
|
|
// independent of each other.
|
|
|
|
if (!MemOpChains.empty())
|
2014-04-27 02:35:24 +08:00
|
|
|
Chain = DAG.getNode(ISD::TokenFactor, dl, MVT::Other, MemOpChains);
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
// Build a sequence of copy-to-reg nodes chained together with token chain and
|
|
|
|
// flag operands which copy the outgoing args into registers. The InFlag in
|
2011-04-15 13:18:47 +08:00
|
|
|
// necessary since all emitted instructions must be stuck together.
|
2009-05-03 21:07:31 +08:00
|
|
|
SDValue InFlag;
|
|
|
|
for (unsigned i = 0, e = RegsToPass.size(); i != e; ++i) {
|
|
|
|
Chain = DAG.getCopyToReg(Chain, dl, RegsToPass[i].first,
|
|
|
|
RegsToPass[i].second, InFlag);
|
|
|
|
InFlag = Chain.getValue(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the callee is a GlobalAddress node (quite common, every direct call is)
|
|
|
|
// turn it into a TargetGlobalAddress node so that legalize doesn't hack it.
|
|
|
|
// Likewise ExternalSymbol -> TargetExternalSymbol.
|
|
|
|
if (GlobalAddressSDNode *G = dyn_cast<GlobalAddressSDNode>(Callee))
|
2010-07-07 06:08:15 +08:00
|
|
|
Callee = DAG.getTargetGlobalAddress(G->getGlobal(), dl, MVT::i16);
|
2009-05-03 21:07:31 +08:00
|
|
|
else if (ExternalSymbolSDNode *E = dyn_cast<ExternalSymbolSDNode>(Callee))
|
2009-08-12 04:47:22 +08:00
|
|
|
Callee = DAG.getTargetExternalSymbol(E->getSymbol(), MVT::i16);
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
// Returns a chain & a flag for retval copy to use.
|
2010-12-21 10:38:05 +08:00
|
|
|
SDVTList NodeTys = DAG.getVTList(MVT::Other, MVT::Glue);
|
2009-05-03 21:07:31 +08:00
|
|
|
SmallVector<SDValue, 8> Ops;
|
|
|
|
Ops.push_back(Chain);
|
|
|
|
Ops.push_back(Callee);
|
|
|
|
|
|
|
|
// Add argument registers to the end of the list so that they are
|
|
|
|
// known live into the call.
|
|
|
|
for (unsigned i = 0, e = RegsToPass.size(); i != e; ++i)
|
|
|
|
Ops.push_back(DAG.getRegister(RegsToPass[i].first,
|
|
|
|
RegsToPass[i].second.getValueType()));
|
|
|
|
|
|
|
|
if (InFlag.getNode())
|
|
|
|
Ops.push_back(InFlag);
|
|
|
|
|
2014-04-27 02:35:24 +08:00
|
|
|
Chain = DAG.getNode(MSP430ISD::CALL, dl, NodeTys, Ops);
|
2009-05-03 21:07:31 +08:00
|
|
|
InFlag = Chain.getValue(1);
|
|
|
|
|
|
|
|
// Create the CALLSEQ_END node.
|
2015-07-09 10:09:04 +08:00
|
|
|
Chain = DAG.getCALLSEQ_END(Chain, DAG.getConstant(NumBytes, dl, PtrVT, true),
|
|
|
|
DAG.getConstant(0, dl, PtrVT, true), InFlag, dl);
|
2009-05-03 21:07:31 +08:00
|
|
|
InFlag = Chain.getValue(1);
|
|
|
|
|
|
|
|
// Handle result values, copying them out of physregs into vregs that we
|
|
|
|
// return.
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
return LowerCallResult(Chain, InFlag, CallConv, isVarArg, Ins, dl,
|
|
|
|
DAG, InVals);
|
2009-05-03 21:07:31 +08:00
|
|
|
}
|
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
/// LowerCallResult - Lower the result values of a call into the
|
|
|
|
/// appropriate copies out of appropriate physical registers.
|
|
|
|
///
|
2016-06-12 23:39:02 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerCallResult(
|
|
|
|
SDValue Chain, SDValue InFlag, CallingConv::ID CallConv, bool isVarArg,
|
|
|
|
const SmallVectorImpl<ISD::InputArg> &Ins, const SDLoc &dl,
|
|
|
|
SelectionDAG &DAG, SmallVectorImpl<SDValue> &InVals) const {
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
// Assign locations to each value returned by this call.
|
|
|
|
SmallVector<CCValAssign, 16> RVLocs;
|
2014-08-07 02:45:26 +08:00
|
|
|
CCState CCInfo(CallConv, isVarArg, DAG.getMachineFunction(), RVLocs,
|
|
|
|
*DAG.getContext());
|
2009-05-03 21:07:31 +08:00
|
|
|
|
2013-10-15 16:19:39 +08:00
|
|
|
AnalyzeReturnValues(CCInfo, RVLocs, Ins);
|
2009-05-03 21:07:31 +08:00
|
|
|
|
|
|
|
// Copy all of the result registers out of their specified physreg.
|
|
|
|
for (unsigned i = 0; i != RVLocs.size(); ++i) {
|
|
|
|
Chain = DAG.getCopyFromReg(Chain, dl, RVLocs[i].getLocReg(),
|
|
|
|
RVLocs[i].getValVT(), InFlag).getValue(1);
|
|
|
|
InFlag = Chain.getValue(2);
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
InVals.push_back(Chain.getValue(0));
|
2009-05-03 21:07:31 +08:00
|
|
|
}
|
|
|
|
|
Major calling convention code refactoring.
Instead of awkwardly encoding calling-convention information with ISD::CALL,
ISD::FORMAL_ARGUMENTS, ISD::RET, and ISD::ARG_FLAGS nodes, TargetLowering
provides three virtual functions for targets to override:
LowerFormalArguments, LowerCall, and LowerRet, which replace the custom
lowering done on the special nodes. They provide the same information, but
in a more immediately usable format.
This also reworks much of the target-independent tail call logic. The
decision of whether or not to perform a tail call is now cleanly split
between target-independent portions, and the target dependent portion
in IsEligibleForTailCallOptimization.
This also synchronizes all in-tree targets, to help enable future
refactoring and feature work.
llvm-svn: 78142
2009-08-05 09:29:28 +08:00
|
|
|
return Chain;
|
2009-05-03 21:07:31 +08:00
|
|
|
}
|
|
|
|
|
2009-05-03 21:03:33 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerShifts(SDValue Op,
|
2010-04-17 23:26:15 +08:00
|
|
|
SelectionDAG &DAG) const {
|
2009-05-03 21:13:17 +08:00
|
|
|
unsigned Opc = Op.getOpcode();
|
2009-05-03 21:03:33 +08:00
|
|
|
SDNode* N = Op.getNode();
|
2009-08-11 06:56:29 +08:00
|
|
|
EVT VT = Op.getValueType();
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl(N);
|
2009-05-03 21:03:33 +08:00
|
|
|
|
2009-12-13 02:55:37 +08:00
|
|
|
// Expand non-constant shifts to loops:
|
2009-05-03 21:03:33 +08:00
|
|
|
if (!isa<ConstantSDNode>(N->getOperand(1)))
|
2009-12-13 02:55:37 +08:00
|
|
|
switch (Opc) {
|
2012-02-07 10:50:20 +08:00
|
|
|
default: llvm_unreachable("Invalid shift opcode!");
|
2009-12-13 02:55:37 +08:00
|
|
|
case ISD::SHL:
|
|
|
|
return DAG.getNode(MSP430ISD::SHL, dl,
|
|
|
|
VT, N->getOperand(0), N->getOperand(1));
|
|
|
|
case ISD::SRA:
|
|
|
|
return DAG.getNode(MSP430ISD::SRA, dl,
|
|
|
|
VT, N->getOperand(0), N->getOperand(1));
|
|
|
|
case ISD::SRL:
|
|
|
|
return DAG.getNode(MSP430ISD::SRL, dl,
|
|
|
|
VT, N->getOperand(0), N->getOperand(1));
|
|
|
|
}
|
2009-05-03 21:03:33 +08:00
|
|
|
|
|
|
|
uint64_t ShiftAmount = cast<ConstantSDNode>(N->getOperand(1))->getZExtValue();
|
|
|
|
|
|
|
|
// Expand the stuff into sequence of shifts.
|
|
|
|
// FIXME: for some shift amounts this might be done better!
|
|
|
|
// E.g.: foo >> (8 + N) => sxt(swpb(foo)) >> N
|
|
|
|
SDValue Victim = N->getOperand(0);
|
2009-05-03 21:16:17 +08:00
|
|
|
|
|
|
|
if (Opc == ISD::SRL && ShiftAmount) {
|
|
|
|
// Emit a special goodness here:
|
|
|
|
// srl A, 1 => clrc; rrc A
|
2009-05-03 21:16:37 +08:00
|
|
|
Victim = DAG.getNode(MSP430ISD::RRC, dl, VT, Victim);
|
2009-05-03 21:16:17 +08:00
|
|
|
ShiftAmount -= 1;
|
|
|
|
}
|
|
|
|
|
2009-05-03 21:03:33 +08:00
|
|
|
while (ShiftAmount--)
|
2009-05-17 18:15:22 +08:00
|
|
|
Victim = DAG.getNode((Opc == ISD::SHL ? MSP430ISD::RLA : MSP430ISD::RRA),
|
2009-05-03 21:13:17 +08:00
|
|
|
dl, VT, Victim);
|
2009-05-03 21:03:33 +08:00
|
|
|
|
|
|
|
return Victim;
|
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerGlobalAddress(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
2009-05-03 21:08:33 +08:00
|
|
|
const GlobalValue *GV = cast<GlobalAddressSDNode>(Op)->getGlobal();
|
|
|
|
int64_t Offset = cast<GlobalAddressSDNode>(Op)->getOffset();
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
2009-05-03 21:08:33 +08:00
|
|
|
|
|
|
|
// Create the TargetGlobalAddress node, folding in the constant offset.
|
2015-07-09 10:09:04 +08:00
|
|
|
SDValue Result = DAG.getTargetGlobalAddress(GV, SDLoc(Op), PtrVT, Offset);
|
|
|
|
return DAG.getNode(MSP430ISD::Wrapper, SDLoc(Op), PtrVT, Result);
|
2009-05-03 21:08:33 +08:00
|
|
|
}
|
|
|
|
|
2009-05-03 21:14:46 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerExternalSymbol(SDValue Op,
|
2010-04-17 23:26:15 +08:00
|
|
|
SelectionDAG &DAG) const {
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl(Op);
|
2009-05-03 21:14:46 +08:00
|
|
|
const char *Sym = cast<ExternalSymbolSDNode>(Op)->getSymbol();
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
|
|
|
SDValue Result = DAG.getTargetExternalSymbol(Sym, PtrVT);
|
2009-05-03 21:14:46 +08:00
|
|
|
|
2015-07-09 10:09:04 +08:00
|
|
|
return DAG.getNode(MSP430ISD::Wrapper, dl, PtrVT, Result);
|
2009-05-03 21:14:46 +08:00
|
|
|
}
|
|
|
|
|
2010-05-01 20:04:32 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerBlockAddress(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl(Op);
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
2010-05-01 20:04:32 +08:00
|
|
|
const BlockAddress *BA = cast<BlockAddressSDNode>(Op)->getBlockAddress();
|
2015-07-09 10:09:04 +08:00
|
|
|
SDValue Result = DAG.getTargetBlockAddress(BA, PtrVT);
|
2010-05-01 20:04:32 +08:00
|
|
|
|
2015-07-09 10:09:04 +08:00
|
|
|
return DAG.getNode(MSP430ISD::Wrapper, dl, PtrVT, Result);
|
2010-05-01 20:04:32 +08:00
|
|
|
}
|
|
|
|
|
2009-10-22 03:16:49 +08:00
|
|
|
static SDValue EmitCMP(SDValue &LHS, SDValue &RHS, SDValue &TargetCC,
|
2016-06-12 23:39:02 +08:00
|
|
|
ISD::CondCode CC, const SDLoc &dl, SelectionDAG &DAG) {
|
2009-05-03 21:12:06 +08:00
|
|
|
// FIXME: Handle bittests someday
|
|
|
|
assert(!LHS.getValueType().isFloatingPoint() && "We don't handle FP yet");
|
|
|
|
|
|
|
|
// FIXME: Handle jump negative someday
|
2009-10-22 03:16:49 +08:00
|
|
|
MSP430CC::CondCodes TCC = MSP430CC::COND_INVALID;
|
2009-05-03 21:12:06 +08:00
|
|
|
switch (CC) {
|
2009-07-15 00:55:14 +08:00
|
|
|
default: llvm_unreachable("Invalid integer condition!");
|
2009-05-03 21:12:06 +08:00
|
|
|
case ISD::SETEQ:
|
2009-10-22 03:16:49 +08:00
|
|
|
TCC = MSP430CC::COND_E; // aka COND_Z
|
2010-01-15 09:29:49 +08:00
|
|
|
// Minor optimization: if LHS is a constant, swap operands, then the
|
2009-11-22 09:14:08 +08:00
|
|
|
// constant can be folded into comparison.
|
2010-01-15 09:29:49 +08:00
|
|
|
if (LHS.getOpcode() == ISD::Constant)
|
2009-11-22 09:14:08 +08:00
|
|
|
std::swap(LHS, RHS);
|
2009-05-03 21:12:06 +08:00
|
|
|
break;
|
|
|
|
case ISD::SETNE:
|
2009-10-22 03:16:49 +08:00
|
|
|
TCC = MSP430CC::COND_NE; // aka COND_NZ
|
2010-01-15 09:29:49 +08:00
|
|
|
// Minor optimization: if LHS is a constant, swap operands, then the
|
2009-11-22 09:14:08 +08:00
|
|
|
// constant can be folded into comparison.
|
2010-01-15 09:29:49 +08:00
|
|
|
if (LHS.getOpcode() == ISD::Constant)
|
2009-11-22 09:14:08 +08:00
|
|
|
std::swap(LHS, RHS);
|
2009-05-03 21:12:06 +08:00
|
|
|
break;
|
|
|
|
case ISD::SETULE:
|
2016-08-17 13:10:15 +08:00
|
|
|
std::swap(LHS, RHS);
|
|
|
|
LLVM_FALLTHROUGH;
|
2009-05-03 21:12:06 +08:00
|
|
|
case ISD::SETUGE:
|
2010-01-16 05:18:02 +08:00
|
|
|
// Turn lhs u>= rhs with lhs constant into rhs u< lhs+1, this allows us to
|
|
|
|
// fold constant into instruction.
|
|
|
|
if (const ConstantSDNode * C = dyn_cast<ConstantSDNode>(LHS)) {
|
|
|
|
LHS = RHS;
|
2015-04-28 22:05:47 +08:00
|
|
|
RHS = DAG.getConstant(C->getSExtValue() + 1, dl, C->getValueType(0));
|
2010-01-16 05:18:02 +08:00
|
|
|
TCC = MSP430CC::COND_LO;
|
|
|
|
break;
|
|
|
|
}
|
2009-10-22 03:16:49 +08:00
|
|
|
TCC = MSP430CC::COND_HS; // aka COND_C
|
2009-05-03 21:12:06 +08:00
|
|
|
break;
|
|
|
|
case ISD::SETUGT:
|
2016-08-17 13:10:15 +08:00
|
|
|
std::swap(LHS, RHS);
|
|
|
|
LLVM_FALLTHROUGH;
|
2009-05-03 21:12:06 +08:00
|
|
|
case ISD::SETULT:
|
2010-01-16 05:18:02 +08:00
|
|
|
// Turn lhs u< rhs with lhs constant into rhs u>= lhs+1, this allows us to
|
|
|
|
// fold constant into instruction.
|
|
|
|
if (const ConstantSDNode * C = dyn_cast<ConstantSDNode>(LHS)) {
|
|
|
|
LHS = RHS;
|
2015-04-28 22:05:47 +08:00
|
|
|
RHS = DAG.getConstant(C->getSExtValue() + 1, dl, C->getValueType(0));
|
2010-01-16 05:18:02 +08:00
|
|
|
TCC = MSP430CC::COND_HS;
|
|
|
|
break;
|
|
|
|
}
|
2009-10-22 03:16:49 +08:00
|
|
|
TCC = MSP430CC::COND_LO; // aka COND_NC
|
2009-05-03 21:12:06 +08:00
|
|
|
break;
|
|
|
|
case ISD::SETLE:
|
2016-08-17 13:10:15 +08:00
|
|
|
std::swap(LHS, RHS);
|
|
|
|
LLVM_FALLTHROUGH;
|
2009-05-03 21:12:06 +08:00
|
|
|
case ISD::SETGE:
|
2010-01-16 05:18:02 +08:00
|
|
|
// Turn lhs >= rhs with lhs constant into rhs < lhs+1, this allows us to
|
|
|
|
// fold constant into instruction.
|
|
|
|
if (const ConstantSDNode * C = dyn_cast<ConstantSDNode>(LHS)) {
|
|
|
|
LHS = RHS;
|
2015-04-28 22:05:47 +08:00
|
|
|
RHS = DAG.getConstant(C->getSExtValue() + 1, dl, C->getValueType(0));
|
2010-01-16 05:18:02 +08:00
|
|
|
TCC = MSP430CC::COND_L;
|
|
|
|
break;
|
|
|
|
}
|
2009-10-22 03:16:49 +08:00
|
|
|
TCC = MSP430CC::COND_GE;
|
2009-05-03 21:12:06 +08:00
|
|
|
break;
|
|
|
|
case ISD::SETGT:
|
2016-08-17 13:10:15 +08:00
|
|
|
std::swap(LHS, RHS);
|
|
|
|
LLVM_FALLTHROUGH;
|
2009-05-03 21:12:06 +08:00
|
|
|
case ISD::SETLT:
|
2010-01-16 05:18:02 +08:00
|
|
|
// Turn lhs < rhs with lhs constant into rhs >= lhs+1, this allows us to
|
|
|
|
// fold constant into instruction.
|
|
|
|
if (const ConstantSDNode * C = dyn_cast<ConstantSDNode>(LHS)) {
|
|
|
|
LHS = RHS;
|
2015-04-28 22:05:47 +08:00
|
|
|
RHS = DAG.getConstant(C->getSExtValue() + 1, dl, C->getValueType(0));
|
2010-01-16 05:18:02 +08:00
|
|
|
TCC = MSP430CC::COND_GE;
|
|
|
|
break;
|
|
|
|
}
|
2009-10-22 03:16:49 +08:00
|
|
|
TCC = MSP430CC::COND_L;
|
2009-05-03 21:12:06 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-04-28 22:05:47 +08:00
|
|
|
TargetCC = DAG.getConstant(TCC, dl, MVT::i8);
|
2010-12-21 10:38:05 +08:00
|
|
|
return DAG.getNode(MSP430ISD::CMP, dl, MVT::Glue, LHS, RHS);
|
2009-05-03 21:12:06 +08:00
|
|
|
}
|
|
|
|
|
2009-05-03 21:19:09 +08:00
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerBR_CC(SDValue Op, SelectionDAG &DAG) const {
|
2009-05-03 21:12:06 +08:00
|
|
|
SDValue Chain = Op.getOperand(0);
|
2009-05-03 21:19:09 +08:00
|
|
|
ISD::CondCode CC = cast<CondCodeSDNode>(Op.getOperand(1))->get();
|
|
|
|
SDValue LHS = Op.getOperand(2);
|
|
|
|
SDValue RHS = Op.getOperand(3);
|
|
|
|
SDValue Dest = Op.getOperand(4);
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl (Op);
|
2009-05-03 21:19:09 +08:00
|
|
|
|
2009-10-22 03:16:49 +08:00
|
|
|
SDValue TargetCC;
|
2009-05-03 21:19:09 +08:00
|
|
|
SDValue Flag = EmitCMP(LHS, RHS, TargetCC, CC, dl, DAG);
|
|
|
|
|
|
|
|
return DAG.getNode(MSP430ISD::BR_CC, dl, Op.getValueType(),
|
2009-10-22 03:16:49 +08:00
|
|
|
Chain, Dest, TargetCC, Flag);
|
2009-05-03 21:12:06 +08:00
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerSETCC(SDValue Op, SelectionDAG &DAG) const {
|
2009-12-12 07:01:29 +08:00
|
|
|
SDValue LHS = Op.getOperand(0);
|
|
|
|
SDValue RHS = Op.getOperand(1);
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl (Op);
|
2009-12-12 07:01:29 +08:00
|
|
|
|
|
|
|
// If we are doing an AND and testing against zero, then the CMP
|
|
|
|
// will not be generated. The AND (or BIT) will generate the condition codes,
|
|
|
|
// but they are different from CMP.
|
2010-01-16 05:18:18 +08:00
|
|
|
// FIXME: since we're doing a post-processing, use a pseudoinstr here, so
|
|
|
|
// lowering & isel wouldn't diverge.
|
2009-12-12 07:01:29 +08:00
|
|
|
bool andCC = false;
|
|
|
|
if (ConstantSDNode *RHSC = dyn_cast<ConstantSDNode>(RHS)) {
|
|
|
|
if (RHSC->isNullValue() && LHS.hasOneUse() &&
|
|
|
|
(LHS.getOpcode() == ISD::AND ||
|
|
|
|
(LHS.getOpcode() == ISD::TRUNCATE &&
|
|
|
|
LHS.getOperand(0).getOpcode() == ISD::AND))) {
|
|
|
|
andCC = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ISD::CondCode CC = cast<CondCodeSDNode>(Op.getOperand(2))->get();
|
|
|
|
SDValue TargetCC;
|
|
|
|
SDValue Flag = EmitCMP(LHS, RHS, TargetCC, CC, dl, DAG);
|
|
|
|
|
|
|
|
// Get the condition codes directly from the status register, if its easy.
|
|
|
|
// Otherwise a branch will be generated. Note that the AND and BIT
|
|
|
|
// instructions generate different flags than CMP, the carry bit can be used
|
|
|
|
// for NE/EQ.
|
|
|
|
bool Invert = false;
|
|
|
|
bool Shift = false;
|
|
|
|
bool Convert = true;
|
|
|
|
switch (cast<ConstantSDNode>(TargetCC)->getZExtValue()) {
|
|
|
|
default:
|
|
|
|
Convert = false;
|
|
|
|
break;
|
|
|
|
case MSP430CC::COND_HS:
|
2014-09-10 14:58:14 +08:00
|
|
|
// Res = SR & 1, no processing is required
|
2009-12-12 07:01:29 +08:00
|
|
|
break;
|
2010-01-16 05:18:18 +08:00
|
|
|
case MSP430CC::COND_LO:
|
2014-09-10 14:58:14 +08:00
|
|
|
// Res = ~(SR & 1)
|
2009-12-12 07:01:29 +08:00
|
|
|
Invert = true;
|
|
|
|
break;
|
2010-01-16 05:18:18 +08:00
|
|
|
case MSP430CC::COND_NE:
|
2009-12-12 07:01:29 +08:00
|
|
|
if (andCC) {
|
2014-09-10 14:58:14 +08:00
|
|
|
// C = ~Z, thus Res = SR & 1, no processing is required
|
2009-12-12 07:01:29 +08:00
|
|
|
} else {
|
2014-09-10 14:58:14 +08:00
|
|
|
// Res = ~((SR >> 1) & 1)
|
2009-12-12 07:01:29 +08:00
|
|
|
Shift = true;
|
2010-02-21 20:28:58 +08:00
|
|
|
Invert = true;
|
2009-12-12 07:01:29 +08:00
|
|
|
}
|
|
|
|
break;
|
2010-01-16 05:18:18 +08:00
|
|
|
case MSP430CC::COND_E:
|
2010-02-21 20:28:58 +08:00
|
|
|
Shift = true;
|
2014-09-10 14:58:14 +08:00
|
|
|
// C = ~Z for AND instruction, thus we can put Res = ~(SR & 1), however,
|
|
|
|
// Res = (SR >> 1) & 1 is 1 word shorter.
|
2009-12-12 07:01:29 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
EVT VT = Op.getValueType();
|
2015-04-28 22:05:47 +08:00
|
|
|
SDValue One = DAG.getConstant(1, dl, VT);
|
2009-12-12 07:01:29 +08:00
|
|
|
if (Convert) {
|
2014-09-10 14:58:14 +08:00
|
|
|
SDValue SR = DAG.getCopyFromReg(DAG.getEntryNode(), dl, MSP430::SR,
|
2010-01-16 05:18:18 +08:00
|
|
|
MVT::i16, Flag);
|
2009-12-12 07:01:29 +08:00
|
|
|
if (Shift)
|
|
|
|
// FIXME: somewhere this is turned into a SRL, lower it MSP specific?
|
|
|
|
SR = DAG.getNode(ISD::SRA, dl, MVT::i16, SR, One);
|
|
|
|
SR = DAG.getNode(ISD::AND, dl, MVT::i16, SR, One);
|
|
|
|
if (Invert)
|
|
|
|
SR = DAG.getNode(ISD::XOR, dl, MVT::i16, SR, One);
|
|
|
|
return SR;
|
|
|
|
} else {
|
2015-04-28 22:05:47 +08:00
|
|
|
SDValue Zero = DAG.getConstant(0, dl, VT);
|
2010-12-21 10:38:05 +08:00
|
|
|
SDVTList VTs = DAG.getVTList(Op.getValueType(), MVT::Glue);
|
2015-02-19 23:26:17 +08:00
|
|
|
SDValue Ops[] = {One, Zero, TargetCC, Flag};
|
2014-04-27 02:35:24 +08:00
|
|
|
return DAG.getNode(MSP430ISD::SELECT_CC, dl, VTs, Ops);
|
2009-12-12 07:01:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerSELECT_CC(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
2009-05-03 21:19:09 +08:00
|
|
|
SDValue LHS = Op.getOperand(0);
|
|
|
|
SDValue RHS = Op.getOperand(1);
|
|
|
|
SDValue TrueV = Op.getOperand(2);
|
|
|
|
SDValue FalseV = Op.getOperand(3);
|
|
|
|
ISD::CondCode CC = cast<CondCodeSDNode>(Op.getOperand(4))->get();
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl (Op);
|
2009-05-03 21:19:09 +08:00
|
|
|
|
2009-10-22 03:16:49 +08:00
|
|
|
SDValue TargetCC;
|
2009-05-03 21:19:09 +08:00
|
|
|
SDValue Flag = EmitCMP(LHS, RHS, TargetCC, CC, dl, DAG);
|
2009-05-03 21:12:23 +08:00
|
|
|
|
2010-12-21 10:38:05 +08:00
|
|
|
SDVTList VTs = DAG.getVTList(Op.getValueType(), MVT::Glue);
|
2015-02-19 23:26:17 +08:00
|
|
|
SDValue Ops[] = {TrueV, FalseV, TargetCC, Flag};
|
2009-05-03 21:12:23 +08:00
|
|
|
|
2014-04-27 02:35:24 +08:00
|
|
|
return DAG.getNode(MSP430ISD::SELECT_CC, dl, VTs, Ops);
|
2009-05-03 21:12:23 +08:00
|
|
|
}
|
|
|
|
|
2009-05-03 21:17:49 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerSIGN_EXTEND(SDValue Op,
|
2010-04-17 23:26:15 +08:00
|
|
|
SelectionDAG &DAG) const {
|
2009-05-03 21:17:49 +08:00
|
|
|
SDValue Val = Op.getOperand(0);
|
2009-08-11 06:56:29 +08:00
|
|
|
EVT VT = Op.getValueType();
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl(Op);
|
2009-05-03 21:17:49 +08:00
|
|
|
|
2009-08-12 04:47:22 +08:00
|
|
|
assert(VT == MVT::i16 && "Only support i16 for now!");
|
2009-05-03 21:17:49 +08:00
|
|
|
|
|
|
|
return DAG.getNode(ISD::SIGN_EXTEND_INREG, dl, VT,
|
|
|
|
DAG.getNode(ISD::ANY_EXTEND, dl, VT, Val),
|
|
|
|
DAG.getValueType(Val.getValueType()));
|
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue
|
|
|
|
MSP430TargetLowering::getReturnAddressFrameIndex(SelectionDAG &DAG) const {
|
2009-12-07 10:28:10 +08:00
|
|
|
MachineFunction &MF = DAG.getMachineFunction();
|
|
|
|
MSP430MachineFunctionInfo *FuncInfo = MF.getInfo<MSP430MachineFunctionInfo>();
|
|
|
|
int ReturnAddrIndex = FuncInfo->getRAIndex();
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(MF.getDataLayout());
|
2009-12-07 10:28:10 +08:00
|
|
|
|
|
|
|
if (ReturnAddrIndex == 0) {
|
|
|
|
// Set up a frame object for the return address.
|
2015-07-09 10:09:04 +08:00
|
|
|
uint64_t SlotSize = MF.getDataLayout().getPointerSize();
|
2016-07-29 02:40:00 +08:00
|
|
|
ReturnAddrIndex = MF.getFrameInfo().CreateFixedObject(SlotSize, -SlotSize,
|
2010-07-03 08:40:23 +08:00
|
|
|
true);
|
2009-12-07 10:28:10 +08:00
|
|
|
FuncInfo->setRAIndex(ReturnAddrIndex);
|
|
|
|
}
|
|
|
|
|
2015-07-09 10:09:04 +08:00
|
|
|
return DAG.getFrameIndex(ReturnAddrIndex, PtrVT);
|
2009-12-07 10:28:10 +08:00
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerRETURNADDR(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
2016-07-29 02:40:00 +08:00
|
|
|
MachineFrameInfo &MFI = DAG.getMachineFunction().getFrameInfo();
|
|
|
|
MFI.setReturnAddressIsTaken(true);
|
2010-05-22 09:47:14 +08:00
|
|
|
|
2014-01-06 08:43:20 +08:00
|
|
|
if (verifyReturnAddressArgumentIsConstant(Op, DAG))
|
2014-01-05 09:47:20 +08:00
|
|
|
return SDValue();
|
|
|
|
|
2009-12-07 10:28:10 +08:00
|
|
|
unsigned Depth = cast<ConstantSDNode>(Op.getOperand(0))->getZExtValue();
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl(Op);
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
2009-12-07 10:28:10 +08:00
|
|
|
|
|
|
|
if (Depth > 0) {
|
|
|
|
SDValue FrameAddr = LowerFRAMEADDR(Op, DAG);
|
|
|
|
SDValue Offset =
|
2015-07-09 10:09:04 +08:00
|
|
|
DAG.getConstant(DAG.getDataLayout().getPointerSize(), dl, MVT::i16);
|
|
|
|
return DAG.getLoad(PtrVT, dl, DAG.getEntryNode(),
|
|
|
|
DAG.getNode(ISD::ADD, dl, PtrVT, FrameAddr, Offset),
|
[SelectionDAG] Get rid of bool parameters in SelectionDAG::getLoad, getStore, and friends.
Summary:
Instead, we take a single flags arg (a bitset).
Also add a default 0 alignment, and change the order of arguments so the
alignment comes before the flags.
This greatly simplifies many callsites, and fixes a bug in
AMDGPUISelLowering, wherein the order of the args to getLoad was
inverted. It also greatly simplifies the process of adding another flag
to getLoad.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, jyknight, dsanders, nemanjai, llvm-commits
Differential Revision: http://reviews.llvm.org/D22249
llvm-svn: 275592
2016-07-16 02:27:10 +08:00
|
|
|
MachinePointerInfo());
|
2009-12-07 10:28:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Just load the return address.
|
|
|
|
SDValue RetAddrFI = getReturnAddressFrameIndex(DAG);
|
2015-07-09 10:09:04 +08:00
|
|
|
return DAG.getLoad(PtrVT, dl, DAG.getEntryNode(), RetAddrFI,
|
[SelectionDAG] Get rid of bool parameters in SelectionDAG::getLoad, getStore, and friends.
Summary:
Instead, we take a single flags arg (a bitset).
Also add a default 0 alignment, and change the order of arguments so the
alignment comes before the flags.
This greatly simplifies many callsites, and fixes a bug in
AMDGPUISelLowering, wherein the order of the args to getLoad was
inverted. It also greatly simplifies the process of adding another flag
to getLoad.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, jyknight, dsanders, nemanjai, llvm-commits
Differential Revision: http://reviews.llvm.org/D22249
llvm-svn: 275592
2016-07-16 02:27:10 +08:00
|
|
|
MachinePointerInfo());
|
2009-12-07 10:28:10 +08:00
|
|
|
}
|
|
|
|
|
2010-04-17 23:26:15 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerFRAMEADDR(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
2016-07-29 02:40:00 +08:00
|
|
|
MachineFrameInfo &MFI = DAG.getMachineFunction().getFrameInfo();
|
|
|
|
MFI.setFrameAddressIsTaken(true);
|
2010-05-22 09:47:14 +08:00
|
|
|
|
2009-12-07 10:28:10 +08:00
|
|
|
EVT VT = Op.getValueType();
|
2013-05-25 10:42:55 +08:00
|
|
|
SDLoc dl(Op); // FIXME probably not meaningful
|
2009-12-07 10:28:10 +08:00
|
|
|
unsigned Depth = cast<ConstantSDNode>(Op.getOperand(0))->getZExtValue();
|
|
|
|
SDValue FrameAddr = DAG.getCopyFromReg(DAG.getEntryNode(), dl,
|
2014-09-10 14:58:14 +08:00
|
|
|
MSP430::FP, VT);
|
2009-12-07 10:28:10 +08:00
|
|
|
while (Depth--)
|
2010-09-21 14:44:06 +08:00
|
|
|
FrameAddr = DAG.getLoad(VT, dl, DAG.getEntryNode(), FrameAddr,
|
[SelectionDAG] Get rid of bool parameters in SelectionDAG::getLoad, getStore, and friends.
Summary:
Instead, we take a single flags arg (a bitset).
Also add a default 0 alignment, and change the order of arguments so the
alignment comes before the flags.
This greatly simplifies many callsites, and fixes a bug in
AMDGPUISelLowering, wherein the order of the args to getLoad was
inverted. It also greatly simplifies the process of adding another flag
to getLoad.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, jyknight, dsanders, nemanjai, llvm-commits
Differential Revision: http://reviews.llvm.org/D22249
llvm-svn: 275592
2016-07-16 02:27:10 +08:00
|
|
|
MachinePointerInfo());
|
2009-12-07 10:28:10 +08:00
|
|
|
return FrameAddr;
|
|
|
|
}
|
|
|
|
|
2012-11-22 01:28:27 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerVASTART(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
|
|
|
MachineFunction &MF = DAG.getMachineFunction();
|
|
|
|
MSP430MachineFunctionInfo *FuncInfo = MF.getInfo<MSP430MachineFunctionInfo>();
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
2012-11-22 01:28:27 +08:00
|
|
|
|
|
|
|
// Frame index of first vararg argument
|
2015-07-09 10:09:04 +08:00
|
|
|
SDValue FrameIndex =
|
|
|
|
DAG.getFrameIndex(FuncInfo->getVarArgsFrameIndex(), PtrVT);
|
2012-11-22 01:28:27 +08:00
|
|
|
const Value *SV = cast<SrcValueSDNode>(Op.getOperand(2))->getValue();
|
|
|
|
|
|
|
|
// Create a store of the frame index to the location operand
|
[SelectionDAG] Get rid of bool parameters in SelectionDAG::getLoad, getStore, and friends.
Summary:
Instead, we take a single flags arg (a bitset).
Also add a default 0 alignment, and change the order of arguments so the
alignment comes before the flags.
This greatly simplifies many callsites, and fixes a bug in
AMDGPUISelLowering, wherein the order of the args to getLoad was
inverted. It also greatly simplifies the process of adding another flag
to getLoad.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, jyknight, dsanders, nemanjai, llvm-commits
Differential Revision: http://reviews.llvm.org/D22249
llvm-svn: 275592
2016-07-16 02:27:10 +08:00
|
|
|
return DAG.getStore(Op.getOperand(0), SDLoc(Op), FrameIndex, Op.getOperand(1),
|
|
|
|
MachinePointerInfo(SV));
|
2012-11-22 01:28:27 +08:00
|
|
|
}
|
|
|
|
|
2013-07-02 03:44:44 +08:00
|
|
|
SDValue MSP430TargetLowering::LowerJumpTable(SDValue Op,
|
|
|
|
SelectionDAG &DAG) const {
|
|
|
|
JumpTableSDNode *JT = cast<JumpTableSDNode>(Op);
|
2015-07-09 10:09:04 +08:00
|
|
|
auto PtrVT = getPointerTy(DAG.getDataLayout());
|
|
|
|
SDValue Result = DAG.getTargetJumpTable(JT->getIndex(), PtrVT);
|
|
|
|
return DAG.getNode(MSP430ISD::Wrapper, SDLoc(JT), PtrVT, Result);
|
2013-07-02 03:44:44 +08:00
|
|
|
}
|
|
|
|
|
2009-11-08 01:15:06 +08:00
|
|
|
/// getPostIndexedAddressParts - returns true by value, base pointer and
|
|
|
|
/// offset pointer and addressing mode by reference if this node can be
|
|
|
|
/// combined with a load / store to form a post-indexed load / store.
|
|
|
|
bool MSP430TargetLowering::getPostIndexedAddressParts(SDNode *N, SDNode *Op,
|
|
|
|
SDValue &Base,
|
|
|
|
SDValue &Offset,
|
|
|
|
ISD::MemIndexedMode &AM,
|
|
|
|
SelectionDAG &DAG) const {
|
|
|
|
|
|
|
|
LoadSDNode *LD = cast<LoadSDNode>(N);
|
|
|
|
if (LD->getExtensionType() != ISD::NON_EXTLOAD)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
EVT VT = LD->getMemoryVT();
|
|
|
|
if (VT != MVT::i8 && VT != MVT::i16)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (Op->getOpcode() != ISD::ADD)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (ConstantSDNode *RHS = dyn_cast<ConstantSDNode>(Op->getOperand(1))) {
|
|
|
|
uint64_t RHSC = RHS->getZExtValue();
|
|
|
|
if ((VT == MVT::i16 && RHSC != 2) ||
|
|
|
|
(VT == MVT::i8 && RHSC != 1))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Base = Op->getOperand(0);
|
2015-04-28 22:05:47 +08:00
|
|
|
Offset = DAG.getConstant(RHSC, SDLoc(N), VT);
|
2009-11-08 01:15:06 +08:00
|
|
|
AM = ISD::POST_INC;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-03 20:59:50 +08:00
|
|
|
const char *MSP430TargetLowering::getTargetNodeName(unsigned Opcode) const {
|
2015-05-08 05:33:59 +08:00
|
|
|
switch ((MSP430ISD::NodeType)Opcode) {
|
|
|
|
case MSP430ISD::FIRST_NUMBER: break;
|
2009-05-03 20:59:50 +08:00
|
|
|
case MSP430ISD::RET_FLAG: return "MSP430ISD::RET_FLAG";
|
2009-12-07 10:28:41 +08:00
|
|
|
case MSP430ISD::RETI_FLAG: return "MSP430ISD::RETI_FLAG";
|
2009-05-03 21:03:33 +08:00
|
|
|
case MSP430ISD::RRA: return "MSP430ISD::RRA";
|
2009-05-03 21:16:17 +08:00
|
|
|
case MSP430ISD::RLA: return "MSP430ISD::RLA";
|
|
|
|
case MSP430ISD::RRC: return "MSP430ISD::RRC";
|
2009-05-03 21:07:54 +08:00
|
|
|
case MSP430ISD::CALL: return "MSP430ISD::CALL";
|
2009-05-03 21:08:33 +08:00
|
|
|
case MSP430ISD::Wrapper: return "MSP430ISD::Wrapper";
|
2009-05-03 21:19:09 +08:00
|
|
|
case MSP430ISD::BR_CC: return "MSP430ISD::BR_CC";
|
2009-05-03 21:12:06 +08:00
|
|
|
case MSP430ISD::CMP: return "MSP430ISD::CMP";
|
2015-05-08 05:33:59 +08:00
|
|
|
case MSP430ISD::SETCC: return "MSP430ISD::SETCC";
|
2009-05-03 21:19:09 +08:00
|
|
|
case MSP430ISD::SELECT_CC: return "MSP430ISD::SELECT_CC";
|
2009-12-13 02:55:37 +08:00
|
|
|
case MSP430ISD::SHL: return "MSP430ISD::SHL";
|
|
|
|
case MSP430ISD::SRA: return "MSP430ISD::SRA";
|
2015-05-08 05:33:59 +08:00
|
|
|
case MSP430ISD::SRL: return "MSP430ISD::SRL";
|
2009-05-03 20:59:50 +08:00
|
|
|
}
|
2015-05-08 05:33:59 +08:00
|
|
|
return nullptr;
|
2009-05-03 20:59:50 +08:00
|
|
|
}
|
2009-05-03 21:12:23 +08:00
|
|
|
|
2011-07-18 12:54:35 +08:00
|
|
|
bool MSP430TargetLowering::isTruncateFree(Type *Ty1,
|
|
|
|
Type *Ty2) const {
|
2010-02-16 00:12:20 +08:00
|
|
|
if (!Ty1->isIntegerTy() || !Ty2->isIntegerTy())
|
2010-01-16 05:19:43 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return (Ty1->getPrimitiveSizeInBits() > Ty2->getPrimitiveSizeInBits());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MSP430TargetLowering::isTruncateFree(EVT VT1, EVT VT2) const {
|
|
|
|
if (!VT1.isInteger() || !VT2.isInteger())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return (VT1.getSizeInBits() > VT2.getSizeInBits());
|
|
|
|
}
|
|
|
|
|
2011-07-18 12:54:35 +08:00
|
|
|
bool MSP430TargetLowering::isZExtFree(Type *Ty1, Type *Ty2) const {
|
2010-01-16 05:19:43 +08:00
|
|
|
// MSP430 implicitly zero-extends 8-bit results in 16-bit registers.
|
2010-02-16 00:12:20 +08:00
|
|
|
return 0 && Ty1->isIntegerTy(8) && Ty2->isIntegerTy(16);
|
2010-01-16 05:19:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool MSP430TargetLowering::isZExtFree(EVT VT1, EVT VT2) const {
|
|
|
|
// MSP430 implicitly zero-extends 8-bit results in 16-bit registers.
|
|
|
|
return 0 && VT1 == MVT::i8 && VT2 == MVT::i16;
|
|
|
|
}
|
|
|
|
|
2012-12-19 02:21:29 +08:00
|
|
|
bool MSP430TargetLowering::isZExtFree(SDValue Val, EVT VT2) const {
|
|
|
|
return isZExtFree(Val.getValueType(), VT2);
|
|
|
|
}
|
|
|
|
|
2009-05-03 21:12:23 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Other Lowering Code
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2016-07-01 06:52:52 +08:00
|
|
|
MachineBasicBlock *
|
|
|
|
MSP430TargetLowering::EmitShiftInstr(MachineInstr &MI,
|
2010-05-01 08:01:06 +08:00
|
|
|
MachineBasicBlock *BB) const {
|
2009-12-13 02:55:37 +08:00
|
|
|
MachineFunction *F = BB->getParent();
|
|
|
|
MachineRegisterInfo &RI = F->getRegInfo();
|
2016-07-01 06:52:52 +08:00
|
|
|
DebugLoc dl = MI.getDebugLoc();
|
2015-01-30 07:46:42 +08:00
|
|
|
const TargetInstrInfo &TII = *F->getSubtarget().getInstrInfo();
|
2009-12-13 02:55:37 +08:00
|
|
|
|
|
|
|
unsigned Opc;
|
|
|
|
const TargetRegisterClass * RC;
|
2016-07-01 06:52:52 +08:00
|
|
|
switch (MI.getOpcode()) {
|
2012-02-07 10:50:20 +08:00
|
|
|
default: llvm_unreachable("Invalid shift opcode!");
|
2009-12-13 02:55:37 +08:00
|
|
|
case MSP430::Shl8:
|
|
|
|
Opc = MSP430::SHL8r1;
|
2012-04-20 15:30:17 +08:00
|
|
|
RC = &MSP430::GR8RegClass;
|
2009-12-13 02:55:37 +08:00
|
|
|
break;
|
|
|
|
case MSP430::Shl16:
|
|
|
|
Opc = MSP430::SHL16r1;
|
2012-04-20 15:30:17 +08:00
|
|
|
RC = &MSP430::GR16RegClass;
|
2009-12-13 02:55:37 +08:00
|
|
|
break;
|
|
|
|
case MSP430::Sra8:
|
|
|
|
Opc = MSP430::SAR8r1;
|
2012-04-20 15:30:17 +08:00
|
|
|
RC = &MSP430::GR8RegClass;
|
2009-12-13 02:55:37 +08:00
|
|
|
break;
|
|
|
|
case MSP430::Sra16:
|
|
|
|
Opc = MSP430::SAR16r1;
|
2012-04-20 15:30:17 +08:00
|
|
|
RC = &MSP430::GR16RegClass;
|
2009-12-13 02:55:37 +08:00
|
|
|
break;
|
|
|
|
case MSP430::Srl8:
|
|
|
|
Opc = MSP430::SAR8r1c;
|
2012-04-20 15:30:17 +08:00
|
|
|
RC = &MSP430::GR8RegClass;
|
2009-12-13 02:55:37 +08:00
|
|
|
break;
|
|
|
|
case MSP430::Srl16:
|
|
|
|
Opc = MSP430::SAR16r1c;
|
2012-04-20 15:30:17 +08:00
|
|
|
RC = &MSP430::GR16RegClass;
|
2009-12-13 02:55:37 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const BasicBlock *LLVM_BB = BB->getBasicBlock();
|
2015-10-20 09:18:39 +08:00
|
|
|
MachineFunction::iterator I = ++BB->getIterator();
|
2009-12-13 02:55:37 +08:00
|
|
|
|
|
|
|
// Create loop block
|
|
|
|
MachineBasicBlock *LoopBB = F->CreateMachineBasicBlock(LLVM_BB);
|
|
|
|
MachineBasicBlock *RemBB = F->CreateMachineBasicBlock(LLVM_BB);
|
|
|
|
|
|
|
|
F->insert(I, LoopBB);
|
|
|
|
F->insert(I, RemBB);
|
|
|
|
|
|
|
|
// Update machine-CFG edges by transferring all successors of the current
|
|
|
|
// block to the block containing instructions after shift.
|
2014-03-02 20:27:27 +08:00
|
|
|
RemBB->splice(RemBB->begin(), BB, std::next(MachineBasicBlock::iterator(MI)),
|
2010-07-07 04:24:04 +08:00
|
|
|
BB->end());
|
|
|
|
RemBB->transferSuccessorsAndUpdatePHIs(BB);
|
2009-12-13 02:55:37 +08:00
|
|
|
|
2017-02-28 16:27:43 +08:00
|
|
|
// Add edges BB => LoopBB => RemBB, BB => RemBB, LoopBB => LoopBB
|
2009-12-13 02:55:37 +08:00
|
|
|
BB->addSuccessor(LoopBB);
|
|
|
|
BB->addSuccessor(RemBB);
|
|
|
|
LoopBB->addSuccessor(RemBB);
|
|
|
|
LoopBB->addSuccessor(LoopBB);
|
|
|
|
|
2012-04-20 15:30:17 +08:00
|
|
|
unsigned ShiftAmtReg = RI.createVirtualRegister(&MSP430::GR8RegClass);
|
|
|
|
unsigned ShiftAmtReg2 = RI.createVirtualRegister(&MSP430::GR8RegClass);
|
2009-12-13 02:55:37 +08:00
|
|
|
unsigned ShiftReg = RI.createVirtualRegister(RC);
|
|
|
|
unsigned ShiftReg2 = RI.createVirtualRegister(RC);
|
2016-07-01 06:52:52 +08:00
|
|
|
unsigned ShiftAmtSrcReg = MI.getOperand(2).getReg();
|
|
|
|
unsigned SrcReg = MI.getOperand(1).getReg();
|
|
|
|
unsigned DstReg = MI.getOperand(0).getReg();
|
2009-12-13 02:55:37 +08:00
|
|
|
|
|
|
|
// BB:
|
|
|
|
// cmp 0, N
|
|
|
|
// je RemBB
|
2010-01-15 09:29:49 +08:00
|
|
|
BuildMI(BB, dl, TII.get(MSP430::CMP8ri))
|
|
|
|
.addReg(ShiftAmtSrcReg).addImm(0);
|
2009-12-13 02:55:37 +08:00
|
|
|
BuildMI(BB, dl, TII.get(MSP430::JCC))
|
|
|
|
.addMBB(RemBB)
|
|
|
|
.addImm(MSP430CC::COND_E);
|
|
|
|
|
|
|
|
// LoopBB:
|
|
|
|
// ShiftReg = phi [%SrcReg, BB], [%ShiftReg2, LoopBB]
|
|
|
|
// ShiftAmt = phi [%N, BB], [%ShiftAmt2, LoopBB]
|
|
|
|
// ShiftReg2 = shift ShiftReg
|
|
|
|
// ShiftAmt2 = ShiftAmt - 1;
|
|
|
|
BuildMI(LoopBB, dl, TII.get(MSP430::PHI), ShiftReg)
|
|
|
|
.addReg(SrcReg).addMBB(BB)
|
|
|
|
.addReg(ShiftReg2).addMBB(LoopBB);
|
|
|
|
BuildMI(LoopBB, dl, TII.get(MSP430::PHI), ShiftAmtReg)
|
|
|
|
.addReg(ShiftAmtSrcReg).addMBB(BB)
|
|
|
|
.addReg(ShiftAmtReg2).addMBB(LoopBB);
|
|
|
|
BuildMI(LoopBB, dl, TII.get(Opc), ShiftReg2)
|
|
|
|
.addReg(ShiftReg);
|
|
|
|
BuildMI(LoopBB, dl, TII.get(MSP430::SUB8ri), ShiftAmtReg2)
|
|
|
|
.addReg(ShiftAmtReg).addImm(1);
|
|
|
|
BuildMI(LoopBB, dl, TII.get(MSP430::JCC))
|
|
|
|
.addMBB(LoopBB)
|
|
|
|
.addImm(MSP430CC::COND_NE);
|
|
|
|
|
|
|
|
// RemBB:
|
|
|
|
// DestReg = phi [%SrcReg, BB], [%ShiftReg, LoopBB]
|
2010-07-07 04:24:04 +08:00
|
|
|
BuildMI(*RemBB, RemBB->begin(), dl, TII.get(MSP430::PHI), DstReg)
|
2009-12-13 02:55:37 +08:00
|
|
|
.addReg(SrcReg).addMBB(BB)
|
|
|
|
.addReg(ShiftReg2).addMBB(LoopBB);
|
|
|
|
|
2016-07-01 06:52:52 +08:00
|
|
|
MI.eraseFromParent(); // The pseudo instruction is gone now.
|
2009-12-13 02:55:37 +08:00
|
|
|
return RemBB;
|
|
|
|
}
|
|
|
|
|
2016-07-01 06:52:52 +08:00
|
|
|
MachineBasicBlock *
|
|
|
|
MSP430TargetLowering::EmitInstrWithCustomInserter(MachineInstr &MI,
|
2010-05-01 08:01:06 +08:00
|
|
|
MachineBasicBlock *BB) const {
|
2016-07-01 06:52:52 +08:00
|
|
|
unsigned Opc = MI.getOpcode();
|
2009-12-13 02:55:37 +08:00
|
|
|
|
|
|
|
if (Opc == MSP430::Shl8 || Opc == MSP430::Shl16 ||
|
|
|
|
Opc == MSP430::Sra8 || Opc == MSP430::Sra16 ||
|
|
|
|
Opc == MSP430::Srl8 || Opc == MSP430::Srl16)
|
2010-05-01 08:01:06 +08:00
|
|
|
return EmitShiftInstr(MI, BB);
|
2009-12-13 02:55:37 +08:00
|
|
|
|
2015-01-30 07:46:42 +08:00
|
|
|
const TargetInstrInfo &TII = *BB->getParent()->getSubtarget().getInstrInfo();
|
2016-07-01 06:52:52 +08:00
|
|
|
DebugLoc dl = MI.getDebugLoc();
|
2009-12-13 02:55:37 +08:00
|
|
|
|
|
|
|
assert((Opc == MSP430::Select16 || Opc == MSP430::Select8) &&
|
2009-05-03 21:12:23 +08:00
|
|
|
"Unexpected instr type to insert");
|
|
|
|
|
|
|
|
// To "insert" a SELECT instruction, we actually have to insert the diamond
|
|
|
|
// control-flow pattern. The incoming instruction knows the destination vreg
|
|
|
|
// to set, the condition code register to branch on, the true/false values to
|
|
|
|
// select between, and a branch opcode to use.
|
|
|
|
const BasicBlock *LLVM_BB = BB->getBasicBlock();
|
2015-10-20 09:18:39 +08:00
|
|
|
MachineFunction::iterator I = ++BB->getIterator();
|
2009-05-03 21:12:23 +08:00
|
|
|
|
|
|
|
// thisMBB:
|
|
|
|
// ...
|
|
|
|
// TrueVal = ...
|
|
|
|
// cmpTY ccX, r1, r2
|
|
|
|
// jCC copy1MBB
|
|
|
|
// fallthrough --> copy0MBB
|
|
|
|
MachineBasicBlock *thisMBB = BB;
|
|
|
|
MachineFunction *F = BB->getParent();
|
|
|
|
MachineBasicBlock *copy0MBB = F->CreateMachineBasicBlock(LLVM_BB);
|
|
|
|
MachineBasicBlock *copy1MBB = F->CreateMachineBasicBlock(LLVM_BB);
|
|
|
|
F->insert(I, copy0MBB);
|
|
|
|
F->insert(I, copy1MBB);
|
|
|
|
// Update machine-CFG edges by transferring all successors of the current
|
|
|
|
// block to the new block which will contain the Phi node for the select.
|
2010-07-07 04:24:04 +08:00
|
|
|
copy1MBB->splice(copy1MBB->begin(), BB,
|
2014-03-02 20:27:27 +08:00
|
|
|
std::next(MachineBasicBlock::iterator(MI)), BB->end());
|
2010-07-07 04:24:04 +08:00
|
|
|
copy1MBB->transferSuccessorsAndUpdatePHIs(BB);
|
2009-05-03 21:12:23 +08:00
|
|
|
// Next, add the true and fallthrough blocks as its successors.
|
|
|
|
BB->addSuccessor(copy0MBB);
|
|
|
|
BB->addSuccessor(copy1MBB);
|
|
|
|
|
2010-07-07 04:24:04 +08:00
|
|
|
BuildMI(BB, dl, TII.get(MSP430::JCC))
|
2016-07-01 06:52:52 +08:00
|
|
|
.addMBB(copy1MBB)
|
|
|
|
.addImm(MI.getOperand(3).getImm());
|
2010-07-07 04:24:04 +08:00
|
|
|
|
2009-05-03 21:12:23 +08:00
|
|
|
// copy0MBB:
|
|
|
|
// %FalseValue = ...
|
|
|
|
// # fallthrough to copy1MBB
|
|
|
|
BB = copy0MBB;
|
|
|
|
|
|
|
|
// Update machine-CFG edges
|
|
|
|
BB->addSuccessor(copy1MBB);
|
|
|
|
|
|
|
|
// copy1MBB:
|
|
|
|
// %Result = phi [ %FalseValue, copy0MBB ], [ %TrueValue, thisMBB ]
|
|
|
|
// ...
|
|
|
|
BB = copy1MBB;
|
2016-07-01 06:52:52 +08:00
|
|
|
BuildMI(*BB, BB->begin(), dl, TII.get(MSP430::PHI), MI.getOperand(0).getReg())
|
|
|
|
.addReg(MI.getOperand(2).getReg())
|
|
|
|
.addMBB(copy0MBB)
|
|
|
|
.addReg(MI.getOperand(1).getReg())
|
|
|
|
.addMBB(thisMBB);
|
2009-05-03 21:12:23 +08:00
|
|
|
|
2016-07-01 06:52:52 +08:00
|
|
|
MI.eraseFromParent(); // The pseudo instruction is gone now.
|
2009-05-03 21:12:23 +08:00
|
|
|
return BB;
|
|
|
|
}
|